OIDServiceDiscovery.m 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362
  1. /*! @file OIDServiceDiscovery.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 "OIDServiceDiscovery.h"
  17. #import "OIDDefines.h"
  18. #import "OIDErrorUtilities.h"
  19. NS_ASSUME_NONNULL_BEGIN
  20. /*! Field keys associated with an OpenID Connect Discovery Document. */
  21. static NSString *const kIssuerKey = @"issuer";
  22. static NSString *const kAuthorizationEndpointKey = @"authorization_endpoint";
  23. static NSString *const kTokenEndpointKey = @"token_endpoint";
  24. static NSString *const kUserinfoEndpointKey = @"userinfo_endpoint";
  25. static NSString *const kJWKSURLKey = @"jwks_uri";
  26. static NSString *const kRegistrationEndpointKey = @"registration_endpoint";
  27. static NSString *const kEndSessionEndpointKey = @"end_session_endpoint";
  28. static NSString *const kScopesSupportedKey = @"scopes_supported";
  29. static NSString *const kResponseTypesSupportedKey = @"response_types_supported";
  30. static NSString *const kResponseModesSupportedKey = @"response_modes_supported";
  31. static NSString *const kGrantTypesSupportedKey = @"grant_types_supported";
  32. static NSString *const kACRValuesSupportedKey = @"acr_values_supported";
  33. static NSString *const kSubjectTypesSupportedKey = @"subject_types_supported";
  34. static NSString *const kIDTokenSigningAlgorithmValuesSupportedKey =
  35. @"id_token_signing_alg_values_supported";
  36. static NSString *const kIDTokenEncryptionAlgorithmValuesSupportedKey =
  37. @"id_token_encryption_alg_values_supported";
  38. static NSString *const kIDTokenEncryptionEncodingValuesSupportedKey =
  39. @"id_token_encryption_enc_values_supported";
  40. static NSString *const kUserinfoSigningAlgorithmValuesSupportedKey =
  41. @"userinfo_signing_alg_values_supported";
  42. static NSString *const kUserinfoEncryptionAlgorithmValuesSupportedKey =
  43. @"userinfo_encryption_alg_values_supported";
  44. static NSString *const kUserinfoEncryptionEncodingValuesSupportedKey =
  45. @"userinfo_encryption_enc_values_supported";
  46. static NSString *const kRequestObjectSigningAlgorithmValuesSupportedKey =
  47. @"request_object_signing_alg_values_supported";
  48. static NSString *const kRequestObjectEncryptionAlgorithmValuesSupportedKey =
  49. @"request_object_encryption_alg_values_supported";
  50. static NSString *const kRequestObjectEncryptionEncodingValuesSupported =
  51. @"request_object_encryption_enc_values_supported";
  52. static NSString *const kTokenEndpointAuthMethodsSupportedKey =
  53. @"token_endpoint_auth_methods_supported";
  54. static NSString *const kTokenEndpointAuthSigningAlgorithmValuesSupportedKey =
  55. @"token_endpoint_auth_signing_alg_values_supported";
  56. static NSString *const kDisplayValuesSupportedKey = @"display_values_supported";
  57. static NSString *const kClaimTypesSupportedKey = @"claim_types_supported";
  58. static NSString *const kClaimsSupportedKey = @"claims_supported";
  59. static NSString *const kServiceDocumentationKey = @"service_documentation";
  60. static NSString *const kClaimsLocalesSupportedKey = @"claims_locales_supported";
  61. static NSString *const kUILocalesSupportedKey = @"ui_locales_supported";
  62. static NSString *const kClaimsParameterSupportedKey = @"claims_parameter_supported";
  63. static NSString *const kRequestParameterSupportedKey = @"request_parameter_supported";
  64. static NSString *const kRequestURIParameterSupportedKey = @"request_uri_parameter_supported";
  65. static NSString *const kRequireRequestURIRegistrationKey = @"require_request_uri_registration";
  66. static NSString *const kOPPolicyURIKey = @"op_policy_uri";
  67. static NSString *const kOPTosURIKey = @"op_tos_uri";
  68. @implementation OIDServiceDiscovery {
  69. NSDictionary *_discoveryDictionary;
  70. }
  71. - (nonnull instancetype)init OID_UNAVAILABLE_USE_INITIALIZER(@selector(initWithDictionary:error:))
  72. - (nullable instancetype)initWithJSON:(NSString *)serviceDiscoveryJSON error:(NSError **)error {
  73. NSData *jsonData = [serviceDiscoveryJSON dataUsingEncoding:NSUTF8StringEncoding];
  74. return [self initWithJSONData:jsonData error:error];
  75. }
  76. - (nullable instancetype)initWithJSONData:(NSData *)serviceDiscoveryJSONData
  77. error:(NSError **_Nullable)error {
  78. NSError *jsonError;
  79. NSDictionary *json =
  80. [NSJSONSerialization JSONObjectWithData:serviceDiscoveryJSONData options:0 error:&jsonError];
  81. if (!json || jsonError) {
  82. *error = [OIDErrorUtilities errorWithCode:OIDErrorCodeJSONDeserializationError
  83. underlyingError:jsonError
  84. description:jsonError.localizedDescription];
  85. return nil;
  86. }
  87. if (![json isKindOfClass:[NSDictionary class]]) {
  88. *error = [OIDErrorUtilities errorWithCode:OIDErrorCodeInvalidDiscoveryDocument
  89. underlyingError:nil
  90. description:@"Discovery document isn't a dictionary"];
  91. return nil;
  92. }
  93. return [self initWithDictionary:json error:error];
  94. }
  95. - (nullable instancetype)initWithDictionary:(NSDictionary *)serviceDiscoveryDictionary
  96. error:(NSError **_Nullable)error {
  97. if (![[self class] dictionaryHasRequiredFields:serviceDiscoveryDictionary error:error]) {
  98. return nil;
  99. }
  100. self = [super init];
  101. if (self) {
  102. _discoveryDictionary = [serviceDiscoveryDictionary copy];
  103. }
  104. return self;
  105. }
  106. #pragma mark -
  107. /*! @brief Checks to see if the specified dictionary contains the required fields.
  108. @discussion This test is not meant to provide semantic analysis of the document (eg. fields
  109. where the value @c none is not an allowed option would not cause this method to fail if
  110. their value was @c none.) We are just testing to make sure we can meet the nullability
  111. contract we promised in the header.
  112. */
  113. + (BOOL)dictionaryHasRequiredFields:(NSDictionary<NSString *, id> *)dictionary
  114. error:(NSError **_Nullable)error {
  115. static NSString *const kMissingFieldErrorText = @"Missing field: %@";
  116. static NSString *const kInvalidURLFieldErrorText = @"Invalid URL: %@";
  117. NSArray *requiredFields = @[
  118. kIssuerKey,
  119. kAuthorizationEndpointKey,
  120. kTokenEndpointKey,
  121. kJWKSURLKey,
  122. kResponseTypesSupportedKey,
  123. kSubjectTypesSupportedKey,
  124. kIDTokenSigningAlgorithmValuesSupportedKey
  125. ];
  126. for (NSString *field in requiredFields) {
  127. if (!dictionary[field]) {
  128. if (error) {
  129. NSString *errorText = [NSString stringWithFormat:kMissingFieldErrorText, field];
  130. *error = [OIDErrorUtilities errorWithCode:OIDErrorCodeInvalidDiscoveryDocument
  131. underlyingError:nil
  132. description:errorText];
  133. }
  134. return NO;
  135. }
  136. }
  137. // Check required URL fields are valid URLs.
  138. NSArray *requiredURLFields = @[
  139. kIssuerKey,
  140. kTokenEndpointKey,
  141. kJWKSURLKey
  142. ];
  143. for (NSString *field in requiredURLFields) {
  144. if (![NSURL URLWithString:dictionary[field]]) {
  145. if (error) {
  146. NSString *errorText = [NSString stringWithFormat:kInvalidURLFieldErrorText, field];
  147. *error = [OIDErrorUtilities errorWithCode:OIDErrorCodeInvalidDiscoveryDocument
  148. underlyingError:nil
  149. description:errorText];
  150. }
  151. return NO;
  152. }
  153. }
  154. return YES;
  155. }
  156. #pragma mark - NSCopying
  157. - (instancetype)copyWithZone:(nullable NSZone *)zone {
  158. // The documentation for NSCopying specifically advises us to return a reference to the original
  159. // instance in the case where instances are immutable (as ours is):
  160. // "Implement NSCopying by retaining the original instead of creating a new copy when the class
  161. // and its contents are immutable."
  162. return self;
  163. }
  164. #pragma mark - NSSecureCoding
  165. + (BOOL)supportsSecureCoding {
  166. return YES;
  167. }
  168. - (nullable instancetype)initWithCoder:(NSCoder *)aDecoder {
  169. NSError *error;
  170. NSDictionary *dictionary = [[NSDictionary alloc] initWithCoder:aDecoder];
  171. self = [self initWithDictionary:dictionary error:&error];
  172. if (error) {
  173. return nil;
  174. }
  175. return self;
  176. }
  177. - (void)encodeWithCoder:(NSCoder *)aCoder {
  178. [_discoveryDictionary encodeWithCoder:aCoder];
  179. }
  180. #pragma mark - Properties
  181. - (NSDictionary<NSString *, NSString *> *)discoveryDictionary {
  182. return _discoveryDictionary;
  183. }
  184. - (NSURL *)issuer {
  185. return [NSURL URLWithString:_discoveryDictionary[kIssuerKey]];
  186. }
  187. - (NSURL *)authorizationEndpoint {
  188. return [NSURL URLWithString:_discoveryDictionary[kAuthorizationEndpointKey]];
  189. }
  190. - (NSURL *)tokenEndpoint {
  191. return [NSURL URLWithString:_discoveryDictionary[kTokenEndpointKey]];
  192. }
  193. - (nullable NSURL *)userinfoEndpoint {
  194. return [NSURL URLWithString:_discoveryDictionary[kUserinfoEndpointKey]];
  195. }
  196. - (NSURL *)jwksURL {
  197. return [NSURL URLWithString:_discoveryDictionary[kJWKSURLKey]];
  198. }
  199. - (nullable NSURL *)registrationEndpoint {
  200. return [NSURL URLWithString:_discoveryDictionary[kRegistrationEndpointKey]];
  201. }
  202. - (nullable NSURL *)endSessionEndpoint {
  203. return [NSURL URLWithString:_discoveryDictionary[kEndSessionEndpointKey]];
  204. }
  205. - (nullable NSArray<NSString *> *)scopesSupported {
  206. return _discoveryDictionary[kScopesSupportedKey];
  207. }
  208. - (NSArray<NSString *> *)responseTypesSupported {
  209. return _discoveryDictionary[kResponseTypesSupportedKey];
  210. }
  211. - (nullable NSArray<NSString *> *)responseModesSupported {
  212. return _discoveryDictionary[kResponseModesSupportedKey];
  213. }
  214. - (nullable NSArray<NSString *> *)grantTypesSupported {
  215. return _discoveryDictionary[kGrantTypesSupportedKey];
  216. }
  217. - (nullable NSArray<NSString *> *)acrValuesSupported {
  218. return _discoveryDictionary[kACRValuesSupportedKey];
  219. }
  220. - (NSArray<NSString *> *)subjectTypesSupported {
  221. return _discoveryDictionary[kSubjectTypesSupportedKey];
  222. }
  223. - (NSArray<NSString *> *) IDTokenSigningAlgorithmValuesSupported {
  224. return _discoveryDictionary[kIDTokenSigningAlgorithmValuesSupportedKey];
  225. }
  226. - (nullable NSArray<NSString *> *)IDTokenEncryptionAlgorithmValuesSupported {
  227. return _discoveryDictionary[kIDTokenEncryptionAlgorithmValuesSupportedKey];
  228. }
  229. - (nullable NSArray<NSString *> *)IDTokenEncryptionEncodingValuesSupported {
  230. return _discoveryDictionary[kIDTokenEncryptionEncodingValuesSupportedKey];
  231. }
  232. - (nullable NSArray<NSString *> *)userinfoSigningAlgorithmValuesSupported {
  233. return _discoveryDictionary[kUserinfoSigningAlgorithmValuesSupportedKey];
  234. }
  235. - (nullable NSArray<NSString *> *)userinfoEncryptionAlgorithmValuesSupported {
  236. return _discoveryDictionary[kUserinfoEncryptionAlgorithmValuesSupportedKey];
  237. }
  238. - (nullable NSArray<NSString *> *)userinfoEncryptionEncodingValuesSupported {
  239. return _discoveryDictionary[kUserinfoEncryptionEncodingValuesSupportedKey];
  240. }
  241. - (nullable NSArray<NSString *> *)requestObjectSigningAlgorithmValuesSupported {
  242. return _discoveryDictionary[kRequestObjectSigningAlgorithmValuesSupportedKey];
  243. }
  244. - (nullable NSArray<NSString *> *) requestObjectEncryptionAlgorithmValuesSupported {
  245. return _discoveryDictionary[kRequestObjectEncryptionAlgorithmValuesSupportedKey];
  246. }
  247. - (nullable NSArray<NSString *> *) requestObjectEncryptionEncodingValuesSupported {
  248. return _discoveryDictionary[kRequestObjectEncryptionEncodingValuesSupported];
  249. }
  250. - (nullable NSArray<NSString *> *)tokenEndpointAuthMethodsSupported {
  251. return _discoveryDictionary[kTokenEndpointAuthMethodsSupportedKey];
  252. }
  253. - (nullable NSArray<NSString *> *)tokenEndpointAuthSigningAlgorithmValuesSupported {
  254. return _discoveryDictionary[kTokenEndpointAuthSigningAlgorithmValuesSupportedKey];
  255. }
  256. - (nullable NSArray<NSString *> *)displayValuesSupported {
  257. return _discoveryDictionary[kDisplayValuesSupportedKey];
  258. }
  259. - (nullable NSArray<NSString *> *)claimTypesSupported {
  260. return _discoveryDictionary[kClaimTypesSupportedKey];
  261. }
  262. - (nullable NSArray<NSString *> *)claimsSupported {
  263. return _discoveryDictionary[kClaimsSupportedKey];
  264. }
  265. - (nullable NSURL *)serviceDocumentation {
  266. return [NSURL URLWithString:_discoveryDictionary[kServiceDocumentationKey]];
  267. }
  268. - (nullable NSArray<NSString *> *)claimsLocalesSupported {
  269. return _discoveryDictionary[kClaimsLocalesSupportedKey];
  270. }
  271. - (nullable NSArray<NSString *> *)UILocalesSupported {
  272. return _discoveryDictionary[kUILocalesSupportedKey];
  273. }
  274. - (BOOL)claimsParameterSupported {
  275. return [_discoveryDictionary[kClaimsParameterSupportedKey] boolValue];
  276. }
  277. - (BOOL)requestParameterSupported {
  278. return [_discoveryDictionary[kRequestParameterSupportedKey] boolValue];
  279. }
  280. - (BOOL)requestURIParameterSupported {
  281. // Default is true/YES.
  282. if (!_discoveryDictionary[kRequestURIParameterSupportedKey]) {
  283. return YES;
  284. }
  285. return [_discoveryDictionary[kRequestURIParameterSupportedKey] boolValue];
  286. }
  287. - (BOOL)requireRequestURIRegistration {
  288. return [_discoveryDictionary[kRequireRequestURIRegistrationKey] boolValue];
  289. }
  290. - (nullable NSURL *)OPPolicyURI {
  291. return [NSURL URLWithString:_discoveryDictionary[kOPPolicyURIKey]];
  292. }
  293. - (nullable NSURL *)OPTosURI {
  294. return [NSURL URLWithString:_discoveryDictionary[kOPTosURIKey]];
  295. }
  296. @end
  297. NS_ASSUME_NONNULL_END