Moar docs and TooManyRequests exception

This commit is contained in:
Jonas Dellinger 2020-06-02 23:55:50 +02:00
parent ef60a9f1f2
commit 2a9b0338f5
6 changed files with 104 additions and 6 deletions

View File

@ -11,7 +11,7 @@ namespace Client
private static readonly string clientId = Environment.GetEnvironmentVariable("SPOTIFY_CLIENT_ID"); private static readonly string clientId = Environment.GetEnvironmentVariable("SPOTIFY_CLIENT_ID");
private static EmbedIOAuthServer _server; private static EmbedIOAuthServer _server;
public static async Task Main(string[] args) public static async Task Main()
{ {
_server = new EmbedIOAuthServer(new Uri("http://localhost:5000/callback"), 5000); _server = new EmbedIOAuthServer(new Uri("http://localhost:5000/callback"), 5000);
await _server.Start(); await _server.Start();

View File

@ -11,7 +11,7 @@ namespace SpotifyAPI.Web
/// Requests a new token using client_ids and client_secrets. /// Requests a new token using client_ids and client_secrets.
/// If the token is expired, simply call the funtion again to get a new token /// If the token is expired, simply call the funtion again to get a new token
/// </summary> /// </summary>
/// <param name="request"></param> /// <param name="request">The request-model which contains required and optional parameters.</param>
/// <remarks> /// <remarks>
/// https://developer.spotify.com/documentation/general/guides/authorization-guide/#client-credentials-flow /// https://developer.spotify.com/documentation/general/guides/authorization-guide/#client-credentials-flow
/// </remarks> /// </remarks>
@ -21,7 +21,7 @@ namespace SpotifyAPI.Web
/// <summary> /// <summary>
/// Refresh an already received token via Authorization Code Auth /// Refresh an already received token via Authorization Code Auth
/// </summary> /// </summary>
/// <param name="request"></param> /// <param name="request">The request-model which contains required and optional parameters.</param>
/// <remarks> /// <remarks>
/// https://developer.spotify.com/documentation/general/guides/authorization-guide/#authorization-code-flow /// https://developer.spotify.com/documentation/general/guides/authorization-guide/#authorization-code-flow
/// </remarks> /// </remarks>
@ -31,11 +31,31 @@ namespace SpotifyAPI.Web
/// <summary> /// <summary>
/// Reequest an initial token via Authorization Code Auth /// Reequest an initial token via Authorization Code Auth
/// </summary> /// </summary>
/// <param name="request"></param> /// <param name="request">The request-model which contains required and optional parameters.</param>
/// <remarks> /// <remarks>
/// https://developer.spotify.com/documentation/general/guides/authorization-guide/#authorization-code-flow /// https://developer.spotify.com/documentation/general/guides/authorization-guide/#authorization-code-flow
/// </remarks> /// </remarks>
/// <returns></returns> /// <returns></returns>
Task<AuthorizationCodeTokenResponse> RequestToken(AuthorizationCodeTokenRequest request); Task<AuthorizationCodeTokenResponse> RequestToken(AuthorizationCodeTokenRequest request);
/// <summary>
/// Swaps out a received code with a access token using a remote server
/// </summary>
/// <param name="request">The request-model which contains required and optional parameters.</param>
/// <remarks>
/// https://developer.spotify.com/documentation/ios/guides/token-swap-and-refresh/
/// </remarks>
/// <returns></returns>
Task<AuthorizationCodeTokenResponse> RequestToken(TokenSwapTokenRequest request);
/// <summary>
/// Gets a refreshed access token using an already received refresh token using a remote server
/// </summary>
/// <param name="request"></param>
/// <remarks>
/// https://developer.spotify.com/documentation/ios/guides/token-swap-and-refresh/
/// </remarks>
/// <returns></returns>
Task<AuthorizationCodeRefreshResponse> RequestToken(TokenSwapRefreshRequest request);
} }
} }

View File

