Skip to content

Commit

Permalink
Improve UDP logging while logger queue is suspended
Browse files Browse the repository at this point in the history
  • Loading branch information
tmolitor-stud-tu committed Dec 31, 2024
1 parent d6b1d7b commit 1bea74e
Show file tree
Hide file tree
Showing 5 changed files with 67 additions and 32 deletions.
5 changes: 5 additions & 0 deletions Monal/Classes/HelperTools.h
Original file line number Diff line number Diff line change
Expand Up @@ -85,6 +85,10 @@ void swizzle(Class c, SEL orig, SEL new);
-(id) initWithObj:(id) obj;
@end

@interface DDLogMessage(TaggedMessage)
@property (nonatomic) BOOL ml_isDirect;
@end

@interface HelperTools : NSObject

@property (class, nonatomic, strong, nullable) DDFileLogger* fileLogger;
Expand All @@ -94,6 +98,7 @@ void swizzle(Class c, SEL orig, SEL new);
+(void) installExceptionHandler;
+(int) pendingCrashreportCount;
+(void) flushLogsWithTimeout:(double) timeout;
+(BOOL) isAppSuspended;
+(void) signalSuspension;
+(void) signalResumption;
+(void) __attribute__((noreturn)) MLAssertWithText:(NSString*) text andUserData:(id _Nullable) additionalData andFile:(const char* const) file andLine:(int) line andFunc:(const char* const) func;
Expand Down
70 changes: 40 additions & 30 deletions Monal/Classes/HelperTools.m
Original file line number Diff line number Diff line change
Expand Up @@ -332,6 +332,18 @@ -(id) initWithObj:(id) obj
}
@end

@implementation DDLogMessage(TaggedMessage)
@dynamic ml_isDirect;
-(void) setMl_isDirect:(BOOL) value
{
objc_setAssociatedObject(self, @selector(ml_isDirect), @(value), OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
-(BOOL) ml_isDirect
{
return ((NSNumber*)objc_getAssociatedObject(self, @selector(ml_isDirect))).boolValue;
}
@end

@implementation NSUserDefaults (SerializeNSObject)
-(id) swizzled_objectForKey:(NSString*) defaultName
{
Expand Down Expand Up @@ -401,31 +413,22 @@ -(void) swizzled_queueLogMessage:(DDLogMessage*) logMessage asynchronously:(BOOL
udpLoggerEnabled = [[HelperTools defaultsDB] boolForKey:@"udpLoggerEnabled"];
});

//use udp logger to log all messages, even if the loggging queue is in suspended state
//this hopefully enables us to catch strange bugs sometimes hanging and then watchdog-killing the app when resuming from resumption
if(udpLoggerEnabled && _suspensionHandling_isSuspended)
{
//this marks a message as already directly logged to allow the udp logger to later ignore the queued log request for the same message
logMessage.ml_isDirect = YES;
//make sure all udp log messages are still logged chronologically and prevent race conditions with our static counter var
dispatch_async([MLUDPLogger getCurrentInstance].loggerQueue, ^{
[MLUDPLogger directlyWriteLogMessage:logMessage];
});
}

//don't do sync logging for any message (usually ERROR), while the global logging queue is suspended
//don't use _suspensionHandling_lock here because that can introduce deadlocks
//(for example if we have log statements in our MLLogFileManager code rotating the logfile and creating a new one)
BOOL isSuspended = _suspensionHandling_isSuspended;
if(isSuspended && udpLoggerEnabled)
{
//use udp logger to log all messages, even if the loggging queue is in suspended state
//this hopefully enables us to catch strange bugs sometimes hanging and then watchdog-killing the app when resuming from resumption
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wdeprecated-declarations"
[MLUDPLogger directlyWriteLogMessage:[[DDLogMessage alloc]
initWithFormat:[NSString stringWithFormat:@"+++ LOG_QUEUE_DISABLED+++ %@", logMessage.messageFormat]
formatted:[NSString stringWithFormat:@"+++ LOG_QUEUE_DISABLED+++ %@", logMessage.message]
level:logMessage.level
flag:logMessage.flag
context:logMessage.context
file:logMessage.file
function:logMessage.function
line:logMessage.line
tag:logMessage.tag
options:logMessage.options
timestamp:logMessage.timestamp]];
#pragma clang diagnostic pop
}
return [self swizzled_queueLogMessage:logMessage asynchronously:isSuspended ? YES : asyncFlag];
return [self swizzled_queueLogMessage:logMessage asynchronously:_suspensionHandling_isSuspended ? YES : asyncFlag];
}

//see https://stackoverflow.com/a/13326633 and https://fek.io/blog/method-swizzling-in-obj-c-and-swift/
Expand Down Expand Up @@ -2121,10 +2124,10 @@ +(NSData* _Nullable) convertLogmessageToJsonData:(DDLogMessage*) logMessage coun
@"queueThreadLabel": nilWrapper([self getQueueThreadLabelFor:logMessage]),
@"processType": [self isAppExtension] ? @"appex" : @"mainapp",
@"processName": nilWrapper([[[NSBundle mainBundle] executablePath] lastPathComponent]),
@"counter": [NSNumber numberWithUnsignedLongLong:*counter],
@"counter": @(*counter),
@"processID": nilWrapper(_processID),
@"qosName": nilWrapper(qos2name(logMessage.qos)),
@"loggingQueueSuspended": bool2str(_suspensionHandling_isSuspended),
@"loggingQueueSuspended": @(_suspensionHandling_isSuspended),
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wdeprecated-declarations"
@"tag": nilWrapper(logMessage.tag),
Expand All @@ -2133,21 +2136,21 @@ +(NSData* _Nullable) convertLogmessageToJsonData:(DDLogMessage*) logMessage coun
NSDictionary* msgDict = @{
@"messageFormat": nilWrapper(logMessage.messageFormat),
@"message": nilWrapper(logMessage.message),
@"level": [NSNumber numberWithInteger:logMessage.level],
@"flag": [NSNumber numberWithInteger:logMessage.flag],
@"context": [NSNumber numberWithInteger:logMessage.context],
@"level": @(logMessage.level),
@"flag": @(logMessage.flag),
@"context": @(logMessage.context),
@"file": nilWrapper(logMessage.file),
@"fileName": nilWrapper(logMessage.fileName),
@"function": nilWrapper(logMessage.function),
@"line": [NSNumber numberWithInteger:logMessage.line],
@"line": @(logMessage.line),
@"representedObject": nilWrapper(logMessage.representedObject),
@"tag": nilWrapper(tag),
@"options": [NSNumber numberWithInteger:logMessage.options],
@"options": @(logMessage.options),
@"timestamp": [dateFormatter stringFromDate:logMessage.timestamp],
@"threadID": nilWrapper(logMessage.threadID),
@"threadName": nilWrapper(logMessage.threadName),
@"queueLabel": nilWrapper(logMessage.queueLabel),
@"qos": [NSNumber numberWithInteger:logMessage.qos],
@"qos": @(logMessage.qos),
};

//encode json into NSData
Expand All @@ -2170,6 +2173,13 @@ +(void) flushLogsWithTimeout:(double) timeout
[MLUDPLogger flushWithTimeout:timeout];
}

