chromium/third_party/ocmock/OCMock/OCMInvocationStub.m

/*
 *  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