OIDErrorUtilities.m 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172
  1. /*! @file OIDErrorUtilities.m
  2. @brief AppAuth iOS SDK
  3. @copyright
  4. Copyright 2015 Google Inc. All Rights Reserved.
  5. @copydetails
  6. Licensed under the Apache License, Version 2.0 (the "License");
  7. you may not use this file except in compliance with the License.
  8. You may obtain a copy of the License at
  9. http://www.apache.org/licenses/LICENSE-2.0
  10. Unless required by applicable law or agreed to in writing, software
  11. distributed under the License is distributed on an "AS IS" BASIS,
  12. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. See the License for the specific language governing permissions and
  14. limitations under the License.
  15. */
  16. #import "OIDErrorUtilities.h"
  17. @implementation OIDErrorUtilities
  18. + (NSError *)errorWithCode:(OIDErrorCode)code
  19. underlyingError:(NSError *)underlyingError
  20. description:(NSString *)description {
  21. NSMutableDictionary *userInfo = [NSMutableDictionary dictionary];
  22. if (underlyingError) {
  23. userInfo[NSUnderlyingErrorKey] = underlyingError;
  24. }
  25. if (description) {
  26. userInfo[NSLocalizedDescriptionKey] = description;
  27. }
  28. // TODO: Populate localized description based on code.
  29. NSError *error = [NSError errorWithDomain:OIDGeneralErrorDomain
  30. code:code
  31. userInfo:userInfo];
  32. return error;
  33. }
  34. + (BOOL)isOAuthErrorDomain:(NSString *)errorDomain {
  35. return errorDomain == OIDOAuthRegistrationErrorDomain
  36. || errorDomain == OIDOAuthAuthorizationErrorDomain
  37. || errorDomain == OIDOAuthTokenErrorDomain;
  38. }
  39. + (NSError *)resourceServerAuthorizationErrorWithCode:(NSInteger)code
  40. errorResponse:(nullable NSDictionary *)errorResponse
  41. underlyingError:(nullable NSError *)underlyingError {
  42. // builds the userInfo dictionary with the full OAuth response and other information
  43. NSMutableDictionary *userInfo = [NSMutableDictionary dictionary];
  44. if (errorResponse) {
  45. userInfo[OIDOAuthErrorResponseErrorKey] = errorResponse;
  46. }
  47. if (underlyingError) {
  48. userInfo[NSUnderlyingErrorKey] = underlyingError;
  49. }
  50. NSError *error = [NSError errorWithDomain:OIDResourceServerAuthorizationErrorDomain
  51. code:code
  52. userInfo:userInfo];
  53. return error;
  54. }
  55. + (NSError *)OAuthErrorWithDomain:(NSString *)oAuthErrorDomain
  56. OAuthResponse:(NSDictionary *)errorResponse
  57. underlyingError:(NSError *)underlyingError {
  58. // not a valid OAuth error
  59. if (![self isOAuthErrorDomain:oAuthErrorDomain]
  60. || !errorResponse
  61. || !errorResponse[OIDOAuthErrorFieldError]
  62. || ![errorResponse[OIDOAuthErrorFieldError] isKindOfClass:[NSString class]]) {
  63. return [[self class] errorWithCode:OIDErrorCodeNetworkError
  64. underlyingError:underlyingError
  65. description:underlyingError.localizedDescription];
  66. }
  67. // builds the userInfo dictionary with the full OAuth response and other information
  68. NSMutableDictionary *userInfo = [NSMutableDictionary dictionary];
  69. userInfo[OIDOAuthErrorResponseErrorKey] = errorResponse;
  70. if (underlyingError) {
  71. userInfo[NSUnderlyingErrorKey] = underlyingError;
  72. }
  73. NSString *oauthErrorCodeString = errorResponse[OIDOAuthErrorFieldError];
  74. NSString *oauthErrorMessage = nil;
  75. if ([errorResponse[OIDOAuthErrorFieldErrorDescription] isKindOfClass:[NSString class]]) {
  76. oauthErrorMessage = errorResponse[OIDOAuthErrorFieldErrorDescription];
  77. } else {
  78. oauthErrorMessage = [errorResponse[OIDOAuthErrorFieldErrorDescription] description];
  79. }
  80. NSString *oauthErrorURI = nil;
  81. if ([errorResponse[OIDOAuthErrorFieldErrorURI] isKindOfClass:[NSString class]]) {
  82. oauthErrorURI = errorResponse[OIDOAuthErrorFieldErrorURI];
  83. } else {
  84. oauthErrorURI = [errorResponse[OIDOAuthErrorFieldErrorURI] description];
  85. }
  86. // builds the error description, using the information supplied by the server if possible
  87. NSMutableString *description = [NSMutableString string];
  88. [description appendString:oauthErrorCodeString];
  89. if (oauthErrorMessage) {
  90. [description appendString:@": "];
  91. [description appendString:oauthErrorMessage];
  92. }
  93. if (oauthErrorURI) {
  94. if ([description length] > 0) {
  95. [description appendString:@" - "];
  96. }
  97. [description appendString:oauthErrorURI];
  98. }
  99. if ([description length] == 0) {
  100. // backup description
  101. [description appendFormat:@"OAuth error: %@ - https://tools.ietf.org/html/rfc6749#section-5.2",
  102. oauthErrorCodeString];
  103. }
  104. userInfo[NSLocalizedDescriptionKey] = description;
  105. // looks up the error code based on the "error" response param
  106. OIDErrorCodeOAuth code = [[self class] OAuthErrorCodeFromString:oauthErrorCodeString];
  107. NSError *error = [NSError errorWithDomain:oAuthErrorDomain
  108. code:code
  109. userInfo:userInfo];
  110. return error;
  111. }
  112. + (NSError *)HTTPErrorWithHTTPResponse:(NSHTTPURLResponse *)HTTPURLResponse
  113. data:(nullable NSData *)data {
  114. NSMutableDictionary *userInfo = [NSMutableDictionary dictionary];
  115. if (data) {
  116. NSString *serverResponse =
  117. [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
  118. if (serverResponse) {
  119. userInfo[NSLocalizedDescriptionKey] = serverResponse;
  120. }
  121. }
  122. NSError *serverError =
  123. [NSError errorWithDomain:OIDHTTPErrorDomain
  124. code:HTTPURLResponse.statusCode
  125. userInfo:userInfo];
  126. return serverError;
  127. }
  128. + (OIDErrorCodeOAuth)OAuthErrorCodeFromString:(NSString *)errorCode {
  129. NSDictionary *errorCodes = @{
  130. @"invalid_request": @(OIDErrorCodeOAuthInvalidRequest),
  131. @"unauthorized_client": @(OIDErrorCodeOAuthUnauthorizedClient),
  132. @"access_denied": @(OIDErrorCodeOAuthAccessDenied),
  133. @"unsupported_response_type": @(OIDErrorCodeOAuthUnsupportedResponseType),
  134. @"invalid_scope": @(OIDErrorCodeOAuthInvalidScope),
  135. @"server_error": @(OIDErrorCodeOAuthServerError),
  136. @"temporarily_unavailable": @(OIDErrorCodeOAuthTemporarilyUnavailable),
  137. @"invalid_client": @(OIDErrorCodeOAuthInvalidClient),
  138. @"invalid_grant": @(OIDErrorCodeOAuthInvalidGrant),
  139. @"unsupported_grant_type": @(OIDErrorCodeOAuthUnsupportedGrantType),
  140. };
  141. NSNumber *code = errorCodes[errorCode];
  142. if (code) {
  143. return [code integerValue];
  144. } else {
  145. return OIDErrorCodeOAuthOther;
  146. }
  147. }
  148. + (void)raiseException:(NSString *)name {
  149. [[self class] raiseException:name message:name];
  150. }
  151. + (void)raiseException:(NSString *)name message:(NSString *)message {
  152. [NSException raise:name format:@"%@", message];
  153. }
  154. @end