| /* |
| * Copyright (c) 2014-2021 Erik Doernenburg and contributors |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); you may |
| * not use these files 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. |
| */ |
| |
| #import "NSInvocation+OCMAdditions.h" |
| #import "OCMInvocationStub.h" |
| #import "OCMArg.h" |
| #import "OCMArgAction.h" |
| |
| #define UNSET_RETURN_VALUE_MARKER ((id)0x01234567) |
| |
| |
| @implementation OCMInvocationStub |
| |
| - (id)init |
| { |
| self = [super init]; |
| invocationActions = [[NSMutableArray alloc] init]; |
| return self; |
| } |
| |
| - (void)dealloc |
| { |
| [invocationActions release]; |
| [super dealloc]; |
| } |
| |
| |
| - (void)addInvocationAction:(id)anAction |
| { |
| [invocationActions addObject:anAction]; |
| } |
| |
| - (NSArray *)invocationActions |
| { |
| return invocationActions; |
| } |
| |
| |
| - (void)handleInvocation:(NSInvocation *)anInvocation |
| { |
| [self invokeArgActionsForInvocation:anInvocation]; |
| id target = [anInvocation target]; |
| |
| BOOL isInInitFamily = [anInvocation methodIsInInitFamily]; |
| BOOL isInCreateFamily = isInInitFamily ? NO : [anInvocation methodIsInCreateFamily]; |
| if(isInInitFamily || isInCreateFamily) |
| { |
| id returnVal = UNSET_RETURN_VALUE_MARKER; |
| [anInvocation setReturnValue:&returnVal]; |
| |
| [self invokeActionsForInvocation:anInvocation]; |
| |
| [anInvocation getReturnValue:&returnVal]; |
| if(returnVal == UNSET_RETURN_VALUE_MARKER) |
| [NSException raise:NSInvalidArgumentException format:@"%@ was stubbed but no return value set. A return value is required for all alloc/copy/new/mutablecopy/init methods. If you intended to return nil, make this explicit with .andReturn(nil)", NSStringFromSelector([anInvocation selector])]; |
| |
| if(isInCreateFamily) |
| { |
| // methods that "create" an object return it with an extra retain count |
| [returnVal retain]; |
| } |
| if(isInInitFamily) |
| { |
| // init family methods "consume" self and retain their return value. Do the retain |
| // first in case the return value and self are the same. The analyzer doesn't |
| // understand this; see #456 for details. In this case we also need to do something |
| // harmless with target or else the analyzer will complain about it not being used. |
| [returnVal retain]; |
| #ifndef __clang_analyzer__ |
| [target release]; |
| #else |
| [target class]; |
| #endif |
| } |
| } |
| else |
| { |
| [self invokeActionsForInvocation:anInvocation]; |
| } |
| } |
| |
| - (void)invokeArgActionsForInvocation:(NSInvocation *)anInvocation |
| { |
| NSMethodSignature *signature = [recordedInvocation methodSignature]; |
| NSUInteger n = [signature numberOfArguments]; |
| for(NSUInteger i = 2; i < n; i++) |
| { |
| id recordedArg = [recordedInvocation getArgumentAtIndexAsObject:i]; |
| id passedArg = [anInvocation getArgumentAtIndexAsObject:i]; |
| |
| if([recordedArg isProxy]) |
| continue; |
| |
| if([recordedArg isKindOfClass:[NSValue class]]) |
| recordedArg = [OCMArg resolveSpecialValues:recordedArg]; |
| |
| if([recordedArg isKindOfClass:[OCMArgAction class]]) |
| [recordedArg handleArgument:passedArg]; |
| } |
| } |
| |
| - (void)invokeActionsForInvocation:(NSInvocation *)anInvocation |
| { |
| [invocationActions makeObjectsPerformSelector:@selector(handleInvocation:) withObject:anInvocation]; |
| } |
| |
| |
| @end |