Skip to content

Commit 16a7c96

Browse files
committed
Add key to SCKillerHelper so it's harder to run manuallly + add error messages when killer fails
1 parent dac2159 commit 16a7c96

File tree

6 files changed

+102
-27
lines changed

6 files changed

+102
-27
lines changed

Common/Utility/SCMiscUtilities.h

+1
Original file line numberDiff line numberDiff line change
@@ -28,5 +28,6 @@
2828

2929
+ (BOOL)errorIsAuthCanceled:(NSError*)err;
3030

31+
+ (NSString*)killerKeyForDate:(NSDate*)date;
3132

3233
@end

Common/Utility/SCMiscUtilities.m

+4
Original file line numberDiff line numberDiff line change
@@ -227,4 +227,8 @@ + (BOOL)errorIsAuthCanceled:(NSError*)err {
227227
return homeDirectoryURLs;
228228
}
229229

230+
+ (NSString*)killerKeyForDate:(NSDate*)date {
231+
return [SCMiscUtilities sha1: [NSString stringWithFormat: @"SelfControlKillerKey%@%@", [SCMiscUtilities getSerialNumber], [date descriptionWithLocale: nil]]];
232+
}
233+
230234
@end

SCError.strings

+4
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,10 @@
3636

3737
// 400-499 = errors generated in the killer
3838
"400" = "SelfControl couldn't manually clear the block, because there was an error running the helper tool.";
39+
"401" = "Something went wrong, and SelfControl couldn't manually clear the block. Try again in a minute. If that doesn't work, go to www.selfcontrolapp.com/support to get help.";
40+
"402" = "The killer helper tool couldn't launch because it had insufficient privileges.";
41+
"403" = "The killer helper tool couldn't launch because it was provided the incorrect key.";
42+
"404" = "The killer helper tool couldn't launch because it was provided an invalid key date.";
3943

4044
// 500-599 = errors generated in the XPC client
4145
"500" = "There was an error trying to install SelfControl's helper tool: %@";

SCKillerHelper/main.m

+27-3
Original file line numberDiff line numberDiff line change
@@ -12,33 +12,57 @@
1212
#import "BlockManager.h"
1313
#import "SCSettings.h"
1414
#import "SCHelperToolUtilities.h"
15+
#import "SCMiscUtilities.h"
1516
#import <ServiceManagement/ServiceManagement.h>
1617
#import "SCMigrationUtilities.h"
1718
#import <sysexits.h>
19+
#import "SCSentry.h"
1820

1921
#define LOG_FILE @"~/Documents/SelfControl-Killer.log"
2022

