/* * Copyright [2019] [Doric.Pub] * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ // // DoricNativeDriver.m // Doric // // Created by pengfei.zhou on 2019/7/26. // #import "DoricNativeDriver.h" #import "DoricJSEngine.h" #import "DoricConstant.h" #import "DoricContextManager.h" #import "DoricPerformanceProfile.h" @interface DoricNativeDriver () @property(nonatomic, strong) DoricJSEngine *jsExecutor; @end @implementation DoricNativeDriver @dynamic registry; - (instancetype)init { if (self = [super init]) { _jsExecutor = [[DoricJSEngine alloc] init]; } return self; } - (DoricRegistry *)registry { return self.jsExecutor.registry; } + (instancetype)instance { static DoricNativeDriver *_instance; static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{ _instance = [[DoricNativeDriver alloc] init]; }); return _instance; } - (DoricAsyncResult *)invokeDoricMethod:(NSString *)method argumentsArray:(NSArray *)args { DoricAsyncResult *ret = [[DoricAsyncResult alloc] init]; __weak typeof(self) _self = self; [self.jsExecutor ensureRunOnJSThread:^{ __strong typeof(_self) self = _self; if (!self) return; @try { JSValue *jsValue = [self.jsExecutor invokeDoricMethod:method argumentsArray:args]; [ret setupResult:jsValue]; } @catch (NSException *exception) { [ret setupError:exception]; } }]; return ret; } - (DoricAsyncResult *)invokeDoricMethod:(NSString *)method, ... { va_list args; va_start(args, method); DoricAsyncResult *ret = [self invokeDoricMethod:method arguments:args]; va_end(args); return ret; } - (DoricAsyncResult *)invokeDoricMethod:(NSString *)method arguments:(va_list)args { DoricAsyncResult *ret = [[DoricAsyncResult alloc] init]; NSMutableArray *array = [[NSMutableArray alloc] init]; id arg; while ((arg = va_arg(args, id)) != nil) { [array addObject:arg]; } __weak typeof(self) _self = self; [self.jsExecutor ensureRunOnJSThread:^{ __strong typeof(_self) self = _self; if (!self) return; @try { JSValue *jsValue = [self.jsExecutor invokeDoricMethod:method argumentsArray:array]; [ret setupResult:jsValue]; } @catch (NSException *exception) { [ret setupError:exception]; } }]; return ret; } - (DoricAsyncResult *)invokeContextEntity:(NSString *)contextId method:(NSString *)method, ... { va_list args; va_start(args, method); DoricAsyncResult *ret = [self invokeContextEntity:contextId method:method arguments:args]; va_end(args); return ret; } - (DoricAsyncResult *)invokeContextEntity:(NSString *)contextId method:(NSString *)method arguments:(va_list)args { DoricAsyncResult *ret = [[DoricAsyncResult alloc] init]; NSMutableArray *array = [[NSMutableArray alloc] init]; NSMutableArray *printedArgs = [[NSMutableArray alloc] init]; [array addObject:contextId]; [array addObject:method]; [printedArgs addObject:method]; id arg = va_arg(args, id); while (arg != nil) { [array addObject:arg]; [printedArgs addObject:arg]; arg = va_arg(args, JSValue *); } DoricPerformanceProfile *performanceProfile = [DoricContextManager.instance getContext:contextId].performanceProfile; NSString *anchorName = [NSString stringWithFormat:@"call:%@", [printedArgs componentsJoinedByString:@","]]; [performanceProfile prepare:anchorName]; __weak typeof(self) _self = self; [self.jsExecutor ensureRunOnJSThread:^{ __strong typeof(_self) self = _self; if (!self) return; @try { [performanceProfile start:anchorName]; JSValue *jsValue = [self.jsExecutor invokeDoricMethod:DORIC_CONTEXT_INVOKE argumentsArray:array]; [ret setupResult:jsValue]; [performanceProfile end:anchorName]; } @catch (NSException *exception) { [ret setupError:exception]; [self.jsExecutor.registry onException:exception inContext:[[DoricContextManager instance] getContext:contextId]]; } }]; return ret; } - (DoricAsyncResult *)invokeContextEntity:(NSString *)contextId method:(NSString *)method argumentsArray:(NSArray *)args { DoricAsyncResult *ret = [[DoricAsyncResult alloc] init]; NSMutableArray *array = [[NSMutableArray alloc] init]; [array addObject:contextId]; [array addObject:method]; for (id arg in args) { [array addObject:arg]; } DoricPerformanceProfile *performanceProfile = [DoricContextManager.instance getContext:contextId].performanceProfile; NSString *anchorName = [NSString stringWithFormat:@"call:%@,%@", method, [args componentsJoinedByString:@","]]; [performanceProfile prepare:anchorName]; __weak typeof(self) _self = self; [self.jsExecutor ensureRunOnJSThread:^{ __strong typeof(_self) self = _self; if (!self) return; @try { [performanceProfile start:anchorName]; JSValue *jsValue = [self.jsExecutor invokeDoricMethod:DORIC_CONTEXT_INVOKE argumentsArray:array]; [ret setupResult:jsValue]; [performanceProfile end:anchorName]; } @catch (NSException *exception) { [ret setupError:exception]; [self.jsExecutor.registry onException:exception inContext:[[DoricContextManager instance] getContext:contextId]]; } }]; return ret; } - (DoricAsyncResult *)createContext:(NSString *)contextId script:(NSString *)script source:(NSString *)source { DoricAsyncResult *ret = [[DoricAsyncResult alloc] init]; DoricPerformanceProfile *performanceProfile = [DoricContextManager.instance getContext:contextId].performanceProfile; [performanceProfile prepare:@"Create"]; __weak typeof(self) _self = self; [self.jsExecutor ensureRunOnJSThread:^{ __strong typeof(_self) self = _self; if (!self) return; @try { [performanceProfile start:@"Create"]; [self.jsExecutor prepareContext:contextId script:script source:source]; [ret setupResult:@YES]; [performanceProfile end:@"Create"]; } @catch (NSException *exception) { [ret setupError:exception]; [self.registry onException:exception inContext:[[DoricContextManager instance] getContext:contextId]]; } }]; return ret; } - (DoricAsyncResult *)destroyContext:(NSString *)contextId { DoricAsyncResult *ret = [[DoricAsyncResult alloc] init]; DoricPerformanceProfile *performanceProfile = [DoricContextManager.instance getContext:contextId].performanceProfile; [performanceProfile prepare:@"Destroy"]; __weak typeof(self) _self = self; [self.jsExecutor ensureRunOnJSThread:^{ __strong typeof(_self) self = _self; if (!self) return; @try { [performanceProfile start:@"Destroy"]; [self.jsExecutor destroyContext:contextId]; [ret setupResult:@YES]; [performanceProfile end:@"Destroy"]; } @catch (NSException *exception) { [ret setupError:exception]; [self.jsExecutor.registry onException:exception inContext:[[DoricContextManager instance] getContext:contextId]]; } }]; return ret; } - (void)ensureSyncInMainQueue:(dispatch_block_t)block { if (strcmp(dispatch_queue_get_label(DISPATCH_CURRENT_QUEUE_LABEL), dispatch_queue_get_label(dispatch_get_main_queue())) == 0) { block(); } else { dispatch_async(dispatch_get_main_queue(), block); } } - (void)dealloc { [self.jsExecutor teardown]; } @end