|  | @@ -108,6 +108,9 @@ static NSString * const kBearerPrefix = @"Bearer ";
 | 
											
												
													
														|  |    // The dispatch queue to be used for enqueuing responses to user. Defaulted to the main dispatch
 |  |    // The dispatch queue to be used for enqueuing responses to user. Defaulted to the main dispatch
 | 
											
												
													
														|  |    // queue
 |  |    // queue
 | 
											
												
													
														|  |    dispatch_queue_t _responseQueue;
 |  |    dispatch_queue_t _responseQueue;
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +  // Whether the call is finished. If it is, should not call finishWithError again.
 | 
											
												
													
														|  | 
 |  | +  BOOL _finished;
 | 
											
												
													
														|  |  }
 |  |  }
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |  @synthesize state = _state;
 |  |  @synthesize state = _state;
 | 
											
										
											
												
													
														|  | @@ -214,9 +217,10 @@ static NSString * const kBearerPrefix = @"Bearer ";
 | 
											
												
													
														|  |  }
 |  |  }
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |  - (void)cancel {
 |  |  - (void)cancel {
 | 
											
												
													
														|  | -  [self finishWithError:[NSError errorWithDomain:kGRPCErrorDomain
 |  | 
 | 
											
												
													
														|  | -                                            code:GRPCErrorCodeCancelled
 |  | 
 | 
											
												
													
														|  | -                                        userInfo:@{NSLocalizedDescriptionKey: @"Canceled by app"}]];
 |  | 
 | 
											
												
													
														|  | 
 |  | +  [self maybeFinishWithError:[NSError errorWithDomain:kGRPCErrorDomain
 | 
											
												
													
														|  | 
 |  | +                                                 code:GRPCErrorCodeCancelled
 | 
											
												
													
														|  | 
 |  | +                                             userInfo:@{NSLocalizedDescriptionKey: @"Canceled by app"}]];
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  |    if (!self.isWaitingForToken) {
 |  |    if (!self.isWaitingForToken) {
 | 
											
												
													
														|  |      [self cancelCall];
 |  |      [self cancelCall];
 | 
											
												
													
														|  |    } else {
 |  |    } else {
 | 
											
										
											
												
													
														|  | @@ -224,6 +228,19 @@ static NSString * const kBearerPrefix = @"Bearer ";
 | 
											
												
													
														|  |    }
 |  |    }
 | 
											
												
													
														|  |  }
 |  |  }
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  | 
 |  | +- (void)maybeFinishWithError:(NSError *)errorOrNil {
 | 
											
												
													
														|  | 
 |  | +  BOOL toFinish = NO;
 | 
											
												
													
														|  | 
 |  | +  @synchronized(self) {
 | 
											
												
													
														|  | 
 |  | +    if (_finished == NO) {
 | 
											
												
													
														|  | 
 |  | +      _finished = YES;
 | 
											
												
													
														|  | 
 |  | +      toFinish = YES;
 | 
											
												
													
														|  | 
 |  | +    }
 | 
											
												
													
														|  | 
 |  | +  }
 | 
											
												
													
														|  | 
 |  | +  if (toFinish == YES) {
 | 
											
												
													
														|  | 
 |  | +    [self finishWithError:errorOrNil];
 | 
											
												
													
														|  | 
 |  | +  }
 | 
											
												
													
														|  | 
 |  | +}
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  |  - (void)dealloc {
 |  |  - (void)dealloc {
 | 
											
												
													
														|  |    __block GRPCWrappedCall *wrappedCall = _wrappedCall;
 |  |    __block GRPCWrappedCall *wrappedCall = _wrappedCall;
 | 
											
												
													
														|  |    dispatch_async(_callQueue, ^{
 |  |    dispatch_async(_callQueue, ^{
 | 
											
										
											
												
													
														|  | @@ -268,9 +285,9 @@ static NSString * const kBearerPrefix = @"Bearer ";
 | 
											
												
													
														|  |          // don't want to throw, because the app shouldn't crash for a behavior
 |  |          // don't want to throw, because the app shouldn't crash for a behavior
 | 
											
												
													
														|  |          // that's on the hands of any server to have. Instead we finish and ask
 |  |          // that's on the hands of any server to have. Instead we finish and ask
 | 
											
												
													
														|  |          // the server to cancel.
 |  |          // the server to cancel.
 | 
											
												
													
														|  | -        [strongSelf finishWithError:[NSError errorWithDomain:kGRPCErrorDomain
 |  | 
 | 
											
												
													
														|  | -                                                        code:GRPCErrorCodeResourceExhausted
 |  | 
 | 
											
												
													
														|  | -                                                    userInfo:@{NSLocalizedDescriptionKey: @"Client does not have enough memory to hold the server response."}]];
 |  | 
 | 
											
												
													
														|  | 
 |  | +        [strongSelf maybeFinishWithError:[NSError errorWithDomain:kGRPCErrorDomain
 | 
											
												
													
														|  | 
 |  | +                                                             code:GRPCErrorCodeResourceExhausted
 | 
											
												
													
														|  | 
 |  | +                                                         userInfo:@{NSLocalizedDescriptionKey: @"Client does not have enough memory to hold the server response."}]];
 | 
											
												
													
														|  |          [strongSelf cancelCall];
 |  |          [strongSelf cancelCall];
 | 
											
												
													
														|  |          return;
 |  |          return;
 | 
											
												
													
														|  |        }
 |  |        }
 | 
											
										
											
												
													
														|  | @@ -340,9 +357,9 @@ static NSString * const kBearerPrefix = @"Bearer ";
 | 
											
												
													
														|  |      [self writeMessage:value withErrorHandler:^{
 |  |      [self writeMessage:value withErrorHandler:^{
 | 
											
												
													
														|  |        __strong GRPCCall *strongSelf = weakSelf;
 |  |        __strong GRPCCall *strongSelf = weakSelf;
 | 
											
												
													
														|  |        if (strongSelf != nil) {
 |  |        if (strongSelf != nil) {
 | 
											
												
													
														|  | -        [strongSelf finishWithError:[NSError errorWithDomain:kGRPCErrorDomain
 |  | 
 | 
											
												
													
														|  | -                                                        code:GRPCErrorCodeInternal
 |  | 
 | 
											
												
													
														|  | -                                                    userInfo:nil]];
 |  | 
 | 
											
												
													
														|  | 
 |  | +        [strongSelf maybeFinishWithError:[NSError errorWithDomain:kGRPCErrorDomain
 | 
											
												
													
														|  | 
 |  | +                                                             code:GRPCErrorCodeInternal
 | 
											
												
													
														|  | 
 |  | +                                                         userInfo:nil]];
 | 
											
												
													
														|  |          // Wrapped call must be canceled when error is reported to upper layers
 |  |          // Wrapped call must be canceled when error is reported to upper layers
 | 
											
												
													
														|  |          [strongSelf cancelCall];
 |  |          [strongSelf cancelCall];
 | 
											
												
													
														|  |        }
 |  |        }
 | 
											
										
											
												
													
														|  | @@ -371,9 +388,9 @@ static NSString * const kBearerPrefix = @"Bearer ";
 | 
											
												
													
														|  |        __weak GRPCCall *weakSelf = self;
 |  |        __weak GRPCCall *weakSelf = self;
 | 
											
												
													
														|  |        [self finishRequestWithErrorHandler:^{
 |  |        [self finishRequestWithErrorHandler:^{
 | 
											
												
													
														|  |          __strong GRPCCall *strongSelf = weakSelf;
 |  |          __strong GRPCCall *strongSelf = weakSelf;
 | 
											
												
													
														|  | -        [strongSelf finishWithError:[NSError errorWithDomain:kGRPCErrorDomain
 |  | 
 | 
											
												
													
														|  | -                                                        code:GRPCErrorCodeInternal
 |  | 
 | 
											
												
													
														|  | -                                                    userInfo:nil]];
 |  | 
 | 
											
												
													
														|  | 
 |  | +        [strongSelf maybeFinishWithError:[NSError errorWithDomain:kGRPCErrorDomain
 | 
											
												
													
														|  | 
 |  | +                                                             code:GRPCErrorCodeInternal
 | 
											
												
													
														|  | 
 |  | +                                                         userInfo:nil]];
 | 
											
												
													
														|  |          // Wrapped call must be canceled when error is reported to upper layers
 |  |          // Wrapped call must be canceled when error is reported to upper layers
 | 
											
												
													
														|  |          [strongSelf cancelCall];
 |  |          [strongSelf cancelCall];
 | 
											
												
													
														|  |        }];
 |  |        }];
 | 
											
										
											
												
													
														|  | @@ -426,7 +443,7 @@ static NSString * const kBearerPrefix = @"Bearer ";
 | 
											
												
													
														|  |          }
 |  |          }
 | 
											
												
													
														|  |          error = [NSError errorWithDomain:error.domain code:error.code userInfo:userInfo];
 |  |          error = [NSError errorWithDomain:error.domain code:error.code userInfo:userInfo];
 | 
											
												
													
														|  |        }
 |  |        }
 | 
											
												
													
														|  | -      [strongSelf finishWithError:error];
 |  | 
 | 
											
												
													
														|  | 
 |  | +      [strongSelf maybeFinishWithError:error];
 | 
											
												
													
														|  |      }
 |  |      }
 | 
											
												
													
														|  |    }];
 |  |    }];
 | 
											
												
													
														|  |    // Now that the RPC has been initiated, request writes can start.
 |  |    // Now that the RPC has been initiated, request writes can start.
 |