// This file is part of BOINC. // http://boinc.berkeley.edu // Copyright (C) 2009 University of California // // BOINC is free software; you can redistribute it and/or modify it // under the terms of the GNU Lesser General Public License // as published by the Free Software Foundation, // either version 3 of the License, or (at your option) any later version. // // BOINC is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. // See the GNU Lesser General Public License for more details. // // You should have received a copy of the GNU Lesser General Public License // along with BOINC. If not, see . /* PostInstall.cpp */ // Notes on command-line installation to a remote Mac: // // When the installer is run from the Finder, this Postinstall.app will // display up to two dialogs, asking the user whether or not to: // [1] allow non-administrative users to run the BOINC Manager // (asked only if this Mac has any non-administrative users) // [2] set BOINC as the screensaver for all users who can run BOINC // (asked only if BOINC screensaver is not already set for them) // // The installer can also be run from the command line. This is useful // for installation on remote Macs. However, there is no way to respond // to dialogs during a command-line install. // // The command-line installer sets the following environment variable: // COMMAND_LINE_INSTALL=1 // The postinstall script, postupgrade script, and this Postinstall.app // detect this environment variable and do the following: // * Redirect the Postinstall.app log output to a file // /tmp/BOINCInstallLog.txt // * Suppress the 2 dialogs // * test for the existence of a file /tmp/nonadminusersok.txt; if the // file exists, allow non-administrative users to run BOINC Manager // * test for the existence of a file /tmp/setboincsaver.txt; if the // file exists, set BOINC as the screensaver for all BOINC users. // // Example: To install on a remote Mac from the command line, allowing // non-admin users to run the BOINC Manager and setting BOINC as the // screensaver: // * First SCP the "BOINC Installer.pkg" to the remote Mac's /tmp // directory, then SSh into the remote mac and enter the following // $ touch /tmp/nonadminusersok.txt // $ touch /tmp/setboincsaver.txt // $ sudo installer -pkg "/tmp/BOINC Installer.pkg" -tgt / // $ sudo reboot // #define CREATE_LOG 1 /* for debugging */ #include #include #include // getlogin #include // getpwname, getpwuid, getuid #include // getpwname, getpwuid, getuid #include // getgrnam #include // waitpid #include #include // for MAXPATHLEN #include // for chmod #include #include #include #include // for time() #include #include #include "url.h" using std::vector; using std::string; #include "SetupSecurity.h" #include "translate.h" #define admin_group_name "admin" #define boinc_master_user_name "boinc_master" #define boinc_master_group_name "boinc_master" #define boinc_project_user_name "boinc_project" #define boinc_project_group_name "boinc_project" void Initialize(void); /* function prototypes */ Boolean myFilterProc(DialogRef theDialog, EventRecord *theEvent, DialogItemIndex *itemHit); int DeleteReceipt(void); Boolean IsRestartNeeded(); void CheckUserAndGroupConflicts(); Boolean SetLoginItemOSAScript(long brandID, Boolean deleteLogInItem, char *userName); OSErr GetCurrentScreenSaverSelection(char *moduleName, size_t maxLen); OSErr SetScreenSaverSelection(char *moduleName, char *modulePath, int type); void SetSkinInUserPrefs(char *userName, char *nameOfSkin); Boolean CheckDeleteFile(char *name); void SetEUIDBackToUser (void); static char * PersistentFGets(char *buf, size_t buflen, FILE *f); static void LoadPreferredLanguages(); static Boolean ShowMessage(Boolean allowCancel, const char *format, ...); Boolean IsUserMemberOfGroup(const char *userName, const char *groupName); int CountGroupMembershipEntries(const char *userName, const char *groupName); OSErr UpdateAllVisibleUsers(long brandID); long GetBrandID(void); int TestRPCBind(void); static int compareOSVersionTo(int toMajor, int toMinor); static OSStatus ResynchSystem(void); OSErr FindProcess (OSType typeToFind, OSType creatorToFind, ProcessSerialNumberPtr processSN); pid_t FindProcessPID(char* name, pid_t thePID); static void SleepTicks(UInt32 ticksToSleep); static OSErr QuitOneProcess(OSType signature); static OSErr QuitAppleEventHandler(const AppleEvent *appleEvt, AppleEvent* reply, UInt32 refcon); void print_to_log_file(const char *format, ...); void strip_cr(char *buf); extern int check_security( char *bundlePath, char *dataPath, int use_sandbox, int isManager, char* path_to_error, int len ); /* BEGIN TEMPORARY ITEMS TO ALLOW TRANSLATORS TO START WORK */ void notused() { ShowMessage(true, (char *)_("Yes")); ShowMessage(true, (char *)_("No")); // Future feature ShowMessage(true, (char *)_("Should BOINC run even when no user is logged in?")); } /* END TEMPORARY ITEMS TO ALLOW TRANSLATORS TO START WORK */ #define NUMBRANDS 4 #define MAX_LANGUAGES_TO_TRY 5 static char * Catalog_Name = (char *)"BOINC-Setup"; static char * Catalogs_Dir = (char *)"/Library/Application Support/BOINC Data/locale/"; /* globals */ static Boolean gCommandLineInstall = false; static Boolean gQuitFlag = false; static Boolean currentUserCanRunBOINC = false; static char loginName[256]; static time_t waitPermissionsStartTime; static char *saverName[NUMBRANDS]; static char *saverNameEscaped[NUMBRANDS]; static char *brandName[NUMBRANDS]; static char *appName[NUMBRANDS]; static char *appPath[NUMBRANDS]; static char *appPathEscaped[NUMBRANDS]; static char *receiptNameEscaped[NUMBRANDS]; static char *skinName[NUMBRANDS]; enum { launchWhenDone, logoutRequired, restartRequired, nothingrequired }; /****************************************************************** *** *** *** NOTE: *** *** *** *** On entry, the postinstall or postupgrade script has set the *** *** current directory to the top level of our installer package *** *** *** ******************************************************************/ int main(int argc, char *argv[]) { Boolean Success; ProcessSerialNumber ourProcess, installerPSN; short itemHit; long brandID = 0; int i; pid_t installerPID = 0, coreClientPID = 0; FSRef fileRef; OSStatus err, err_fsref; FILE *f; char s[256]; #ifndef SANDBOX group *grp; #endif // SANDBOX appName[0] = "BOINCManager"; appPath[0] = "/Applications/BOINCManager.app"; appPathEscaped[0] = "/Applications/BOINCManager.app"; brandName[0] = "BOINC"; saverName[0] = "BOINCSaver"; saverNameEscaped[0] = "BOINCSaver"; receiptNameEscaped[0] = "/Library/Receipts/BOINC\\ Installer.pkg"; skinName[0] = "Default"; appName[1] = "GridRepublic Desktop"; appPath[1] = "/Applications/GridRepublic Desktop.app"; appPathEscaped[1] = "/Applications/GridRepublic\\ Desktop.app"; brandName[1] = "GridRepublic"; saverName[1] = "GridRepublic"; saverNameEscaped[1] = "GridRepublic"; receiptNameEscaped[1] = "/Library/Receipts/GridRepublic\\ Installer.pkg"; skinName[1] = "GridRepublic"; appName[2] = "Progress Thru Processors Desktop"; appPath[2] = "/Applications/Progress Thru Processors Desktop.app"; appPathEscaped[2] = "/Applications/Progress\\ Thru\\ Processors\\ Desktop.app"; brandName[2] = "Progress Thru Processors"; saverName[2] = "Progress Thru Processors"; saverNameEscaped[2] = "Progress\\ Thru\\ Processors"; receiptNameEscaped[2] = "/Library/Receipts/Progress\\ Thru\\ Processors\\ Installer.pkg"; skinName[2] = "ProgressThruProcessors"; appName[3] = "Charity Engine Desktop"; appPath[3] = "/Applications/Charity Engine Desktop.app"; appPathEscaped[3] = "/Applications/Charity\\ Engine\\ Desktop.app"; brandName[3] = "Charity Engine"; saverName[3] = "Charity Engine"; saverNameEscaped[3] = "Charity\\ Engine"; receiptNameEscaped[3] = "/Library/Receipts/Charity\\ Engine\\ Installer.pkg"; skinName[3] = "Charity Engine"; ::GetCurrentProcess (&ourProcess); puts("Starting PostInstall app\n"); fflush(stdout); // getlogin() gives unreliable results under OS 10.6.2, so use environment strncpy(loginName, getenv("USER"), sizeof(loginName)-1); printf("login name = %s\n", loginName); fflush(stdout); if (getenv("COMMAND_LINE_INSTALL") != NULL) { gCommandLineInstall = true; puts("command-line install\n"); fflush(stdout); } for (i=0; i= NUMBRANDS)) { // Safety check brandID = 0; } LoadPreferredLanguages(); if (compareOSVersionTo(10, 5) < 0) { ::SetFrontProcess(&ourProcess); // Remove everything we've installed // "\pSorry, this version of GridRepublic requires system 10.5 or higher." s[0] = sprintf(s+1, "Sorry, this version of %s requires system 10.5 or higher.", brandName[brandID]); StandardAlert (kAlertStopAlert, (StringPtr)s, NULL, NULL, &itemHit); // "rm -rf /Applications/GridRepublic\\ Desktop.app" sprintf(s, "rm -rf %s", appPathEscaped[brandID]); system (s); // "rm -rf /Library/Screen\\ Savers/GridRepublic.saver" sprintf(s, "rm -rf /Library/Screen\\ Savers/%s.saver", saverNameEscaped[brandID]); system (s); // "rm -rf /Library/Receipts/GridRepublic.pkg" sprintf(s, "rm -rf %s", receiptNameEscaped[brandID]); system (s); // We don't customize BOINC Data directory name for branding system ("rm -rf /Library/Application\\ Support/BOINC\\ Data"); err = kill(installerPID, SIGKILL); ExitToShell(); } sleep (2); // Install all_projects_list.xml file, but only if one doesn't // already exist, since a pre-existing one is probably newer. f = fopen("/Library/Application Support/BOINC Data/all_projects_list.xml", "r"); if (f) { fclose(f); // Already exists unlink("/Library/Application Support/BOINC Data/installer_projects_list.xml"); } else { unlink("/Library/Application Support/BOINC Data/all_projects_list.xml"); rename("/Library/Application Support/BOINC Data/installer_projects_list.xml", "/Library/Application Support/BOINC Data/all_projects_list.xml"); } Success = false; #ifdef SANDBOX CheckUserAndGroupConflicts(); for (i=0; i<5; ++i) { err = CreateBOINCUsersAndGroups(); if (err != noErr) { printf("CreateBOINCUsersAndGroups returned %ld (repetition=%d)", err, i); fflush(stdout); continue; } // err = SetBOINCAppOwnersGroupsAndPermissions("/Applications/GridRepublic Desktop.app"); err = SetBOINCAppOwnersGroupsAndPermissions(appPath[brandID]); if (err != noErr) { printf("SetBOINCAppOwnersGroupsAndPermissions returned %ld (repetition=%d)", err, i); fflush(stdout); continue; } err = SetBOINCDataOwnersGroupsAndPermissions(); if (err != noErr) { printf("SetBOINCDataOwnersGroupsAndPermissions returned %ld (repetition=%d)", err, i); fflush(stdout); continue; } err = check_security( appPath[brandID], "/Library/Application Support/BOINC Data", true, false, NULL, 0 ); if (err != noErr) { printf("check_security returned %ld (repetition=%d)", err, i); fflush(stdout); } else { break; } } #else // ! defined(SANDBOX) // The BOINC Manager and Core Client have the set-user-ID-on-execution // flag set, so their ownership is important and must match the // ownership of the BOINC Data directory. // Find an appropriate admin user to set as owner of installed files // First, try the user currently logged in grp = getgrnam(admin_group_name); i = 0; while ((p = grp->gr_mem[i]) != NULL) { // Step through all users in group admin if (strcmp(p, loginName) == 0) { Success = true; // Logged in user is a member of group admin break; } ++i; } // If currently logged in user is not admin, use first non-root admin user if (!Success) { i = 0; while ((p = grp->gr_mem[i]) != NULL) { // Step through all users in group admin if (strcmp(p, "root") != 0) break; ++i; } } // Set owner of branded BOINCManager and contents, including core client // "chown -Rf username /Applications/GridRepublic\\ Desktop.app" sprintf(s, "chown -Rf %s %s", p, appPathEscaped[brandID]); system (s); // Set owner of BOINC Screen Saver // "chown -Rf username /Library/Screen\\ Savers/GridRepublic.saver" sprintf(s, "chown -Rf %s /Library/Screen\\ Savers/%s.saver", p, saverNameEscaped[brandID]); system (s); // We don't customize BOINC Data directory name for branding // "chown -Rf username /Library/Application\\ Support/BOINC\\ Data" sprintf(s, "chown -Rf %s /Library/Application\\ Support/BOINC\\ Data", p); system (s); // "chmod -R a+s /Applications/GridRepublic\\ Desktop.app" sprintf(s, "chmod -R a+s %s", appPathEscaped[brandID]); system (s); #endif // ! defined(SANDBOX) // Remove any branded versions of BOINC other than ours (i.e., old versions) for (i=0; i< NUMBRANDS; i++) { if (i == brandID) continue; // "rm -rf /Applications/GridRepublic\\ Desktop.app" sprintf(s, "rm -rf %s", appPathEscaped[i]); system (s); // "rm -rf /Library/Screen\\ Savers/GridRepublic.saver" sprintf(s, "rm -rf /Library/Screen\\ Savers/%s.saver", saverNameEscaped[i]); system (s); } if (brandID == 0) { // Installing generic BOINC system ("rm -f /Library/Application\\ Support/BOINC\\ Data/Branding"); } // err_fsref = FSPathMakeRef((StringPtr)"/Applications/GridRepublic Desktop.app", &fileRef, NULL); err_fsref = FSPathMakeRef((StringPtr)appPath[brandID], &fileRef, NULL); if (err_fsref == noErr) err = LSRegisterFSRef(&fileRef, true); err = UpdateAllVisibleUsers(brandID); if (err != noErr) return err; #if 0 // WaitPermissions is not needed when using wrapper #ifdef SANDBOX pid_t waitPermissionsPID = 0; uid_t saved_euid, saved_uid, b_m_uid; passwd *pw; Boolean restartNeeded; DialogRef theWin; restartNeeded = IsRestartNeeded(); printf("IsRestartNeeded() returned %d\n", (int)restartNeeded); fflush(stdout); if (!restartNeeded) { // Wait for BOINC's RPC socket address to become available to user boinc_master, in // case we are upgrading from a version which did not run as user boinc_master. saved_uid = getuid(); saved_euid = geteuid(); pw = getpwnam(boinc_master_user_name); b_m_uid = pw->pw_uid; seteuid(b_m_uid); for (i=0; i<120; i++) { err = TestRPCBind(); if (err == noErr) break; sleep(1); } seteuid(saved_euid); FSRef theFSRef; err = FSPathMakeRef((StringPtr)"/Library/Application Support/BOINC Data/WaitPermissions.app", &theFSRef, NULL); if (err) { printf("FSPathMakeRef(WaitPermissions) returned error %ld\n", err); fflush(stdout); } // When we first create the boinc_master group and add the current user to the // new group, there is a delay before the new group membership is recognized. // If we launch the BOINC Manager too soon, it will fail with a -1037 permissions // error, so we wait until the current user can access the switcher application. // Apparently, in order to get the changed permissions / group membership, we must // launch a new process belonging to the user. It may also need to be in a new // process group or new session. Neither system() nor popen() works, even after // setting the uid and euid back to the logged in user, but LSOpenFSRef() does. // The WaitPermissions application loops until it can access the switcher // application. err = LSOpenFSRef(&theFSRef, NULL); if (err) { printf("LSOpenFSRef(WaitPermissions) returned error %ld\n", err); fflush(stdout); } waitPermissionsStartTime = time(NULL); for (i=0; i<15; i++) { // Show "Please wait..." alert after 15 seconds waitPermissionsPID = FindProcessPID("WaitPermissions", 0); if (waitPermissionsPID == 0) { return 0; } sleep(1); } if (gCommandLineInstall) { printf("Finishing install. Please wait ...\n"); printf("This may take a few more minutes.\n"); fflush(stdout); } else { CreateStandardAlert(kAlertNoteAlert, CFSTR("Finishing install. Please wait ..."), CFSTR("This may take a few more minutes."), NULL, &theWin); HideDialogItem(theWin, kStdOkItemIndex); RemoveDialogItems(theWin, kStdOkItemIndex, 1, false); RunStandardAlert(theWin, &myFilterProc, &itemHit); } } #endif // SANDBOX #endif // WaitPermissions is not needed when using wrapper return 0; } Boolean myFilterProc(DialogRef theDialog, EventRecord *theEvent, DialogItemIndex *itemHit) { static time_t lastCheckTime = 0; time_t now = time(NULL); pid_t waitPermissionsPID = 0; if (now != lastCheckTime) { waitPermissionsPID = FindProcessPID("WaitPermissions", 0); if (waitPermissionsPID == 0) { *itemHit = kStdOkItemIndex; return true; } lastCheckTime = now; // Limit delay to 3 minutes if ((now - waitPermissionsStartTime) > 180) { *itemHit = kStdOkItemIndex; return true; } } return false; } // After installation has completed, delete the installer receipt. // If we don't need to logout the user, also launch BOINC Manager. int DeleteReceipt() { ProcessSerialNumber installerPSN; long brandID = 0; int i; pid_t installerPID = 0; OSStatus err; Boolean restartNeeded = true; FSRef fileRef; char s[256]; struct stat sbuf; OSStatus err_fsref; Initialize(); restartNeeded = IsRestartNeeded(); printf("IsRestartNeeded() returned %d\n", (int)restartNeeded); fflush(stdout); brandID = GetBrandID(); // Remove installer package receipt so we can run installer again if needed to fix permissions // "rm -rf /Library/Receipts/GridRepublic.pkg" sprintf(s, "rm -rf %s", receiptNameEscaped[brandID]); system (s); // err_fsref = FSPathMakeRef((StringPtr)"/Applications/GridRepublic Desktop.app", &fileRef, NULL); err_fsref = FSPathMakeRef((StringPtr)appPath[brandID], &fileRef, NULL); if (!restartNeeded) { err = FindProcess ('APPL', 'xins', &installerPSN); if (err == noErr) { err = GetProcessPID(&installerPSN , &installerPID); // Launch BOINC Manager when user closes installer or after 15 seconds for (i=0; i<15; i++) { // Wait 15 seconds max for installer to quit sleep (1); if (err == noErr) { if (FindProcessPID(NULL, installerPID) == 0) { break; } } } } // If system is set up to run BOINC Client as a daemon using launchd, launch it // as a daemon and allow time for client to start before launching BOINC Manager. err = stat("/Library/LaunchDaemons/edu.berkeley.boinc.plist", &sbuf); if (err == noErr) { system("launchctl unload /Library/LaunchDaemons/edu.berkeley.boinc.plist"); i = system("launchctl load /Library/LaunchDaemons/edu.berkeley.boinc.plist"); if (i == 0) sleep (2); } err = LSOpenFSRef(&fileRef, NULL); } return 0; } // BOINC Installer.app wrote a file to tell us whether a restart is required Boolean IsRestartNeeded() { FILE *restartNeededFile; int value; restartNeededFile = fopen("/tmp/BOINC_restart_flag", "r"); if (!restartNeededFile) { restartNeededFile = fopen("/private/tmp/BOINC_restart_flag", "r"); } if (restartNeededFile) { fscanf(restartNeededFile,"%d", &value); fclose(restartNeededFile); return (value != 0); } return true; } // Some newer versions of the OS define users and groups which may conflict with // our previously created boinc_master or boinc_project user or group. This could // also happen when the user installs new software. So we must check for such // duplicate UserIDs and groupIDs; if found, we delete our user or group so that // the PostInstall application will create a new one that does not conflict. // // Older versions of the installer created our users and groups at the first // unused IDs at or above 25. Apple now recommends using IDs at or above 501, // to reduce the likelihood of conflicts with future UserIDs and groupIDs. // If we have previously created UserIDs and / or groupIDs below 501, this code // now removes them so we can create new ones above 500. void CheckUserAndGroupConflicts() { #ifdef SANDBOX passwd *pw = NULL; group *grp = NULL; gid_t boinc_master_gid = 0, boinc_project_gid = 0; uid_t boinc_master_uid = 0, boinc_project_uid = 0; FILE *f; char cmd[256], buf[256]; int entryCount; OSErr err = noErr; if (compareOSVersionTo(10, 5) < 0) { // This fails under OS 10.4, but should not be needed under OS 10.4 return; } printf("Checking user and group conflicts\n"); fflush(stdout); entryCount = 0; grp = getgrnam(boinc_master_group_name); if (grp) { boinc_master_gid = grp->gr_gid; printf("boinc_master group ID = %d\n", (int)boinc_master_gid); fflush(stdout); if (boinc_master_gid > 500) { sprintf(cmd, "dscl . -search /Groups PrimaryGroupID %d", boinc_master_gid); f = popen(cmd, "r"); if (f) { while (PersistentFGets(buf, sizeof(buf), f)) { if (strstr(buf, "PrimaryGroupID")) { ++entryCount; } } pclose(f); } } } if ((boinc_master_gid < 501) || (entryCount > 1)) { err = system ("dscl . -delete /groups/boinc_master"); // User boinc_master must have group boinc_master as its primary group. // Since this group no longer exists, delete the user as well. if (err) { fprintf(stdout, "dscl . -delete /groups/boinc_master returned %d\n", err); fflush(stdout); } err = system ("dscl . -delete /users/boinc_master"); if (err) { fprintf(stdout, "dscl . -delete /users/boinc_master returned %d\n", err); fflush(stdout); } ResynchSystem(); } entryCount = 0; grp = getgrnam(boinc_project_group_name); if (grp) { boinc_project_gid = grp->gr_gid; printf("boinc_project group ID = %d\n", (int)boinc_project_gid); fflush(stdout); if (boinc_project_gid > 500) { sprintf(cmd, "dscl . -search /Groups PrimaryGroupID %d", boinc_project_gid); f = popen(cmd, "r"); if (f) { while (PersistentFGets(buf, sizeof(buf), f)) { if (strstr(buf, "PrimaryGroupID")) { ++entryCount; } } pclose(f); } } } if ((boinc_project_gid < 501) || (entryCount > 1)) { err = system ("dscl . -delete /groups/boinc_project"); if (err) { fprintf(stdout, "dscl . -delete /groups/boinc_project returned %d\n", err); fflush(stdout); } // User boinc_project must have group boinc_project as its primary group. // Since this group no longer exists, delete the user as well. err = system ("dscl . -delete /users/boinc_project"); if (err) { fprintf(stdout, "dscl . -delete /users/boinc_project returned %d\n", err); fflush(stdout); } ResynchSystem(); } if ((boinc_master_gid < 500) && (boinc_project_gid < 500)) { return; } entryCount = 0; pw = getpwnam(boinc_master_user_name); if (pw) { boinc_master_uid = pw->pw_uid; printf("boinc_master user ID = %d\n", (int)boinc_master_uid); fflush(stdout); sprintf(cmd, "dscl . -search /Users UniqueID %d", boinc_master_uid); f = popen(cmd, "r"); if (f) { while (PersistentFGets(buf, sizeof(buf), f)) { if (strstr(buf, "UniqueID")) { ++entryCount; } } pclose(f); } } if (entryCount > 1) { err = system ("dscl . -delete /users/boinc_master"); if (err) { fprintf(stdout, "dscl . -delete /users/boinc_master returned %d\n", err); fflush(stdout); } ResynchSystem(); } entryCount = 0; pw = getpwnam(boinc_project_user_name); if (pw) { boinc_project_uid = pw->pw_uid; printf("boinc_project user ID = %d\n", (int)boinc_project_uid); fflush(stdout); sprintf(cmd, "dscl . -search /Users UniqueID %d", boinc_project_uid); f = popen(cmd, "r"); if (f) { while (PersistentFGets(buf, sizeof(buf), f)) { if (strstr(buf, "UniqueID")) { ++entryCount; } } pclose(f); } } if (entryCount > 1) { system ("dscl . -delete /users/boinc_project"); if (err) { fprintf(stdout, "dscl . -delete /users/boinc_project returned %d\n", err); fflush(stdout); } ResynchSystem(); } #endif // SANDBOX } enum { kSystemEventsCreator = 'sevs' }; Boolean SetLoginItemOSAScript(long brandID, Boolean deleteLogInItem, char *userName) { int i, j; char cmd[2048]; char systemEventsPath[1024]; ProcessSerialNumber SystemEventsPSN; FSRef appRef; OSErr err, err2; fprintf(stdout, "Adjusting login items for user %s\n", userName); fflush(stdout); // We must launch the System Events application for the target user err = FindProcess ('APPL', kSystemEventsCreator, &SystemEventsPSN); if (err == noErr) { // Find SystemEvents process. If found, quit it in case // it is running under a different user. fprintf(stdout, "Telling System Events to quit (at start of SetLoginItemOSAScript)\n"); fflush(stdout); err = QuitOneProcess(kSystemEventsCreator); if (err != noErr) { fprintf(stdout, "QuitOneProcess(kSystemEventsCreator) returned error %d \n", (int) err); fflush(stdout); } // Wait for the process to be gone for (i=0; i<50; ++i) { // 5 seconds max delay SleepTicks(6); // 6 Ticks == 1/10 second err = FindProcess ('APPL', kSystemEventsCreator, &SystemEventsPSN); if (err != noErr) break; } if (i >= 50) { fprintf(stdout, "Failed to make System Events quit\n"); fflush(stdout); err = noErr; goto cleanupSystemEvents; } sleep(4); } err = LSFindApplicationForInfo(kSystemEventsCreator, NULL, NULL, &appRef, NULL); if (err != noErr) { fprintf(stdout, "LSFindApplicationForInfo(kSystemEventsCreator) returned error %d \n", (int) err); fflush(stdout); goto cleanupSystemEvents; } else { FSRefMakePath(&appRef, (UInt8*)systemEventsPath, sizeof(systemEventsPath)); fprintf(stdout, "SystemEvents is at %s\n", systemEventsPath); fprintf(stdout, "Launching SystemEvents for user %s\n", userName); fflush(stdout); for (j=0; j<5; ++j) { sprintf(cmd, "sudo -u \"%s\" \"%s/Contents/MacOS/System Events\" &", userName, systemEventsPath); err = system(cmd); if (err) { fprintf(stdout, "[2] Command: %s returned error %d (try %d of 5)\n", cmd, (int) err, j); } // Wait for the process to start for (i=0; i<50; ++i) { // 5 seconds max delay SleepTicks(6); // 6 Ticks == 1/10 second err = FindProcess ('APPL', kSystemEventsCreator, &SystemEventsPSN); if (err == noErr) break; } if (i < 50) break; // Exit j loop on success } if (j >= 5) { fprintf(stdout, "Failed to launch System Events for user %s\n", userName); fflush(stdout); err = noErr; goto cleanupSystemEvents; } } sleep(2); for (i=0; i= 50) { fprintf(stdout, "Failed to make System Events quit\n"); fflush(stdout); } sleep(4); return (err == noErr); } // Sets the skin selection in the specified user's preferences to the specified skin void SetSkinInUserPrefs(char *userName, char *nameOfSkin) { passwd *pw; FILE *oldPrefs, *newPrefs; char oldFileName[MAXPATHLEN], tempFilename[MAXPATHLEN]; char buf[1024]; int wroteSkinName; struct stat sbuf; group *grp; OSStatus statErr; if (nameOfSkin[0]) { sprintf(oldFileName, "/Users/%s/Library/Preferences/BOINC Manager Preferences", userName); sprintf(tempFilename, "/Users/%s/Library/Preferences/BOINC Manager NewPrefs", userName); newPrefs = fopen(tempFilename, "w"); if (newPrefs) { wroteSkinName = 0; statErr = stat(oldFileName, &sbuf); oldPrefs = fopen(oldFileName, "r"); if (oldPrefs) { while (fgets(buf, sizeof(buf), oldPrefs)) { if (strstr(buf, "Skin=")) { fprintf(newPrefs, "Skin=%s\n", nameOfSkin); wroteSkinName = 1; } else { fputs(buf, newPrefs); } } fclose(oldPrefs); } if (! wroteSkinName) fprintf(newPrefs, "Skin=%s\n", nameOfSkin); fclose(newPrefs); rename(tempFilename, oldFileName); // Deletes old file if (! statErr) { chown(oldFileName, sbuf.st_uid, sbuf.st_gid); chmod(oldFileName, sbuf.st_mode); } else { chmod(oldFileName, 0664); pw = getpwnam(userName); grp = getgrnam(userName); if (pw && grp) chown(oldFileName, pw->pw_uid, grp->gr_gid); } } } } // Returns true if the user name is in the nologinitems.txt, else false Boolean CheckDeleteFile(char *name) { FILE *f; char buf[64]; size_t len; f = fopen("/Library/Application Support/BOINC Data/nologinitems.txt", "r"); if (!f) return false; while (true) { *buf = '\0'; len = sizeof(buf); fgets(buf, len, f); if (feof(f)) break; strip_cr(buf); if (strcmp(buf, name) == 0) { fclose(f); return true; } } fclose(f); return false; } void SetEUIDBackToUser (void) { uid_t login_uid; passwd *pw; pw = getpwnam(loginName); login_uid = pw->pw_uid; setuid(login_uid); seteuid(login_uid); } static char * PersistentFGets(char *buf, size_t buflen, FILE *f) { char *p = buf; size_t len = buflen; size_t datalen = 0; memset(buf, 0, buflen); while (datalen < (buflen - 1)) { fgets(p, len, f); if (feof(f)) break; if (ferror(f) && (errno != EINTR)) break; if (strchr(buf, '\n')) break; datalen = strlen(buf); p = buf + datalen; len -= datalen; } return (buf[0] ? buf : NULL); } // Because language preferences are set on a per-user basis, we // must get the preferred languages while set to the current // user, before the Apple Installer switches us to root. // So we get the preferred languages in our Installer.app which // writes them to a temporary file which we retrieve here. // We must do it this way because, for unknown reasons, the // CFBundleCopyLocalizationsForPreferences() API does not work // correctly if we seteuid and setuid to the logged in user by // calling SetEUIDBackToUser() after running as root. // static void LoadPreferredLanguages(){ FILE *f; int i; char *p; char language[32]; BOINCTranslationInit(); // GetPreferredLanguages() wrote a list of our preferred languages to a temp file f = fopen("/tmp/BOINC_preferred_languages", "r"); if (!f) return; for (i=0; igr_mem[i]) != NULL) { // Step through all users in group admin if (strcmp(p, userName) == 0) { return true; } ++i; } return false; } // OS 10.7 dscl merge command has a bug such that the command: // dscl . -merge /Groups/GROUPNAME users USERNAME // adds the user to the group even if it was already a member, resulting in // duplicate (multiple) entries. Earlier BOINC versions used this command // but did not check for this, so we remove duplicate entries if present. // Note: We now avoid this problem by instead using the command: // dscl . -merge /Groups/GROUPNAME GroupMembership USERNAME // which correctly avoids duplication. int CountGroupMembershipEntries(const char *userName, const char *groupName) { int count = 0; char cmd[512], buf[2048], escapedUserName[1024]; FILE *f; char *p, *q; // getgrnam(groupName)->gr_mem[] only returns one entry, so we must use dscl escape_url(userName, escapedUserName, sizeof(escapedUserName)); // Avoid confusion if name has embedded spaces sprintf(cmd, "dscl -url . -read /Groups/%s GroupMembership", groupName); f = popen(cmd, "r"); if (f == NULL) return 0; while (PersistentFGets(buf, sizeof(buf), f)) { p = buf; while (p) { p = strstr(p, escapedUserName); if (p) { q = p-1; p += strlen(escapedUserName); // Count only whole words (preceded and followed by white space) so // that if we have both 'jon' and 'jones' we don't count 'jon' twice if (isspace(*q) && isspace(*p)) { ++ count; } } } } pclose(f); return count; } // Find all visible users. // If user is a member of group admin, add user to groups boinc_master and boinc_project. // Optionally add non-admin users to group boinc_master but not to group boinc_project. // Set login item for all members of group boinc_master to launch BOINC Manager. // If our install package included a skin, set those user's preferences to use that skin. // Optionally set BOINC as screensaver for all users running BOINC. OSErr UpdateAllVisibleUsers(long brandID) { passwd *pw; vector human_user_names; vector human_user_IDs; uid_t saved_uid; Boolean deleteLoginItem; char human_user_name[256]; char s[256]; Boolean saverAlreadySetForAll = true; Boolean setSaverForAllUsers = false; Boolean allNonAdminUsersAreSet = true; Boolean allowNonAdminUsersToRunBOINC = false; Boolean found = false; FILE *f; int err; Boolean isAdminGroupMember, isBMGroupMember, isBPGroupMember; struct stat sbuf; #ifdef SANDBOX char cmd[256]; int BMGroupMembershipCount, BPGroupMembershipCount; int i; #endif int userIndex; char buf[256]; char *p; int flag; // char nameOfSkin[256]; // FindSkinName(nameOfSkin, sizeof(nameOfSkin)); // Step through all users puts("Beginning first pass through all users\n"); fflush(stdout); f = popen("dscl . list /Users UniqueID", "r"); if (f) { while (PersistentFGets(buf, sizeof(buf), f)) { p = strrchr(buf, ' '); if (p) { int id = atoi(p+1); if (id < 501) continue; human_user_IDs.push_back((uid_t)id); while (p > buf) { if (*p != ' ') break; --p; } *(p+1) = '\0'; human_user_names.push_back(string(buf)); *(p+1) = ' '; } } pclose(f); } for (userIndex=human_user_names.size(); userIndex>0; --userIndex) { flag = 0; strlcpy(human_user_name, human_user_names[userIndex-1].c_str(), sizeof(human_user_name)); sprintf(cmd, "dscl . -read \"/Users/%s\" NFSHomeDirectory", human_user_name); f = popen(cmd, "r"); if (f) { while (PersistentFGets(buf, sizeof(buf), f)) { p = strrchr(buf, ' '); if (p) { if (strstr(p, "/var/empty") != NULL) flag = 1; } } pclose(f); } sprintf(cmd, "dscl . -read \"/Users/%s\" UserShell", human_user_name); f = popen(cmd, "r"); if (f) { while (PersistentFGets(buf, sizeof(buf), f)) { p = strrchr(buf, ' '); if (p) { if (strstr(p, "/usr/bin/false") != NULL) flag |= 2; } } pclose(f); } if (flag == 3) { // if (Home Directory == "/var/empty") && (UserShell == "/usr/bin/false") human_user_names.erase(human_user_names.begin()+userIndex-1); human_user_IDs.erase(human_user_IDs.begin()+userIndex-1); } } for (userIndex=0; userIndex< (int)human_user_names.size(); ++userIndex) { strlcpy(human_user_name, human_user_names[userIndex].c_str(), sizeof(human_user_name)); printf("[1] Checking user %s\n", human_user_name); fflush(stdout); // getpwnam works with either the full / login name (pw->pw_gecos) // or the short / Posix name (pw->pw_name) pw = getpwnam(human_user_name); if (pw == NULL) { printf("[1] %s not in getpwnam data base\n", human_user_name); fflush(stdout); continue; } printf("[1] User %s: Posix name=%s, Full name=%s\n", human_user_name, pw->pw_name, pw->pw_gecos); fflush(stdout); #ifdef SANDBOX isAdminGroupMember = false; isBMGroupMember = false; isAdminGroupMember = IsUserMemberOfGroup(pw->pw_name, admin_group_name); if (isAdminGroupMember) { // User is a member of group admin, so add user to groups boinc_master and boinc_project printf("[1] User %s is a member of group admin\n", pw->pw_name); fflush(stdout); } else { isBMGroupMember = IsUserMemberOfGroup(pw->pw_name, boinc_master_group_name); if (isBMGroupMember) { // User is a member of group boinc_master printf("[1] Non-admin user %s is a member of group boinc_master\n", pw->pw_name); fflush(stdout); } else { allNonAdminUsersAreSet = false; } } #else // SANDBOX isGroupMember = true; #endif // SANDBOX if (isAdminGroupMember || isBMGroupMember) { if ((strcmp(loginName, human_user_name) == 0) || (strcmp(loginName, pw->pw_name) == 0) || (strcmp(loginName, pw->pw_gecos) == 0)) { currentUserCanRunBOINC = true; } saved_uid = geteuid(); if (compareOSVersionTo(10, 6) < 0) { sprintf(cmd, "sudo -u \"%s\" defaults -currentHost read com.apple.screensaver moduleName", pw->pw_name); f = popen(cmd, "r"); if (f) { found = false; while (PersistentFGets(s, sizeof(s), f)) { if (strstr(s, saverName[brandID])) { found = true; break; } } pclose(f); if (!found) { saverAlreadySetForAll = false; } } } else { seteuid(pw->pw_uid); // Temporarily set effective uid to this user err = GetCurrentScreenSaverSelection(s, sizeof(s) -1); if (err == noErr) { if (!strstr(s, saverName[brandID])) { saverAlreadySetForAll = false; } } seteuid(saved_uid); // Set effective uid back to privileged user } } // End if (isGroupMember) } // End for (userIndex=0; userIndex< human_user_names.size(); ++userIndex) ResynchSystem(); if (allNonAdminUsersAreSet) { puts("[2] All non-admin users are already members of group boinc_master\n"); fflush(stdout); } else { if (gCommandLineInstall) { err = stat("/tmp/nonadminusersok.txt", &sbuf); if (err == noErr) { puts("nonadminusersok.txt file detected\n"); fflush(stdout); unlink("/tmp/nonadminusersok.txt"); allowNonAdminUsersToRunBOINC = true; currentUserCanRunBOINC = true; saverAlreadySetForAll = false; } } else { if (ShowMessage(true, (char *)_("Users who are permitted to administer this computer will automatically be allowed to " "run and control %s.\n\n" "Do you also want non-administrative users to be able to run and control %s on this Mac?"), brandName[brandID], brandName[brandID]) ) { allowNonAdminUsersToRunBOINC = true; currentUserCanRunBOINC = true; saverAlreadySetForAll = false; printf("[2] User answered Yes to allowing non-admin users to run %s\n", brandName[brandID]); fflush(stdout); } else { printf("[2] User answered No to allowing non-admin users to run %s\n", brandName[brandID]); fflush(stdout); } } } if (! saverAlreadySetForAll) { if (gCommandLineInstall) { err = stat("/tmp/setboincsaver.txt", &sbuf); if (err == noErr) { puts("setboincsaver.txt file detected\n"); fflush(stdout); unlink("/tmp/setboincsaver.txt"); setSaverForAllUsers = true; } } else { setSaverForAllUsers = ShowMessage(true, (char *)_("Do you want to set %s as the screensaver for all %s users on this Mac?"), brandName[brandID], brandName[brandID]); } } // Step through all users a second time, setting non-admin users and / or our screensaver puts("Beginning second pass through all users\n"); fflush(stdout); for (userIndex=0; userIndex<(int)human_user_names.size(); ++userIndex) { strlcpy(human_user_name, human_user_names[userIndex].c_str(), sizeof(human_user_name)); printf("[2] Checking user %s\n", human_user_name); fflush(stdout); pw = getpwnam(human_user_name); if (pw == NULL) { // "Deleted Users", "Shared", etc. printf("[2] %s not in getpwnam data base\n", human_user_name); fflush(stdout); continue; } printf("[2] User %s: Posix name=%s, Full name=%s\n", human_user_name, pw->pw_name, pw->pw_gecos); fflush(stdout); #ifdef SANDBOX isAdminGroupMember = false; isBMGroupMember = false; isBPGroupMember = false; isAdminGroupMember = IsUserMemberOfGroup(pw->pw_name, admin_group_name); if (isAdminGroupMember) { // User is a member of group admin, so add user to groups boinc_master and boinc_project printf("[2] User %s is a member of group admin\n", pw->pw_name); fflush(stdout); } // If allNonAdminUsersAreSet, some older BOINC versions added non-admin users only to group // boinc_master; ensure all permitted BOINC users are also members of group boinc_project if (isAdminGroupMember || allowNonAdminUsersToRunBOINC || allNonAdminUsersAreSet) { // OS 10.7 dscl merge command has a bug that it adds the user to the group even if // it was already a member, resulting in duplicate (multiple) entries. Earlier BOINC // versions did not check for this, so we remove duplicate entries if present. BMGroupMembershipCount = CountGroupMembershipEntries(pw->pw_name, boinc_master_group_name); printf("[2] User %s found in group %s member list %d times\n", pw->pw_name, boinc_master_group_name, BMGroupMembershipCount); fflush(stdout); if (BMGroupMembershipCount == 0) { sprintf(cmd, "dscl . -merge /groups/%s GroupMembership \"%s\"", boinc_master_group_name, pw->pw_name); err = system(cmd); printf("[2] %s returned %d\n", cmd, err); fflush(stdout); isBMGroupMember = true; } else { isBMGroupMember = true; for (i=1; ipw_name); err = system(cmd); printf("[2] %s returned %d\n", cmd, err); fflush(stdout); } } BPGroupMembershipCount = CountGroupMembershipEntries(pw->pw_name, boinc_project_group_name); printf("[2] User %s found in group %s member list %d times\n", pw->pw_name, boinc_project_group_name, BPGroupMembershipCount); fflush(stdout); if (BPGroupMembershipCount == 0) { sprintf(cmd, "dscl . -merge /groups/%s GroupMembership \"%s\"", boinc_project_group_name, pw->pw_name); err = system(cmd); printf("[2] %s returned %d\n", cmd, err); fflush(stdout); isBPGroupMember = true; } else { isBPGroupMember = true; for (i=1; ipw_name); err = system(cmd); printf("[2] %s returned %d\n", cmd, err); fflush(stdout); } } } #else // SANDBOX isBMGroupMember = true; #endif // SANDBOX saved_uid = geteuid(); deleteLoginItem = CheckDeleteFile(human_user_name); if (CheckDeleteFile(pw->pw_name)) { deleteLoginItem = true; } if (CheckDeleteFile(pw->pw_gecos)) { deleteLoginItem = true; } if (!isBMGroupMember) { deleteLoginItem = true; } // Set login item for this user printf("[2] calling SetLoginItemOSAScript for user %s, euid = %d, deleteLoginItem = %d\n", pw->pw_name, geteuid(), deleteLoginItem); fflush(stdout); SetLoginItemOSAScript(brandID, deleteLoginItem, pw->pw_name); if (isBMGroupMember) { // For some reason we need to call getpwnam again on OS 10.5 pw = getpwnam(human_user_name); if (pw == NULL) { // "Deleted Users", "Shared", etc. printf("[2] ERROR: %s was in getpwnam data base but now is not!\n", human_user_name); fflush(stdout); continue; } SetSkinInUserPrefs(pw->pw_name, skinName[brandID]); if (setSaverForAllUsers) { if (compareOSVersionTo(10, 6) < 0) { sprintf(s, "sudo -u \"%s\" defaults -currentHost write com.apple.screensaver moduleName %s", pw->pw_name, saverNameEscaped[brandID]); system (s); sprintf(s, "sudo -u \"%s\" defaults -currentHost write com.apple.screensaver modulePath /Library/Screen\\ Savers/%s.saver", pw->pw_name, saverNameEscaped[brandID]); system (s); } else { seteuid(pw->pw_uid); // Temporarily set effective uid to this user sprintf(s, "/Library/Screen Savers/%s.saver", saverName[brandID]); err = SetScreenSaverSelection(saverName[brandID], s, 0); seteuid(saved_uid); // Set effective uid back to privileged user } } } // Delete the BOINC Manager's wxSingleInstanceChecker lock file, in case // it was not deleted (such as due to a crash.) // Lock file name always has "BOINC Manager" even if the application is // branded, due to SetAppName(wxT("BOINC Manager")) in CBOINCGUIApp::OnInit(). // This path must match that in CBOINCGUIApp::DetectDuplicateInstance() sprintf(cmd, "sudo -u \"%s\" rm -f \"/Users/%s/Library/Application Support/BOINC/BOINC Manager-%s\"", pw->pw_name, pw->pw_name, pw->pw_name); err = system(cmd); printf("[2] %s returned %d\n", cmd, err); fflush(stdout); } // End for (userIndex=0; userIndex< human_user_names.size(); ++userIndex) ResynchSystem(); BOINCTranslationCleanup(); return noErr; } OSErr GetCurrentScreenSaverSelection(char *moduleName, size_t maxLen) { OSErr err = noErr; CFStringRef nameKey = CFStringCreateWithCString(NULL,"moduleName",kCFStringEncodingASCII); CFStringRef moduleNameAsCFString; CFDictionaryRef theData; theData = (CFDictionaryRef)CFPreferencesCopyValue(CFSTR("moduleDict"), CFSTR("com.apple.screensaver"), kCFPreferencesCurrentUser, kCFPreferencesCurrentHost ); if (theData == NULL) { CFRelease(nameKey); return (-1); } if (CFDictionaryContainsKey(theData, nameKey) == false) { moduleName[0] = 0; CFRelease(nameKey); CFRelease(theData); return(-1); } moduleNameAsCFString = CFStringCreateCopy(NULL, (CFStringRef)CFDictionaryGetValue(theData, nameKey)); CFStringGetCString(moduleNameAsCFString, moduleName, maxLen, kCFStringEncodingASCII); CFRelease(nameKey); CFRelease(theData); CFRelease(moduleNameAsCFString); return err; } OSErr SetScreenSaverSelection(char *moduleName, char *modulePath, int type) { OSErr err = noErr; CFStringRef preferenceName = CFSTR("com.apple.screensaver"); CFStringRef mainKeyName = CFSTR("moduleDict"); CFDictionaryRef emptyData; CFMutableDictionaryRef newData; Boolean success; CFStringRef nameKey = CFStringCreateWithCString(NULL, "moduleName", kCFStringEncodingASCII); CFStringRef nameValue = CFStringCreateWithCString(NULL, moduleName, kCFStringEncodingASCII); CFStringRef pathKey = CFStringCreateWithCString(NULL, "path", kCFStringEncodingASCII); CFStringRef pathValue = CFStringCreateWithCString(NULL, modulePath, kCFStringEncodingASCII); CFStringRef typeKey = CFStringCreateWithCString(NULL, "type", kCFStringEncodingASCII); CFNumberRef typeValue = CFNumberCreate(NULL, kCFNumberIntType, &type); emptyData = CFDictionaryCreate(NULL, NULL, NULL, 0, NULL, NULL); if (emptyData == NULL) { CFRelease(nameKey); CFRelease(nameValue); CFRelease(pathKey); CFRelease(pathValue); CFRelease(typeKey); CFRelease(typeValue); return(-1); } newData = CFDictionaryCreateMutableCopy(NULL,0, emptyData); if (newData == NULL) { CFRelease(nameKey); CFRelease(nameValue); CFRelease(pathKey); CFRelease(pathValue); CFRelease(typeKey); CFRelease(typeValue); CFRelease(emptyData); return(-1); } CFDictionaryAddValue(newData, nameKey, nameValue); CFDictionaryAddValue(newData, pathKey, pathValue); CFDictionaryAddValue(newData, typeKey, typeValue); CFPreferencesSetValue(mainKeyName, newData, preferenceName, kCFPreferencesCurrentUser, kCFPreferencesCurrentHost); success = CFPreferencesSynchronize(preferenceName, kCFPreferencesCurrentUser, kCFPreferencesCurrentHost); if (!success) { err = -1; } CFRelease(nameKey); CFRelease(nameValue); CFRelease(pathKey); CFRelease(pathValue); CFRelease(typeKey); CFRelease(typeValue); CFRelease(emptyData); CFRelease(newData); return err; } void Initialize() /* Initialize some managers */ { OSErr err; // InitCursor(); err = AEInstallEventHandler( kCoreEventClass, kAEQuitApplication, NewAEEventHandlerUPP((AEEventHandlerProcPtr)QuitAppleEventHandler), 0, false ); if (err != noErr) ExitToShell(); } long GetBrandID() { long iBrandId; iBrandId = 0; // Default value FILE *f = fopen("Contents/Resources/Branding", "r"); if (f) { fscanf(f, "BrandId=%ld\n", &iBrandId); fclose(f); } return iBrandId; } int TestRPCBind() { sockaddr_in addr; int lsock; int retval; lsock = (int)socket(AF_INET, SOCK_STREAM, 0); if (lsock < 0) return -153; memset(&addr, 0, sizeof(addr)); addr.sin_family = AF_INET; addr.sin_port = htons(31416); addr.sin_addr.s_addr = htonl(INADDR_ANY); int one = 1; retval = setsockopt(lsock, SOL_SOCKET, SO_REUSEADDR, (char*)&one, 4); if (! retval) retval = bind(lsock, (const sockaddr*)(&addr), (socklen_t)sizeof(addr)); if (! retval) retval = listen(lsock, 999); close(lsock); return retval; } static OSStatus ResynchSystem() { OSStatus err = noErr; err = system("dscacheutil -flushcache"); err = system("dsmemberutil flushcache"); return noErr; } static int compareOSVersionTo(int toMajor, int toMinor) { SInt32 major, minor; OSStatus err = noErr; err = Gestalt(gestaltSystemVersionMajor, &major); if (err != noErr) { fprintf(stdout, "Gestalt(gestaltSystemVersionMajor) returned error %ld\n", err); fflush(stdout); return -1; // gestaltSystemVersionMajor selector was not available before OS 10.4 } if (major < toMajor) return -1; if (major > toMajor) return 1; err = Gestalt(gestaltSystemVersionMinor, &minor); if (err != noErr) { fprintf(stdout, "Gestalt(gestaltSystemVersionMinor) returned error %ld\n", err); fflush(stdout); return -1; // gestaltSystemVersionMajor selector was not available before OS 10.4 } if (minor < toMinor) return -1; if (minor > toMinor) return 1; return 0; } // --------------------------------------------------------------------------- /* This runs through the process list looking for the indicated application */ /* Searches for process by file type and signature (creator code) */ // --------------------------------------------------------------------------- OSErr FindProcess (OSType typeToFind, OSType creatorToFind, ProcessSerialNumberPtr processSN) { ProcessInfoRec tempInfo; FSSpec procSpec; Str31 processName; OSErr myErr = noErr; /* null out the PSN so we're starting at the beginning of the list */ processSN->lowLongOfPSN = kNoProcess; processSN->highLongOfPSN = kNoProcess; /* initialize the process information record */ tempInfo.processInfoLength = sizeof(ProcessInfoRec); tempInfo.processName = processName; tempInfo.processAppSpec = &procSpec; /* loop through all the processes until we */ /* 1) find the process we want */ /* 2) error out because of some reason (usually, no more processes) */ do { myErr = GetNextProcess(processSN); if (myErr == noErr) GetProcessInformation(processSN, &tempInfo); } while ((tempInfo.processSignature != creatorToFind || tempInfo.processType != typeToFind) && myErr == noErr); return(myErr); } // Uses usleep to sleep for full duration even if a signal is received static void SleepTicks(UInt32 ticksToSleep) { UInt32 endSleep, timeNow, ticksRemaining; timeNow = TickCount(); ticksRemaining = ticksToSleep; endSleep = timeNow + ticksToSleep; while ( (timeNow < endSleep) && (ticksRemaining <= ticksToSleep) ) { usleep(16667 * ticksRemaining); timeNow = TickCount(); ticksRemaining = endSleep - timeNow; } } pid_t FindProcessPID(char* name, pid_t thePID) { FILE *f; char buf[1024]; size_t n = 0; pid_t aPID; if (name != NULL) // Search ny name n = strlen(name); f = popen("ps -a -x -c -o command,pid", "r"); if (f == NULL) { return 0; } while (PersistentFGets(buf, sizeof(buf), f)) { if (name != NULL) { // Search by name if (strncmp(buf, name, n) == 0) { aPID = atol(buf+16); pclose(f); return aPID; } } else { // Search by PID aPID = atol(buf+16); if (aPID == thePID) { pclose(f); return aPID; } } } pclose(f); return 0; } static OSErr QuitOneProcess(OSType signature) { bool done = false; ProcessSerialNumber thisPSN; ProcessInfoRec thisPIR; OSErr err = noErr; Str63 thisProcessName; AEAddressDesc thisPSNDesc; AppleEvent thisQuitEvent, thisReplyEvent; thisPIR.processInfoLength = sizeof (ProcessInfoRec); thisPIR.processName = thisProcessName; thisPIR.processAppSpec = nil; thisPSN.highLongOfPSN = 0; thisPSN.lowLongOfPSN = kNoProcess; while (done == false) { err = GetNextProcess(&thisPSN); if (err == procNotFound) { done = true; // Finished stepping through all running applications. err = noErr; // Success } else { err = GetProcessInformation(&thisPSN,&thisPIR); if (err != noErr) goto bail; if (thisPIR.processSignature == signature) { // is it our target process? err = AECreateDesc(typeProcessSerialNumber, (Ptr)&thisPSN, sizeof(thisPSN), &thisPSNDesc); if (err != noErr) goto bail; // Create the 'quit' Apple event for this process. err = AECreateAppleEvent(kCoreEventClass, kAEQuitApplication, &thisPSNDesc, kAutoGenerateReturnID, kAnyTransactionID, &thisQuitEvent); if (err != noErr) { AEDisposeDesc (&thisPSNDesc); goto bail; // don't know how this could happen, but limp gamely onward } // send the event err = AESend(&thisQuitEvent, &thisReplyEvent, kAEWaitReply, kAENormalPriority, kAEDefaultTimeout, 0L, 0L); AEDisposeDesc (&thisQuitEvent); AEDisposeDesc (&thisPSNDesc); if (err != noErr) goto bail; #if 0 if (err == errAETimeout) { pid_t thisPID; err = GetProcessPID(&thisPSN , &thisPID); if (err == noErr) err = kill(thisPID, SIGKILL); } #endif continue; // There can be multiple instances of the Manager } } } bail: return err; } static OSErr QuitAppleEventHandler( const AppleEvent *appleEvt, AppleEvent* reply, UInt32 refcon ) { gQuitFlag = true; return noErr; } void strip_cr(char *buf) { char *theCR; theCR = strrchr(buf, '\n'); if (theCR) *theCR = '\0'; theCR = strrchr(buf, '\r'); if (theCR) *theCR = '\0'; } // For debugging void print_to_log_file(const char *format, ...) { #if CREATE_LOG FILE *f; va_list args; char path[256], buf[256]; time_t t; strcpy(path, "/Users/Shared/test_log.txt"); // strcpy(path, "/Users/"); // strcat(path, getlogin()); // strcat(path, "/Documents/test_log.txt"); f = fopen(path, "a"); if (!f) return; // freopen(buf, "a", stdout); // freopen(buf, "a", stderr); time(&t); strcpy(buf, asctime(localtime(&t))); strip_cr(buf); fputs(buf, f); fputs(" ", f); va_start(args, format); vfprintf(f, format, args); va_end(args); fputs("\n", f); fflush(f); fclose(f); chmod(path, S_IRUSR|S_IWUSR|S_IRGRP|S_IWGRP|S_IROTH|S_IWOTH); #endif }