2123
int main(int argc, char* argv[]) {
2224
@autoreleasepool {
25+
[SCSentry startSentry: @"com.selfcontrolapp.SCKillerHelper"];
26+
2327
// make sure to expand the tilde before we setuid to 0, otherwise this won't work
2428
NSString* logFilePath = [LOG_FILE stringByExpandingTildeInPath];
2529
NSMutableString* log = [NSMutableString stringWithString: @"===SelfControl-Killer Log File===\n\n"];
2630

2731
if(geteuid()) {
2832
NSLog(@"ERROR: Helper tool must be run as root.");
33+
[SCSentry captureError: [SCErr errorWithCode: 402]];
2934
exit(EXIT_FAILURE);
3035
}
3136

32-
if(argv[1] == NULL) {
37+
if(argv[1] == NULL || argv[2] == NULL) {
3338
NSLog(@"ERROR: Not enough arguments");
3439
exit(EXIT_FAILURE);
3540
}
41+
42+
NSString* killerKey = @(argv[1]);
43+
NSDate* keyDate = [[NSISO8601DateFormatter new] dateFromString: @(argv[2])];
44+
NSTimeInterval keyDateToNow = [[NSDate date] timeIntervalSinceDate: keyDate];
45+
46+
// key date must exist, not be in the future, and be in the past 10 seconds
47+
if (keyDate == nil || keyDateToNow < 0 || keyDateToNow > 10) {
48+
NSLog(@"ERROR: Key date invalid");
49+
[SCSentry captureError: [SCErr errorWithCode: 404]];
50+
exit(EX_USAGE);
51+
}
52+
53+
// keys must match
54+
NSString* correctKey = [SCMiscUtilities killerKeyForDate: keyDate];
55+
if (![correctKey isEqualToString: killerKey]) {
56+
NSLog(@"ERROR: Incorrect key");
57+
[SCSentry captureError: [SCErr errorWithCode: 403]];
58+
exit(EX_USAGE);
59+
}
3660

37-
uid_t controllingUID = (uid_t)[@(argv[1]) intValue];
61+
uid_t controllingUID = (uid_t)[@(argv[3]) intValue];
3862
if (controllingUID <= 0) {
3963
controllingUID = getuid();
4064
}
41-
65+
4266
// we need to setuid to root, otherwise launchctl won't find system launch daemons
4367
// depite the EUID being 0 as expected - not sure why that is
4468
setuid(0);

SelfControl Killer/AppDelegate.m

+38-9
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,8 @@
88

99
#import "AppDelegate.h"
1010
#import "SCSettings.h"
11+
#import "SCMiscUtilities.h"
12+
#import "SCUIUtilities.h"
1113

1214
@interface AppDelegate ()
1315

@@ -19,6 +21,8 @@ @implementation AppDelegate
1921
- (void)applicationDidFinishLaunching:(NSNotification *)aNotification {
2022
[NSApplication sharedApplication].delegate = self;
2123

24+
[SCSentry startSentry: @"com.selfcontrolapp.SelfControl-Killer"];
25+
2226
[self updateUserInterface];
2327
}
2428

@@ -60,14 +64,19 @@ - (IBAction)killButtonClicked:(id)sender {
6064

6165
char uidString[10];
6266
snprintf(uidString, sizeof(uidString), "%d", getuid());
67+
68+
NSDate* keyDate = [NSDate date];
69+
NSString* killerKey = [SCMiscUtilities killerKeyForDate: keyDate];
70+
NSString* keyDateString = [[NSISO8601DateFormatter new] stringFromDate: keyDate];
71+
72+
char* args[] = { (char*)[killerKey UTF8String], (char*)[keyDateString UTF8String], uidString, NULL };
6373

64-
char* args[] = { uidString, NULL };
65-
74+
FILE* pipe = NULL;
6675
status = AuthorizationExecuteWithPrivileges(authorizationRef,
6776
helperToolPath,
6877
kAuthorizationFlagDefaults,
6978
args,
70-
NULL);
79+
&pipe);
7180
if(status) {
7281
NSLog(@"WARNING: Authorized execution of helper tool returned failure status code %d", status);
7382

@@ -79,13 +88,33 @@ - (IBAction)killButtonClicked:(id)sender {
7988
}
8089

8190
return;
82-
} else {
83-
NSAlert* alert = [[NSAlert alloc] init];
84-
[alert setMessageText: @"Success!"];
85-
[alert setInformativeText:@"The block was cleared successfully. You can find the log file, named SelfControl-Killer.log, in your Documents folder."];
86-
[alert addButtonWithTitle: @"OK"];
87-
[alert runModal];
8891
}
92+
93+
// read until the pipe finishes so we wait for execution to end before we
94+
// show the modal (so we can check if the block is cleared properly or not)
95+
for (;;) {
96+
ssize_t bytesRead = read(fileno(pipe), NULL, 256);
97+
if (bytesRead < 1) break;
98+
}
99+
100+
// reload settings since they've probably just been messed with
101+
[[SCSettings sharedSettings] reloadSettings];
102+
103+
// send some debug info to Sentry to help us track this issue
104+
[SCSentry captureMessage: @"User manually cleared SelfControl block from the SelfControl Killer app"];
105+
106+
if ([SCBlockUtilities anyBlockIsRunning]) {
107+
// ruh roh! the block wasn't cleared successfully, since it's still running
108+
NSError* err = [SCErr errorWithCode: 401];
109+
[SCSentry captureError: err];
110+
[SCUIUtilities presentError: err];
111+
} else {
112+
NSAlert* alert = [[NSAlert alloc] init];
113+
[alert setMessageText: @"Success!"];
114+
[alert setInformativeText:@"The block was cleared successfully. You can find the log file, named SelfControl-Killer.log, in your Documents folder. If you're still having issues, please check out the SelfControl FAQ on GitHub."];
115+
[alert addButtonWithTitle: @"OK"];
116+
[alert runModal];
117+
}
89118

90119
[self.viewButton setEnabled: YES];
91120
}

TimerWindowController.m

+28-15
Original file line numberDiff line numberDiff line change
@@ -353,8 +353,11 @@ - (IBAction)killBlock:(id)sender {
353353
char uidString[10];
354354
snprintf(uidString, sizeof(uidString), "%d", getuid());
355355

356-
char* args[] = { uidString, NULL };
356+
NSDate* keyDate = [NSDate date];
357+
NSString* killerKey = [SCMiscUtilities killerKeyForDate: keyDate];
358+
NSString* keyDateString = [[NSISO8601DateFormatter new] stringFromDate: keyDate];
357359

360+
char* args[] = { (char*)[killerKey UTF8String], (char*)[keyDateString UTF8String], uidString, NULL };
358361

359362
FILE* pipe = NULL;
360363
status = AuthorizationExecuteWithPrivileges(authorizationRef,
@@ -366,13 +369,16 @@ - (IBAction)killBlock:(id)sender {
366369
if(status) {
367370
NSLog(@"WARNING: Authorized execution of helper tool returned failure status code %d", status);
368371

369-
NSError* err = [SCErr errorWithCode: 400];
370-
[SCSentry captureError: err];
371-
[SCUIUtilities presentError: err];
372+
/// AUTH_CANCELLED_STATUS just means auth is cancelled, not really an "error" per se
373+
if (status != AUTH_CANCELLED_STATUS) {
374+
NSError* err = [SCErr errorWithCode: 400];
375+
[SCSentry captureError: err];
376+
[SCUIUtilities presentError: err];
377+
}
372378

373379
return;
374380
}
375-
381+
376382
// read until the pipe finishes so we wait for execution to end before we
377383
// show the modal (this also helps make the focus ordering better)
378384
for (;;) {
@@ -391,18 +397,25 @@ - (IBAction)killBlock:(id)sender {
391397
waitUntilDone:YES];
392398

393399
// send some debug info to Sentry to help us track this issue
394-
// disabled for now because the best current method might collect user PII we don't want
400+
// detailed logs disabled for now because the best current method might collect user PII we don't want
395401
[SCSentry captureMessage: @"User manually cleared SelfControl block from the timer window"];
396-
// [SCSentry captureMessage: @"User manually cleared SelfControl block from the timer window" withScopeBlock:^(SentryScope * _Nonnull scope) {
397-
// SentryAttachment* fileAttachment = [[SentryAttachment alloc] initWithPath: [@"~/Documents/SelfControl-Killer.log" stringByExpandingTildeInPath]];
398-
// [scope addAttachment: fileAttachment];
399-
// }];
402+
// [SCSentry captureMessage: @"User manually cleared SelfControl block from the timer window" withScopeBlock:^(SentryScope * _Nonnull scope) {
403+
// SentryAttachment* fileAttachment = [[SentryAttachment alloc] initWithPath: [@"~/Documents/SelfControl-Killer.log" stringByExpandingTildeInPath]];
404+
// [scope addAttachment: fileAttachment];
405+
// }];
400406

401-
NSAlert* alert = [[NSAlert alloc] init];
402-
[alert setMessageText: @"Success!"];
403-
[alert setInformativeText:@"The block was cleared successfully. You can find the log file, named SelfControl-Killer.log, in your Documents folder. If you're still having issues, please check out the SelfControl FAQ on GitHub."];
404-
[alert addButtonWithTitle: @"OK"];
405-
[alert runModal];
407+
if ([SCBlockUtilities anyBlockIsRunning]) {
408+
// ruh roh! the block wasn't cleared successfully, since it's still running
409+
NSError* err = [SCErr errorWithCode: 401];
410+
[SCSentry captureError: err];
411+
[SCUIUtilities presentError: err];
412+
} else {
413+
NSAlert* alert = [[NSAlert alloc] init];
414+
[alert setMessageText: @"Success!"];
415+
[alert setInformativeText:@"The block was cleared successfully. You can find the log file, named SelfControl-Killer.log, in your Documents folder. If you're still having issues, please check out the SelfControl FAQ on GitHub."];
416+
[alert addButtonWithTitle: @"OK"];
417+
[alert runModal];
418+
}
406419
}
407420

408421
- (NSString*)selfControlKillerHelperToolPath {

0 commit comments

Comments
 (0)