#include "common.h" #include "cli.h" // TODO: set on fire. cli.{h,c} handle both parsing and defaults, so there's // no need to set those here. also, in order to scope metadata by path, // each stream will need its own configuration... so this won't work as // a global any more. In the end the goal is to make the output format // able to declare not just that something happened and what flags were // attached, but what path it was watching that caused those events (so // that the path itself can be used for routing that information to the // relevant callback). // // Structure for storing metadata parsed from the commandline static struct { FSEventStreamEventId sinceWhen; CFTimeInterval latency; FSEventStreamCreateFlags flags; CFMutableArrayRef paths; enum FSEventWatchOutputFormat format; } config = { (UInt64) kFSEventStreamEventIdSinceNow, (double) 0.3, (CFOptionFlags) kFSEventStreamCreateFlagNone, NULL, kFSEventWatchOutputFormatClassic }; // Prototypes static void append_path(const char* path); static inline void parse_cli_settings(int argc, const char* argv[]); static void callback(FSEventStreamRef streamRef, void* clientCallBackInfo, size_t numEvents, void* eventPaths, const FSEventStreamEventFlags eventFlags[], const FSEventStreamEventId eventIds[]); // Resolve a path and append it to the CLI settings structure // The FSEvents API will, internally, resolve paths using a similar scheme. // Performing this ahead of time makes things less confusing, IMHO. static void append_path(const char* path) { #ifdef DEBUG fprintf(stderr, "\n"); fprintf(stderr, "append_path called for: %s\n", path); #endif #if MAC_OS_X_VERSION_MIN_REQUIRED >= 1060 #ifdef DEBUG fprintf(stderr, "compiled against 10.6+, using CFURLCreateFileReferenceURL\n"); #endif CFURLRef url = CFURLCreateFromFileSystemRepresentation(NULL, (const UInt8*)path, (CFIndex)strlen(path), false); CFURLRef placeholder = CFURLCopyAbsoluteURL(url); CFRelease(url); CFMutableArrayRef imaginary = NULL; // if we don't have an existing url, spin until we get to a parent that // does exist, saving any imaginary components for appending back later while(!CFURLResourceIsReachable(placeholder, NULL)) { #ifdef DEBUG fprintf(stderr, "path does not exist\n"); #endif CFStringRef child; if (imaginary == NULL) { imaginary = CFArrayCreateMutable(NULL, 0, &kCFTypeArrayCallBacks); } child = CFURLCopyLastPathComponent(placeholder); CFArrayInsertValueAtIndex(imaginary, 0, child); CFRelease(child); url = CFURLCreateCopyDeletingLastPathComponent(NULL, placeholder); CFRelease(placeholder); placeholder = url; #ifdef DEBUG fprintf(stderr, "parent: "); CFShow(placeholder); #endif } #ifdef DEBUG fprintf(stderr, "path exists\n"); #endif // realpath() doesn't always return the correct case for a path, so this // is a funky workaround that converts a path into a (volId/inodeId) pair // and asks what the path should be for that. since it looks at the actual // inode instead of returning the same case passed in like realpath() // appears to do for HFS+, it should always be correct. url = CFURLCreateFileReferenceURL(NULL, placeholder, NULL); CFRelease(placeholder); placeholder = CFURLCreateFilePathURL(NULL, url, NULL); CFRelease(url); #ifdef DEBUG fprintf(stderr, "path resolved to: "); CFShow(placeholder); #endif // if we stripped off any imaginary path components, append them back on if (imaginary != NULL) { CFIndex count = CFArrayGetCount(imaginary); for (CFIndex i = 0; i= 6)) { config.flags |= kFSEventStreamCreateFlagIgnoreSelf; } else { fprintf(stderr, "MacOSX 10.6 or later is required for --ignore-self\n"); exit(EXIT_FAILURE); } } if (args_info.file_events_flag) { if ((osMajorVersion == 10) & (osMinorVersion >= 7)) { config.flags |= kFSEventStreamCreateFlagFileEvents; } else { fprintf(stderr, "MacOSX 10.7 or later required for --file-events\n"); exit(EXIT_FAILURE); } } if (args_info.inputs_num == 0) { append_path("."); } else { for (unsigned int i=0; i < args_info.inputs_num; ++i) { append_path(args_info.inputs[i]); } } cli_parser_free(&args_info); #ifdef DEBUG fprintf(stderr, "config.sinceWhen %llu\n", config.sinceWhen); fprintf(stderr, "config.latency %f\n", config.latency); // STFU clang #if __LP64__ fprintf(stderr, "config.flags %#.8x\n", config.flags); #else fprintf(stderr, "config.flags %#.8lx\n", config.flags); #endif FLAG_CHECK_STDERR(config.flags, kFSEventStreamCreateFlagUseCFTypes, " Using CF instead of C types"); FLAG_CHECK_STDERR(config.flags, kFSEventStreamCreateFlagNoDefer, " NoDefer latency modifier enabled"); FLAG_CHECK_STDERR(config.flags, kFSEventStreamCreateFlagWatchRoot, " WatchRoot notifications enabled"); FLAG_CHECK_STDERR(config.flags, kFSEventStreamCreateFlagIgnoreSelf, " IgnoreSelf enabled"); FLAG_CHECK_STDERR(config.flags, kFSEventStreamCreateFlagFileEvents, " FileEvents enabled"); fprintf(stderr, "config.paths\n"); long numpaths = CFArrayGetCount(config.paths); for (long i = 0; i < numpaths; i++) { char path[PATH_MAX]; CFStringGetCString(CFArrayGetValueAtIndex(config.paths, i), path, PATH_MAX, kCFStringEncodingUTF8); fprintf(stderr, " %s\n", path); } fprintf(stderr, "\n"); #endif } // original output format for rb-fsevent static void classic_output_format(size_t numEvents, char** paths) { for (size_t i = 0; i < numEvents; i++) { fprintf(stdout, "%s:", paths[i]); } fprintf(stdout, "\n"); } // output format used in the Yoshimasa Niwa branch of rb-fsevent static void niw_output_format(size_t numEvents, char** paths, const FSEventStreamEventFlags eventFlags[], const FSEventStreamEventId eventIds[]) { for (size_t i = 0; i < numEvents; i++) { fprintf(stdout, "%lu:%llu:%s\n", (unsigned long)eventFlags[i], (unsigned long long)eventIds[i], paths[i]); } fprintf(stdout, "\n"); } static void tstring_output_format(size_t numEvents, char** paths, const FSEventStreamEventFlags eventFlags[], const FSEventStreamEventId eventIds[], TSITStringFormat format) { CFMutableArrayRef events = CFArrayCreateMutable(kCFAllocatorDefault, 0, &kCFTypeArrayCallBacks); for (size_t i = 0; i < numEvents; i++) { CFMutableDictionaryRef event = CFDictionaryCreateMutable(kCFAllocatorDefault, 0, &kCFTypeDictionaryKeyCallBacks, &kCFTypeDictionaryValueCallBacks); CFStringRef path = CFStringCreateWithBytes(kCFAllocatorDefault, (const UInt8*)paths[i], (CFIndex)strlen(paths[i]), kCFStringEncodingUTF8, false); CFDictionarySetValue(event, CFSTR("path"), path); CFNumberRef flags = CFNumberCreate(kCFAllocatorDefault, kCFNumberIntType, &eventFlags[i]); CFDictionarySetValue(event, CFSTR("flags"), flags); CFNumberRef ident = CFNumberCreate(kCFAllocatorDefault, kCFNumberLongLongType, &eventIds[i]); CFDictionarySetValue(event, CFSTR("id"), ident); CFArrayAppendValue(events, event); CFRelease(event); CFRelease(path); CFRelease(flags); CFRelease(ident); } CFMutableDictionaryRef meta = CFDictionaryCreateMutable(kCFAllocatorDefault, 0, &kCFTypeDictionaryKeyCallBacks, &kCFTypeDictionaryValueCallBacks); CFDictionarySetValue(meta, CFSTR("events"), events); CFNumberRef num = CFNumberCreate(kCFAllocatorDefault, kCFNumberCFIndexType, &numEvents); CFDictionarySetValue(meta, CFSTR("numEvents"), num); CFDataRef data = TSICTStringCreateRenderedDataFromObjectWithFormat(meta, format); fprintf(stdout, "%s", CFDataGetBytePtr(data)); CFRelease(events); CFRelease(num); CFRelease(meta); CFRelease(data); } static void callback(__attribute__((unused)) FSEventStreamRef streamRef, __attribute__((unused)) void* clientCallBackInfo, size_t numEvents, void* eventPaths, const FSEventStreamEventFlags eventFlags[], const FSEventStreamEventId eventIds[]) { char** paths = eventPaths; #ifdef DEBUG fprintf(stderr, "\n"); fprintf(stderr, "FSEventStreamCallback fired!\n"); fprintf(stderr, " numEvents: %lu\n", numEvents); for (size_t i = 0; i < numEvents; i++) { fprintf(stderr, "\n"); fprintf(stderr, " event ID: %llu\n", eventIds[i]); // STFU clang #if __LP64__ fprintf(stderr, " event flags: %#.8x\n", eventFlags[i]); #else fprintf(stderr, " event flags: %#.8lx\n", eventFlags[i]); #endif FLAG_CHECK_STDERR(eventFlags[i], kFSEventStreamEventFlagMustScanSubDirs, " Recursive scanning of directory required"); FLAG_CHECK_STDERR(eventFlags[i], kFSEventStreamEventFlagUserDropped, " Buffering problem: events dropped user-side"); FLAG_CHECK_STDERR(eventFlags[i], kFSEventStreamEventFlagKernelDropped, " Buffering problem: events dropped kernel-side"); FLAG_CHECK_STDERR(eventFlags[i], kFSEventStreamEventFlagEventIdsWrapped, " Event IDs have wrapped"); FLAG_CHECK_STDERR(eventFlags[i], kFSEventStreamEventFlagHistoryDone, " All historical events have been processed"); FLAG_CHECK_STDERR(eventFlags[i], kFSEventStreamEventFlagRootChanged, " Root path has changed"); FLAG_CHECK_STDERR(eventFlags[i], kFSEventStreamEventFlagMount, " A new volume was mounted at this path"); FLAG_CHECK_STDERR(eventFlags[i], kFSEventStreamEventFlagUnmount, " A volume was unmounted from this path"); FLAG_CHECK_STDERR(eventFlags[i], kFSEventStreamEventFlagItemCreated, " Item created"); FLAG_CHECK_STDERR(eventFlags[i], kFSEventStreamEventFlagItemRemoved, " Item removed"); FLAG_CHECK_STDERR(eventFlags[i], kFSEventStreamEventFlagItemInodeMetaMod, " Item metadata modified"); FLAG_CHECK_STDERR(eventFlags[i], kFSEventStreamEventFlagItemRenamed, " Item renamed"); FLAG_CHECK_STDERR(eventFlags[i], kFSEventStreamEventFlagItemModified, " Item modified"); FLAG_CHECK_STDERR(eventFlags[i], kFSEventStreamEventFlagItemFinderInfoMod, " Item Finder Info modified"); FLAG_CHECK_STDERR(eventFlags[i], kFSEventStreamEventFlagItemChangeOwner, " Item changed ownership"); FLAG_CHECK_STDERR(eventFlags[i], kFSEventStreamEventFlagItemXattrMod, " Item extended attributes modified"); FLAG_CHECK_STDERR(eventFlags[i], kFSEventStreamEventFlagItemIsFile, " Item is a file"); FLAG_CHECK_STDERR(eventFlags[i], kFSEventStreamEventFlagItemIsDir, " Item is a directory"); FLAG_CHECK_STDERR(eventFlags[i], kFSEventStreamEventFlagItemIsSymlink, " Item is a symbolic link"); fprintf(stderr, " event path: %s\n", paths[i]); fprintf(stderr, "\n"); } fprintf(stderr, "\n"); #endif if (config.format == kFSEventWatchOutputFormatClassic) { classic_output_format(numEvents, paths); } else if (config.format == kFSEventWatchOutputFormatNIW) { niw_output_format(numEvents, paths, eventFlags, eventIds); } else if (config.format == kFSEventWatchOutputFormatTNetstring) { tstring_output_format(numEvents, paths, eventFlags, eventIds, kTSITStringFormatTNetstring); } else if (config.format == kFSEventWatchOutputFormatOTNetstring) { tstring_output_format(numEvents, paths, eventFlags, eventIds, kTSITStringFormatOTNetstring); } fflush(stdout); } int main(int argc, const char* argv[]) { /* * a subprocess will initially inherit the process group of its parent. the * process group may have a control terminal associated with it, which would * be the first tty device opened by the group leader. typically the group * leader is your shell and the control terminal is your login device. a * subset of signals triggered on the control terminal are sent to all members * of the process group, in large part to facilitate sane and consistent * cleanup (ex: control terminal was closed). * * so why the overly descriptive lecture style comment? * 1. SIGINT and SIGQUIT are among the signals with this behavior * 2. a number of applications gank the above for their own use * 3. ruby's insanely useful "guard" is one of these applications * 4. despite having some level of understanding of POSIX signals and a few * of the scenarios that might cause problems, i learned this one only * after reading ruby 1.9's process.c * 5. if left completely undocumented, even slightly obscure bugfixes * may be removed as cruft by a future maintainer * * hindsight is 20/20 addition: if you're single-threaded and blocking on IO * with a subprocess, then handlers for deferrable signals might not get run * when you expect them to. In the case of Ruby 1.8, that means making use of * IO::select, which will preserve correct signal handling behavior. */ if (setpgid(0,0) < 0) { fprintf(stderr, "Unable to set new process group.\n"); return 1; } parse_cli_settings(argc, argv); FSEventStreamContext context = {0, NULL, NULL, NULL, NULL}; FSEventStreamRef stream; stream = FSEventStreamCreate(kCFAllocatorDefault, (FSEventStreamCallback)&callback, &context, config.paths, config.sinceWhen, config.latency, config.flags); #ifdef DEBUG FSEventStreamShow(stream); fprintf(stderr, "\n"); #endif FSEventStreamScheduleWithRunLoop(stream, CFRunLoopGetCurrent(), kCFRunLoopDefaultMode); FSEventStreamStart(stream); CFRunLoopRun(); FSEventStreamFlushSync(stream); FSEventStreamStop(stream); return 0; }