OIDAuthState.h 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  1. /*! @file OIDAuthState.h
  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 <Foundation/Foundation.h>
  17. @class OIDAuthorizationRequest;
  18. @class OIDAuthorizationResponse;
  19. @class OIDAuthState;
  20. @class OIDRegistrationResponse;
  21. @class OIDTokenResponse;
  22. @class OIDTokenRequest;
  23. @protocol OIDAuthStateChangeDelegate;
  24. @protocol OIDAuthStateErrorDelegate;
  25. @protocol OIDExternalUserAgent;
  26. @protocol OIDExternalUserAgentSession;
  27. NS_ASSUME_NONNULL_BEGIN
  28. /*! @brief Represents a block used to call an action with a fresh access token.
  29. @param accessToken A valid access token if available.
  30. @param idToken A valid ID token if available.
  31. @param error The error if an error occurred.
  32. */
  33. typedef void (^OIDAuthStateAction)(NSString *_Nullable accessToken,
  34. NSString *_Nullable idToken,
  35. NSError *_Nullable error);
  36. /*! @brief The method called when the @c
  37. OIDAuthState.authStateByPresentingAuthorizationRequest:presentingViewController:callback:
  38. method has completed or failed.
  39. @param authState The auth state, if the authorization request succeeded.
  40. @param error The error if an error occurred.
  41. */
  42. typedef void (^OIDAuthStateAuthorizationCallback)(OIDAuthState *_Nullable authState,
  43. NSError *_Nullable error);
  44. /*! @brief A convenience class that retains the auth state between @c OIDAuthorizationResponse%s
  45. and @c OIDTokenResponse%s.
  46. */
  47. @interface OIDAuthState : NSObject <NSSecureCoding>
  48. /*! @brief The most recent refresh token received from the server.
  49. @discussion Rather than using this property directly, you should call
  50. @c OIDAuthState.performActionWithFreshTokens:.
  51. @remarks refresh_token
  52. @see https://tools.ietf.org/html/rfc6749#section-5.1
  53. */
  54. @property(nonatomic, readonly, nullable) NSString *refreshToken;
  55. /*! @brief The scope of the current authorization grant.
  56. @discussion This represents the latest scope returned by the server and may be a subset of the
  57. scope that was initially granted.
  58. @remarks scope
  59. */
  60. @property(nonatomic, readonly, nullable) NSString *scope;
  61. /*! @brief The most recent authorization response used to update the authorization state. For the
  62. implicit flow, this will contain the latest access token.
  63. */
  64. @property(nonatomic, readonly) OIDAuthorizationResponse *lastAuthorizationResponse;
  65. /*! @brief The most recent token response used to update this authorization state. This will
  66. contain the latest access token.
  67. */
  68. @property(nonatomic, readonly, nullable) OIDTokenResponse *lastTokenResponse;
  69. /*! @brief The most recent registration response used to update this authorization state. This will
  70. contain the latest client credentials.
  71. */
  72. @property(nonatomic, readonly, nullable) OIDRegistrationResponse *lastRegistrationResponse;
  73. /*! @brief The authorization error that invalidated this @c OIDAuthState.
  74. @discussion The authorization error encountered by @c OIDAuthState or set by the user via
  75. @c OIDAuthState.updateWithAuthorizationError: that invalidated this @c OIDAuthState.
  76. Authorization errors from @c OIDAuthState will always have a domain of
  77. @c ::OIDOAuthAuthorizationErrorDomain or @c ::OIDOAuthTokenErrorDomain. Note: that after
  78. unarchiving the @c OIDAuthState object, the \NSError_userInfo property of this error will
  79. be nil.
  80. */
  81. @property(nonatomic, readonly, nullable) NSError *authorizationError;
  82. /*! @brief Returns YES if the authorization state is not known to be invalid.
  83. @discussion Returns YES if no OAuth errors have been received, and the last call resulted in a
  84. successful access token or id token. This does not mean that the access is fresh - just
  85. that it was valid the last time it was used. Note that network and other transient errors
  86. do not invalidate the authorized state. If NO, you should authenticate the user again,
  87. using a fresh authorization request. Invalid @c OIDAuthState objects may still be useful in
  88. that case, to hint at the previously authorized user and streamline the re-authentication
  89. experience.
  90. */
  91. @property(nonatomic, readonly) BOOL isAuthorized;
  92. /*! @brief The @c OIDAuthStateChangeDelegate delegate.
  93. @discussion Use the delegate to observe state changes (and update storage) as well as error
  94. states.
  95. */
  96. @property(nonatomic, weak, nullable) id<OIDAuthStateChangeDelegate> stateChangeDelegate;
  97. /*! @brief The @c OIDAuthStateErrorDelegate delegate.
  98. @discussion Use the delegate to observe state changes (and update storage) as well as error
  99. states.
  100. */
  101. @property(nonatomic, weak, nullable) id<OIDAuthStateErrorDelegate> errorDelegate;
  102. /*! @brief Convenience method to create a @c OIDAuthState by presenting an authorization request
  103. and performing the authorization code exchange in the case of code flow requests. For
  104. the hybrid flow, the caller should validate the id_token and c_hash, then perform the token
  105. request (@c OIDAuthorizationService.performTokenRequest:callback:)
  106. and update the OIDAuthState with the results (@c
  107. OIDAuthState.updateWithTokenResponse:error:).
  108. @param authorizationRequest The authorization request to present.
  109. @param externalUserAgent A external user agent that can present an external user-agent request.
  110. @param callback The method called when the request has completed or failed.
  111. @return A @c OIDExternalUserAgentSession instance which will terminate when it
  112. receives a @c OIDExternalUserAgentSession.cancel message, or after processing a
  113. @c OIDExternalUserAgentSession.resumeExternalUserAgentFlowWithURL: message.
  114. */
  115. + (id<OIDExternalUserAgentSession>)
  116. authStateByPresentingAuthorizationRequest:(OIDAuthorizationRequest *)authorizationRequest
  117. externalUserAgent:(id<OIDExternalUserAgent>)externalUserAgent
  118. callback:(OIDAuthStateAuthorizationCallback)callback;
  119. /*! @internal
  120. @brief Unavailable. Please use @c initWithAuthorizationResponse:.
  121. */
  122. - (instancetype)init NS_UNAVAILABLE;
  123. /*! @brief Creates an auth state from an authorization response.
  124. @param authorizationResponse The authorization response.
  125. */
  126. - (instancetype)initWithAuthorizationResponse:(OIDAuthorizationResponse *)authorizationResponse;
  127. /*! @brief Creates an auth state from an authorization and token response.
  128. @param authorizationResponse The authorization response.
  129. @param tokenResponse The token response.
  130. */
  131. - (instancetype)initWithAuthorizationResponse:(OIDAuthorizationResponse *)authorizationResponse
  132. tokenResponse:(nullable OIDTokenResponse *)tokenResponse;
  133. /*! @brief Creates an auth state from an registration response.
  134. @param registrationResponse The registration response.
  135. */
  136. - (instancetype)initWithRegistrationResponse:(OIDRegistrationResponse *)registrationResponse;
  137. /*! @brief Creates an auth state from an authorization, token and registration response.
  138. @param authorizationResponse The authorization response.
  139. @param tokenResponse The token response.
  140. @param registrationResponse The registration response.
  141. */
  142. - (instancetype)initWithAuthorizationResponse:
  143. (nullable OIDAuthorizationResponse *)authorizationResponse
  144. tokenResponse:(nullable OIDTokenResponse *)tokenResponse
  145. registrationResponse:(nullable OIDRegistrationResponse *)registrationResponse
  146. NS_DESIGNATED_INITIALIZER;
  147. /*! @brief Updates the authorization state based on a new authorization response.
  148. @param authorizationResponse The new authorization response to update the state with.
  149. @param error Any error encountered when performing the authorization request. Errors in the
  150. domain @c ::OIDOAuthAuthorizationErrorDomain are reflected in the auth state, other errors
  151. are assumed to be transient, and ignored.
  152. @discussion Typically called with the response from an incremental authorization request,
  153. or if using the implicit flow. Will clear the @c #lastTokenResponse property.
  154. */
  155. - (void)updateWithAuthorizationResponse:(nullable OIDAuthorizationResponse *)authorizationResponse
  156. error:(nullable NSError *)error;
  157. /*! @brief Updates the authorization state based on a new token response.
  158. @param tokenResponse The new token response to update the state from.
  159. @param error Any error encountered when performing the authorization request. Errors in the
  160. domain @c ::OIDOAuthTokenErrorDomain are reflected in the auth state, other errors
  161. are assumed to be transient, and ignored.
  162. @discussion Typically called with the response from an authorization code exchange, or a token
  163. refresh.
  164. */
  165. - (void)updateWithTokenResponse:(nullable OIDTokenResponse *)tokenResponse
  166. error:(nullable NSError *)error;
  167. /*! @brief Updates the authorization state based on a new registration response.
  168. @param registrationResponse The new registration response to update the state with.
  169. @discussion Typically called with the response from a successful client registration
  170. request. Will reset the auth state.
  171. */
  172. - (void)updateWithRegistrationResponse:(nullable OIDRegistrationResponse *)registrationResponse;
  173. /*! @brief Updates the authorization state based on an authorization error.
  174. @param authorizationError The authorization error.
  175. @discussion Call this method if you receive an authorization error during an API call to
  176. invalidate the authentication state of this @c OIDAuthState. Don't call with errors
  177. unrelated to authorization, such as transient network errors.
  178. The OIDAuthStateErrorDelegate.authState:didEncounterAuthorizationError: method of
  179. @c #errorDelegate will be called with the error.
  180. You may optionally use the convenience method
  181. OIDErrorUtilities.resourceServerAuthorizationErrorWithCode:errorResponse:underlyingError:
  182. to create \NSError objects for use here.
  183. The latest error received is stored in @c #authorizationError. Note: that after unarchiving
  184. this object, the \NSError_userInfo property of this error will be nil.
  185. */
  186. - (void)updateWithAuthorizationError:(NSError *)authorizationError;
  187. /*! @brief Calls the block with a valid access token (refreshing it first, if needed), or if a
  188. refresh was needed and failed, with the error that caused it to fail.
  189. @param action The block to execute with a fresh token. This block will be executed on the main
  190. thread.
  191. */
  192. - (void)performActionWithFreshTokens:(OIDAuthStateAction)action;
  193. /*! @brief Calls the block with a valid access token (refreshing it first, if needed), or if a
  194. refresh was needed and failed, with the error that caused it to fail.
  195. @param action The block to execute with a fresh token. This block will be executed on the main
  196. thread.
  197. @param additionalParameters Additional parameters for the token request if token is
  198. refreshed.
  199. */
  200. - (void)performActionWithFreshTokens:(OIDAuthStateAction)action
  201. additionalRefreshParameters:
  202. (nullable NSDictionary<NSString *, NSString *> *)additionalParameters;
  203. /*! @brief Calls the block with a valid access token (refreshing it first, if needed), or if a
  204. refresh was needed and failed, with the error that caused it to fail.
  205. @param action The block to execute with a fresh token. This block will be executed on the main
  206. thread.
  207. @param additionalParameters Additional parameters for the token request if token is
  208. refreshed.
  209. @param dispatchQueue The dispatchQueue on which to dispatch the action block.
  210. */
  211. - (void)performActionWithFreshTokens:(OIDAuthStateAction)action
  212. additionalRefreshParameters:
  213. (nullable NSDictionary<NSString *, NSString *> *)additionalParameters
  214. dispatchQueue:(dispatch_queue_t)dispatchQueue;
  215. /*! @brief Forces a token refresh the next time @c OIDAuthState.performActionWithFreshTokens: is
  216. called, even if the current tokens are considered valid.
  217. */
  218. - (void)setNeedsTokenRefresh;
  219. /*! @brief Creates a token request suitable for refreshing an access token.
  220. @return A @c OIDTokenRequest suitable for using a refresh token to obtain a new access token.
  221. @discussion After performing the refresh, call @c OIDAuthState.updateWithTokenResponse:error:
  222. to update the authorization state based on the response. Rather than doing the token refresh
  223. yourself, you should use @c OIDAuthState.performActionWithFreshTokens:.
  224. @see https://tools.ietf.org/html/rfc6749#section-1.5
  225. */
  226. - (nullable OIDTokenRequest *)tokenRefreshRequest;
  227. /*! @brief Creates a token request suitable for refreshing an access token.
  228. @param additionalParameters Additional parameters for the token request.
  229. @return A @c OIDTokenRequest suitable for using a refresh token to obtain a new access token.
  230. @discussion After performing the refresh, call @c OIDAuthState.updateWithTokenResponse:error:
  231. to update the authorization state based on the response. Rather than doing the token refresh
  232. yourself, you should use @c OIDAuthState.performActionWithFreshTokens:.
  233. @see https://tools.ietf.org/html/rfc6749#section-1.5
  234. */
  235. - (nullable OIDTokenRequest *)tokenRefreshRequestWithAdditionalParameters:
  236. (nullable NSDictionary<NSString *, NSString *> *)additionalParameters;
  237. @end
  238. NS_ASSUME_NONNULL_END