@ -15,26 +15,67 @@ namespace SpotifyAPI.Web
[System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Design", "CA1062")] [System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Design", "CA1062")]
public OAuthClient(SpotifyClientConfig config) : base(ValidateConfig(config)) { } public OAuthClient(SpotifyClientConfig config) : base(ValidateConfig(config)) { }
/// <summary>
/// Requests a new token using client_ids and client_secrets.
/// If the token is expired, simply call the funtion again to get a new token
/// </summary>
/// <param name="request">The request-model which contains required and optional parameters.</param>
/// <remarks>
/// https://developer.spotify.com/documentation/general/guides/authorization-guide/#client-credentials-flow
/// </remarks>
/// <returns></returns>1
public Task<CredentialsTokenResponse> RequestToken(ClientCredentialsRequest request) public Task<CredentialsTokenResponse> RequestToken(ClientCredentialsRequest request)
{ {
return RequestToken(request, API); return RequestToken(request, API);
} }
/// <summary>
/// Refresh an already received token via Authorization Code Auth
/// </summary>
/// <param name="request">The request-model which contains required and optional parameters.</param>
/// <remarks>
/// https://developer.spotify.com/documentation/general/guides/authorization-guide/#authorization-code-flow
/// </remarks>
/// <returns></returns>
public Task<AuthorizationCodeRefreshResponse> RequestToken(AuthorizationCodeRefreshRequest request) public Task<AuthorizationCodeRefreshResponse> RequestToken(AuthorizationCodeRefreshRequest request)
{ {
return RequestToken(request, API); return RequestToken(request, API);
} }
/// <summary>
/// Reequest an initial token via Authorization Code Auth
/// </summary>
/// <param name="request">The request-model which contains required and optional parameters.</param>
/// <remarks>
/// https://developer.spotify.com/documentation/general/guides/authorization-guide/#authorization-code-flow
/// </remarks>
/// <returns></returns>
public Task<AuthorizationCodeTokenResponse> RequestToken(AuthorizationCodeTokenRequest request) public Task<AuthorizationCodeTokenResponse> RequestToken(AuthorizationCodeTokenRequest request)
{ {
return RequestToken(request, API); return RequestToken(request, API);
} }
/// <summary>
/// Swaps out a received code with a access token using a remote server
/// </summary>
/// <param name="request">The request-model which contains required and optional parameters.</param>
/// <remarks>
/// https://developer.spotify.com/documentation/ios/guides/token-swap-and-refresh/
/// </remarks>
/// <returns></returns>
public Task<AuthorizationCodeTokenResponse> RequestToken(TokenSwapTokenRequest request) public Task<AuthorizationCodeTokenResponse> RequestToken(TokenSwapTokenRequest request)
{ {
return RequestToken(request, API); return RequestToken(request, API);
} }
/// <summary>
/// Gets a refreshed access token using an already received refresh token using a remote server
/// </summary>
/// <param name="request"></param>
/// <remarks>
/// https://developer.spotify.com/documentation/ios/guides/token-swap-and-refresh/
/// </remarks>
/// <returns></returns>
public Task<AuthorizationCodeRefreshResponse> RequestToken(TokenSwapRefreshRequest request) public Task<AuthorizationCodeRefreshResponse> RequestToken(TokenSwapRefreshRequest request)
{ {
return RequestToken(request, API); return RequestToken(request, API);

View File

@ -0,0 +1,31 @@
using System.Globalization;
using System.Runtime.Serialization;
using System;
using SpotifyAPI.Web.Http;
namespace SpotifyAPI.Web
{
[Serializable]
public class APITooManyRequestsException : APIException
{
public TimeSpan RetryAfter { get; }
public APITooManyRequestsException(IResponse response) : base(response)
{
Ensure.ArgumentNotNull(response, nameof(response));
if (response.Headers.TryGetValue("Retry-After", out string retryAfter))
{
RetryAfter = TimeSpan.FromSeconds(int.Parse(retryAfter, CultureInfo.InvariantCulture));
}
}
public APITooManyRequestsException() { }
public APITooManyRequestsException(string message) : base(message) { }
public APITooManyRequestsException(string message, Exception innerException) : base(message, innerException) { }
protected APITooManyRequestsException(SerializationInfo info, StreamingContext context) : base(info, context) { }
}
}

View File

@ -283,6 +283,8 @@ namespace SpotifyAPI.Web.Http
{ {
case HttpStatusCode.Unauthorized: case HttpStatusCode.Unauthorized:
throw new APIUnauthorizedException(response); throw new APIUnauthorizedException(response);
case HttpStatusCode.TooManyRequests:
throw new APITooManyRequestsException(response);
default: default:
throw new APIException(response); throw new APIException(response);
} }

View File

@ -9,8 +9,6 @@ namespace SpotifyAPI.Web.Http
/// </summary> /// </summary>
public class AuthorizationCodeAuthenticator : IAuthenticator public class AuthorizationCodeAuthenticator : IAuthenticator
{ {
public event EventHandler<AuthorizationCodeTokenResponse>? TokenRefreshed;
/// <summary> /// <summary>
/// Initiate a new instance. The token will be refreshed once it expires. /// Initiate a new instance. The token will be refreshed once it expires.
/// The initialToken will be updated with the new values on refresh! /// The initialToken will be updated with the new values on refresh!
@ -26,6 +24,12 @@ namespace SpotifyAPI.Web.Http
ClientSecret = clientSecret; ClientSecret = clientSecret;
} }
/// <summary>
/// This event is called once a new refreshed token was aquired
/// </summary>
public event EventHandler<AuthorizationCodeTokenResponse>? TokenRefreshed;
/// <summary> /// <summary>
/// The ClientID, defined in a spotify application in your Spotify Developer Dashboard /// The ClientID, defined in a spotify application in your Spotify Developer Dashboard
/// </summary> /// </summary>