+(BOOL) isAppSuspended
{
@synchronized(_suspensionHandling_lock) {
return _suspensionHandling_isSuspended;
}
}

+(void) signalSuspension
{
@synchronized(_suspensionHandling_lock) {
Expand Down
1 change: 1 addition & 0 deletions Monal/Classes/MLUDPLogger.h
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ FOUNDATION_EXPORT DDLoggerName const DDLoggerNameUDP NS_SWIFT_NAME(DDLoggerName.

+(void) flushWithTimeout:(double) timeout;
+(void) directlyWriteLogMessage:(DDLogMessage*) logMessage;
+(instancetype) getCurrentInstance;

@end

Expand Down
17 changes: 16 additions & 1 deletion Monal/Classes/MLUDPLogger.m
Original file line number Diff line number Diff line change
Expand Up @@ -87,7 +87,13 @@ +(void) directlyWriteLogMessage:(DDLogMessage*) logMessage
[[self class] logError:@"Ignoring call to directlySyncWriteLogMessage: _self still nil!"];
return;
}
return [_self logMessage:logMessage];
return [_self realLogMessage:logMessage isDirect:YES];
}

+(instancetype) getCurrentInstance
{
return (MLUDPLogger*)_self;

}

-(void) dealloc
Expand Down Expand Up @@ -243,13 +249,22 @@ -(void) createConnectionIfNeeded
}

-(void) logMessage:(DDLogMessage*) logMessage
{
return [self realLogMessage:logMessage isDirect:NO];
}

-(void) realLogMessage:(DDLogMessage*) logMessage isDirect:(BOOL) direct
{
static uint64_t counter = 0;

//early return if deactivated
if(![[HelperTools defaultsDB] boolForKey: @"udpLoggerEnabled"])
return;

//ignore log messages already udp-logged as direct messages when handling queued messages
if(!direct && logMessage.ml_isDirect)
return;

NSError* error = nil;
NSData* rawData = [HelperTools convertLogmessageToJsonData:logMessage counter:&counter andError:&error];
if(error != nil || rawData == nil)
Expand Down
6 changes: 5 additions & 1 deletion UDPLogServer/server.py
Original file line number Diff line number Diff line change
Expand Up @@ -161,7 +161,11 @@ def formatLogline(entry):
kwargs = flag_to_kwargs(decoded["flag"] if "flag" in decoded else None)

# print original formatted log message
logline = ("%d: %s" % (decoded["tag"]["counter"], formatLogline(decoded)))
logline = "%s%d: %s" % (
"+++ LOG_QUEUE_DISABLED +++ " if "tag" in decoded and "loggingQueueSuspended" in decoded["tag"] and decoded["tag"]["loggingQueueSuspended"]==True else "",
decoded["tag"]["counter"],
formatLogline(decoded)
)
if logfd:
print(logline, file=logfd)
print(colorize(logline, **kwargs), flush=True)
Expand Down

0 comments on commit 1bea74e

Please sign in to comment.