Execute an application on Mac OS X when a particular type of USB device is connected?
Asked Answered
O

5

21

I need to implement a Mac OS X application. In my application I need to do two things:

  1. Execute / Open an application when a particular type of USB device is connected to the system.
  2. Read the data from USB and upload it to a web server.

I do not have much experience in Mac OS X development. Can anyone please suggest the best documents to reach my goals?

Ozzy answered 30/8, 2011 at 7:45 Comment(0)
T
1

It really depends on what sort of application you are looking at.

It does look like there is no way to do it in a similar fashion to udev for example.

Two possible solutions are:

  • Write a custom wrapper driver for your device
  • Use libusb and have a daemon to wait for certain device.

And in fact one could write a program with libusb which will handle this sort of tasks according to a given config file, that would be also cross-platform since libusb supports quite a few platforms.

Tolson answered 19/9, 2011 at 4:43 Comment(0)
U
33

You can use launchd. Try man launchd and man launchd.plist.

It seems that launchd can work with USB events, even though this feature is poorly documented. I found it on: man xpc_set_event_stream_handler

Here's an example. If you put the following into: ~/Library/LaunchAgents/com.example.plist, your program should start when a USB device is connected.

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC -//Apple Computer//DTD PLIST 1.0//EN http://www.apple.com/DTDs/PropertyList-1.0.dtd >
<plist version="1.0">
<dict>
    <key>Label</key>
    <string>com.example.program</string>
    <key>ProgramArguments</key>
    <array>
    <string>/path/to/program</string>
    </array>
    <key>LaunchEvents</key>
    <dict>
            <key>com.apple.iokit.matching</key>
            <dict>
                    <key>com.apple.device-attach</key>
                    <dict>
                            <key>idProduct</key>
                            <integer>1234</integer>
                            <key>idVendor</key>
                            <integer>1337</integer>
                            <key>IOProviderClass</key>
                            <string>IOUSBDevice</string>
                            <key>IOMatchLaunchStream</key>
                            <true/>
                    </dict>
            </dict>
    </dict>
</dict>
</plist>
Ultracentrifuge answered 4/9, 2012 at 8:40 Comment(7)
just make sure that if you go this route and do not want your launched process to stay running (i.e. KeepAlive = false) that your code calls xpc_set_event_stream_handler() otherwise you will find your app being respawned every 10 seconds in response to the same matching event over and over again.Issue
Does this work under OS X 10.6? I'm getting "LaunchEvents key not recognized". (See https://mcmap.net/q/659520/-launchevents-key-unrecognized-in-launchd-plist-detecting-usb-device/558639)Suffuse
Is there a way to use this with shell scripts being launched from launchd instead of an Xcode app (eg. /path/to/program is a shell script). The shell script is being relaunched every 10 seconds and I can't find a command line alternative form xpc_set_event_stream_handler() or any way to remove the event from the stream.Impossibility
@Impossibility Found this amazing utility that will call xpc_set_event_stream_handler to remove the event from the queue, and then call your desired application. Allowed me to use launchd instead of an Xcode app. Check out github.com/snosrap/xpc_set_event_stream_handlerRambler
For everyone else that finds this post first and wonders what the field values should be (productID, vendorID, IOProviderClass) and wonders how to fix the script being rerun every 10 seconds, this answer covers it all.Insult
The link from @Insult didn’t work for providing the info I needed with my SteelSeries Headset. However I was able to use system_profiler -xml SPUSBDataType > out.xml to get a list then I just searched for SteelSeries and then converted the product_id and vendor_id from hex to decimal. I use this along with SwitchAudioSource to set my default sound to my headphones whenever they’re plugged in/detectedDesex
Upgraded to an M1 mac and can't seem to get this working any longerDesex
K
4

Depending on the type of device you might able to set an application to open automatically via the iPhoto/Image Capture preferences. That will work only for a limited class of devices, for an application already present on the computer and will require changing the preferences on the computer manually.

In general, there's no way to automatically run arbitrary applications on CD/DVD/USB insert because it's a security problem.

Kitti answered 14/9, 2011 at 19:24 Comment(0)
J
4

Julien Pilet's answer worked for me. However, to get it to not constantly relaunch the app when the device is still connected when closing the app, I had to:

  • call xpc_set_event_stream_handler() in my app delegate applicationDidFinishLaunching:
    xpc_set_event_stream_handler("com.apple.iokit.matching", NULL, ^(xpc_object_t event) {     
        // Every event has the key XPC_EVENT_KEY_NAME set to a string that
        // is the name you gave the event in your launchd.plist.
        const char *name = xpc_dictionary_get_string(event, XPC_EVENT_KEY_NAME);

        // IOKit events have the IORegistryEntryNumber as a payload.
        uint64_t id = xpc_dictionary_get_uint64(event, "IOMatchLaunchServiceID");
        // Reconstruct the node you were interested in here using the IOKit
        // APIs.
        NSLog(@"Received event: %s: %llu",name,id);
    });
  • add KeepAlive/false key/value pair to the plist
  • add IOMatchLaunchStream/true key/value pair to the com.apple.device-attach dict in the plist. This is in addition to the IOMatchStream key already there. Not sure why that has to be there, I found a reference to it here: http://asciiwwdc.com/2013/sessions/702

Also don't forget to register the plist with the system using

launchctl load <path to your plist>

Note that this seems to work, but I never get the NSLog message from the xpc stream handler.

Jamal answered 11/2, 2014 at 20:17 Comment(1)
From documentation: The IOMatchLaunchStream key is required to be present and be a Boolean set to true for use with XPC Events. It will be filtered out of the rest of the dictionary when given to IOKit to match. The reasons for this are historical and not applicable to other event streams.Fai
S
1

You may be able to set Folder Actions to run a command on mount. This would assume that the device always mounts in the same place, i.e. /Volumes/My\ Device/ - if peripherals were added or removed in between mounts, the mount point may change. You can learn more about Folder Actions by right clicking a directory and clicking "Folder Actions Setup". The trick would be to make sure that the device always mounts in the same place.

Alternatively, you may be able to use launchd to run a command on mount. This link may help. Lingon is a great app to edit daemons.

Either way, you could use the Folder Action or daemon to call a simple script to grab the contents of the device and upload them to wherever you please.

Staurolite answered 17/9, 2011 at 5:50 Comment(0)
T
1

It really depends on what sort of application you are looking at.

It does look like there is no way to do it in a similar fashion to udev for example.

Two possible solutions are:

  • Write a custom wrapper driver for your device
  • Use libusb and have a daemon to wait for certain device.

And in fact one could write a program with libusb which will handle this sort of tasks according to a given config file, that would be also cross-platform since libusb supports quite a few platforms.

Tolson answered 19/9, 2011 at 4:43 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.