1/* 2 * Copyright 2018 The WebRTC project authors. All Rights Reserved. 3 * 4 * Use of this source code is governed by a BSD-style license 5 * that can be found in the LICENSE file in the root of the source 6 * tree. An additional intellectual property rights grant can be found 7 * in the file PATENTS. All contributing project authors may 8 * be found in the AUTHORS file in the root of the source tree. 9 */ 10 11#import "RTCCallbackLogger.h" 12 13#include <memory> 14 15#include "rtc_base/checks.h" 16#include "rtc_base/log_sinks.h" 17#include "rtc_base/logging.h" 18 19class CallbackLogSink : public rtc::LogSink { 20 public: 21 CallbackLogSink(RTCCallbackLoggerMessageHandler callbackHandler) 22 : callback_handler_(callbackHandler) {} 23 24 void OnLogMessage(const std::string &message) override { 25 if (callback_handler_) { 26 callback_handler_([NSString stringWithUTF8String:message.c_str()]); 27 } 28 } 29 30 private: 31 RTCCallbackLoggerMessageHandler callback_handler_; 32}; 33 34class CallbackWithSeverityLogSink : public rtc::LogSink { 35 public: 36 CallbackWithSeverityLogSink(RTCCallbackLoggerMessageAndSeverityHandler callbackHandler) 37 : callback_handler_(callbackHandler) {} 38 39 void OnLogMessage(const std::string& message) override { RTC_NOTREACHED(); } 40 41 void OnLogMessage(const std::string& message, rtc::LoggingSeverity severity) override { 42 if (callback_handler_) { 43 RTCLoggingSeverity loggingSeverity = NativeSeverityToObjcSeverity(severity); 44 callback_handler_([NSString stringWithUTF8String:message.c_str()], loggingSeverity); 45 } 46 } 47 48 private: 49 static RTCLoggingSeverity NativeSeverityToObjcSeverity(rtc::LoggingSeverity severity) { 50 switch (severity) { 51 case rtc::LS_VERBOSE: 52 return RTCLoggingSeverityVerbose; 53 case rtc::LS_INFO: 54 return RTCLoggingSeverityInfo; 55 case rtc::LS_WARNING: 56 return RTCLoggingSeverityWarning; 57 case rtc::LS_ERROR: 58 return RTCLoggingSeverityError; 59 case rtc::LS_NONE: 60 return RTCLoggingSeverityNone; 61 } 62 } 63 64 RTCCallbackLoggerMessageAndSeverityHandler callback_handler_; 65}; 66 67@implementation RTC_OBJC_TYPE (RTCCallbackLogger) { 68 BOOL _hasStarted; 69 std::unique_ptr<rtc::LogSink> _logSink; 70} 71 72@synthesize severity = _severity; 73 74- (instancetype)init { 75 self = [super init]; 76 if (self != nil) { 77 _severity = RTCLoggingSeverityInfo; 78 } 79 return self; 80} 81 82- (void)dealloc { 83 [self stop]; 84} 85 86- (void)start:(nullable RTCCallbackLoggerMessageHandler)handler { 87 if (_hasStarted) { 88 return; 89 } 90 91 _logSink.reset(new CallbackLogSink(handler)); 92 93 rtc::LogMessage::AddLogToStream(_logSink.get(), [self rtcSeverity]); 94 _hasStarted = YES; 95} 96 97- (void)startWithMessageAndSeverityHandler: 98 (nullable RTCCallbackLoggerMessageAndSeverityHandler)handler { 99 if (_hasStarted) { 100 return; 101 } 102 103 _logSink.reset(new CallbackWithSeverityLogSink(handler)); 104 105 rtc::LogMessage::AddLogToStream(_logSink.get(), [self rtcSeverity]); 106 _hasStarted = YES; 107} 108 109- (void)stop { 110 if (!_hasStarted) { 111 return; 112 } 113 RTC_DCHECK(_logSink); 114 rtc::LogMessage::RemoveLogToStream(_logSink.get()); 115 _hasStarted = NO; 116 _logSink.reset(); 117} 118 119#pragma mark - Private 120 121- (rtc::LoggingSeverity)rtcSeverity { 122 switch (_severity) { 123 case RTCLoggingSeverityVerbose: 124 return rtc::LS_VERBOSE; 125 case RTCLoggingSeverityInfo: 126 return rtc::LS_INFO; 127 case RTCLoggingSeverityWarning: 128 return rtc::LS_WARNING; 129 case RTCLoggingSeverityError: 130 return rtc::LS_ERROR; 131 case RTCLoggingSeverityNone: 132 return rtc::LS_NONE; 133 } 134} 135 136@end 137