2020-05-13 22:49:54 +01:00
|
|
|
using System.Text;
|
|
|
|
using System;
|
|
|
|
using System.Net.Http;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
using SpotifyAPI.Web.Http;
|
|
|
|
|
|
|
|
namespace SpotifyAPI.Web
|
|
|
|
{
|
|
|
|
public class OAuthClient : APIClient, IOAuthClient
|
|
|
|
{
|
|
|
|
public OAuthClient() : this(SpotifyClientConfig.CreateDefault()) { }
|
2020-06-02 15:34:24 +01:00
|
|
|
public OAuthClient(IAPIConnector apiConnector) : base(apiConnector) { }
|
2020-05-13 22:49:54 +01:00
|
|
|
|
|
|
|
[System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Design", "CA1062")]
|
|
|
|
public OAuthClient(SpotifyClientConfig config) : base(ValidateConfig(config)) { }
|
|
|
|
|
2020-08-06 19:48:49 +01:00
|
|
|
/// <summary>
|
|
|
|
/// Requests a new token using pkce flow
|
|
|
|
/// </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-with-proof-key-for-code-exchange-pkce
|
|
|
|
/// </remarks>
|
|
|
|
/// <returns></returns>1
|
|
|
|
public Task<PKCETokenResponse> RequestToken(PKCETokenRequest request)
|
|
|
|
{
|
|
|
|
return RequestToken(request, API);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// Refreshes a token using pkce flow
|
|
|
|
/// </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-with-proof-key-for-code-exchange-pkce
|
|
|
|
/// </remarks>
|
|
|
|
/// <returns></returns>1
|
|
|
|
public Task<PKCETokenResponse> RequestToken(PKCETokenRefreshRequest request)
|
|
|
|
{
|
|
|
|
return RequestToken(request, API);
|
|
|
|
}
|
|
|
|
|
2020-06-02 22:55:50 +01:00
|
|
|
/// <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
|
2020-06-07 12:28:49 +01:00
|
|
|
public Task<ClientCredentialsTokenResponse> RequestToken(ClientCredentialsRequest request)
|
2020-05-14 22:26:40 +01:00
|
|
|
{
|
|
|
|
return RequestToken(request, API);
|
|
|
|
}
|
|
|
|
|
2020-06-02 22:55:50 +01:00
|
|
|
/// <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>
|
2020-05-15 19:06:24 +01:00
|
|
|
public Task<AuthorizationCodeRefreshResponse> RequestToken(AuthorizationCodeRefreshRequest request)
|
|
|
|
{
|
|
|
|
return RequestToken(request, API);
|
|
|
|
}
|
|
|
|
|
2020-06-02 22:55:50 +01:00
|
|
|
/// <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>
|
2020-05-15 19:06:24 +01:00
|
|
|
public Task<AuthorizationCodeTokenResponse> RequestToken(AuthorizationCodeTokenRequest request)
|
|
|
|
{
|
|
|
|
return RequestToken(request, API);
|
|
|
|
}
|
|
|
|
|
2020-06-02 22:55:50 +01:00
|
|
|
/// <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>
|
2020-06-02 15:34:24 +01:00
|
|
|
public Task<AuthorizationCodeTokenResponse> RequestToken(TokenSwapTokenRequest request)
|
|
|
|
{
|
|
|
|
return RequestToken(request, API);
|
|
|
|
}
|
|
|
|
|
2020-06-02 22:55:50 +01:00
|
|
|
/// <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>
|
2020-06-02 15:34:24 +01:00
|
|
|
public Task<AuthorizationCodeRefreshResponse> RequestToken(TokenSwapRefreshRequest request)
|
|
|
|
{
|
|
|
|
return RequestToken(request, API);
|
|
|
|
}
|
|
|
|
|
2020-08-06 19:48:49 +01:00
|
|
|
public static Task<PKCETokenResponse> RequestToken(PKCETokenRequest request, IAPIConnector apiConnector)
|
|
|
|
{
|
|
|
|
Ensure.ArgumentNotNull(request, nameof(request));
|
|
|
|
Ensure.ArgumentNotNull(apiConnector, nameof(apiConnector));
|
|
|
|
|
2020-11-13 13:43:00 +00:00
|
|
|
var form = new List<KeyValuePair<string?, string?>>
|
2020-08-06 19:48:49 +01:00
|
|
|
{
|
2020-11-13 13:43:00 +00:00
|
|
|
new KeyValuePair<string?, string?>("client_id", request.ClientId),
|
|
|
|
new KeyValuePair<string?, string?>("grant_type", "authorization_code"),
|
|
|
|
new KeyValuePair<string?, string?>("code", request.Code),
|
|
|
|
new KeyValuePair<string?, string?>("redirect_uri", request.RedirectUri.ToString()),
|
|
|
|
new KeyValuePair<string?, string?>("code_verifier", request.CodeVerifier),
|
2020-08-06 19:48:49 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
return SendOAuthRequest<PKCETokenResponse>(apiConnector, form, null, null);
|
|
|
|
}
|
|
|
|
|
|
|
|
public static Task<PKCETokenResponse> RequestToken(PKCETokenRefreshRequest request, IAPIConnector apiConnector)
|
|
|
|
{
|
|
|
|
Ensure.ArgumentNotNull(request, nameof(request));
|
|
|
|
Ensure.ArgumentNotNull(apiConnector, nameof(apiConnector));
|
|
|
|
|
2020-11-13 13:43:00 +00:00
|
|
|
var form = new List<KeyValuePair<string?, string?>>
|
2020-08-06 19:48:49 +01:00
|
|
|
{
|
2020-11-13 13:43:00 +00:00
|
|
|
new KeyValuePair<string?, string?>("client_id", request.ClientId),
|
|
|
|
new KeyValuePair<string?, string?>("grant_type", "refresh_token"),
|
|
|
|
new KeyValuePair<string?, string?>("refresh_token", request.RefreshToken),
|
2020-08-06 19:48:49 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
return SendOAuthRequest<PKCETokenResponse>(apiConnector, form, null, null);
|
|
|
|
}
|
|
|
|
|
2020-06-02 15:34:24 +01:00
|
|
|
public static Task<AuthorizationCodeRefreshResponse> RequestToken(
|
|
|
|
TokenSwapRefreshRequest request, IAPIConnector apiConnector
|
|
|
|
)
|
|
|
|
{
|
|
|
|
Ensure.ArgumentNotNull(request, nameof(request));
|
|
|
|
Ensure.ArgumentNotNull(apiConnector, nameof(apiConnector));
|
|
|
|
|
2020-11-13 13:43:00 +00:00
|
|
|
var form = new List<KeyValuePair<string?, string?>>
|
2020-06-02 15:34:24 +01:00
|
|
|
{
|
2020-11-13 13:43:00 +00:00
|
|
|
new KeyValuePair<string?, string?>("refresh_token", request.RefreshToken)
|
2020-06-02 15:34:24 +01:00
|
|
|
};
|
2020-06-04 19:56:47 +01:00
|
|
|
#pragma warning disable CA2000
|
2020-06-02 15:34:24 +01:00
|
|
|
return apiConnector.Post<AuthorizationCodeRefreshResponse>(
|
|
|
|
request.RefreshUri, null, new FormUrlEncodedContent(form)
|
|
|
|
);
|
2020-06-04 19:56:47 +01:00
|
|
|
#pragma warning restore CA2000
|
2020-06-02 15:34:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public static Task<AuthorizationCodeTokenResponse> RequestToken(
|
|
|
|
TokenSwapTokenRequest request, IAPIConnector apiConnector
|
|
|
|
)
|
|
|
|
{
|
|
|
|
Ensure.ArgumentNotNull(request, nameof(request));
|
|
|
|
Ensure.ArgumentNotNull(apiConnector, nameof(apiConnector));
|
|
|
|
|
2020-11-13 13:43:00 +00:00
|
|
|
var form = new List<KeyValuePair<string?, string?>>
|
2020-06-02 15:34:24 +01:00
|
|
|
{
|
2020-11-13 13:43:00 +00:00
|
|
|
new KeyValuePair<string?, string?>("code", request.Code)
|
2020-06-02 15:34:24 +01:00
|
|
|
};
|
|
|
|
|
2020-06-04 19:56:47 +01:00
|
|
|
#pragma warning disable CA2000
|
2020-06-02 15:34:24 +01:00
|
|
|
return apiConnector.Post<AuthorizationCodeTokenResponse>(
|
|
|
|
request.TokenUri, null, new FormUrlEncodedContent(form)
|
|
|
|
);
|
2020-06-04 19:56:47 +01:00
|
|
|
#pragma warning restore CA2000
|
2020-06-02 15:34:24 +01:00
|
|
|
}
|
2020-05-15 19:06:24 +01:00
|
|
|
|
2020-06-07 12:28:49 +01:00
|
|
|
public static Task<ClientCredentialsTokenResponse> RequestToken(
|
2020-05-14 22:26:40 +01:00
|
|
|
ClientCredentialsRequest request, IAPIConnector apiConnector
|
|
|
|
)
|
2020-05-13 22:49:54 +01:00
|
|
|
{
|
|
|
|
Ensure.ArgumentNotNull(request, nameof(request));
|
2020-05-14 22:26:40 +01:00
|
|
|
Ensure.ArgumentNotNull(apiConnector, nameof(apiConnector));
|
2020-05-13 22:49:54 +01:00
|
|
|
|
2020-11-13 13:43:00 +00:00
|
|
|
var form = new List<KeyValuePair<string?, string?>>
|
2020-05-13 22:49:54 +01:00
|
|
|
{
|
2020-11-13 13:43:00 +00:00
|
|
|
new KeyValuePair<string?, string?>("grant_type", "client_credentials")
|
2020-05-13 22:49:54 +01:00
|
|
|
};
|
|
|
|
|
2020-06-07 12:28:49 +01:00
|
|
|
return SendOAuthRequest<ClientCredentialsTokenResponse>(apiConnector, form, request.ClientId, request.ClientSecret);
|
2020-05-15 19:06:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public static Task<AuthorizationCodeRefreshResponse> RequestToken(
|
|
|
|
AuthorizationCodeRefreshRequest request, IAPIConnector apiConnector
|
|
|
|
)
|
|
|
|
{
|
|
|
|
Ensure.ArgumentNotNull(request, nameof(request));
|
|
|
|
Ensure.ArgumentNotNull(apiConnector, nameof(apiConnector));
|
|
|
|
|
2020-11-13 13:43:00 +00:00
|
|
|
var form = new List<KeyValuePair<string?, string?>>
|
2020-05-13 22:49:54 +01:00
|
|
|
{
|
2020-11-13 13:43:00 +00:00
|
|
|
new KeyValuePair<string?, string?>("grant_type", "refresh_token"),
|
|
|
|
new KeyValuePair<string?, string?>("refresh_token", request.RefreshToken)
|
2020-05-15 19:06:24 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
return SendOAuthRequest<AuthorizationCodeRefreshResponse>(apiConnector, form, request.ClientId, request.ClientSecret);
|
|
|
|
}
|
|
|
|
|
|
|
|
public static Task<AuthorizationCodeTokenResponse> RequestToken(
|
|
|
|
AuthorizationCodeTokenRequest request, IAPIConnector apiConnector
|
|
|
|
)
|
|
|
|
{
|
|
|
|
Ensure.ArgumentNotNull(request, nameof(request));
|
|
|
|
Ensure.ArgumentNotNull(apiConnector, nameof(apiConnector));
|
|
|
|
|
2020-11-13 13:43:00 +00:00
|
|
|
var form = new List<KeyValuePair<string?, string?>>
|
2020-05-15 19:06:24 +01:00
|
|
|
{
|
2020-11-13 13:43:00 +00:00
|
|
|
new KeyValuePair<string?, string?>("grant_type", "authorization_code"),
|
|
|
|
new KeyValuePair<string?, string?>("code", request.Code),
|
|
|
|
new KeyValuePair<string?, string?>("redirect_uri", request.RedirectUri.ToString())
|
2020-05-13 22:49:54 +01:00
|
|
|
};
|
|
|
|
|
2020-05-15 19:06:24 +01:00
|
|
|
return SendOAuthRequest<AuthorizationCodeTokenResponse>(apiConnector, form, request.ClientId, request.ClientSecret);
|
|
|
|
}
|
|
|
|
|
|
|
|
private static Task<T> SendOAuthRequest<T>(
|
|
|
|
IAPIConnector apiConnector,
|
2020-11-13 13:43:00 +00:00
|
|
|
List<KeyValuePair<string?, string?>> form,
|
2020-08-06 19:48:49 +01:00
|
|
|
string? clientId,
|
|
|
|
string? clientSecret)
|
2020-05-15 19:06:24 +01:00
|
|
|
{
|
|
|
|
var headers = BuildAuthHeader(clientId, clientSecret);
|
2020-06-04 19:56:47 +01:00
|
|
|
#pragma warning disable CA2000
|
2020-05-15 19:06:24 +01:00
|
|
|
return apiConnector.Post<T>(SpotifyUrls.OAuthToken, null, new FormUrlEncodedContent(form), headers);
|
2020-06-04 19:56:47 +01:00
|
|
|
#pragma warning restore CA2000
|
2020-05-15 19:06:24 +01:00
|
|
|
}
|
|
|
|
|
2020-08-06 19:48:49 +01:00
|
|
|
private static Dictionary<string, string> BuildAuthHeader(string? clientId, string? clientSecret)
|
2020-05-15 19:06:24 +01:00
|
|
|
{
|
2020-08-06 19:48:49 +01:00
|
|
|
if (clientId == null || clientSecret == null)
|
|
|
|
{
|
|
|
|
return new Dictionary<string, string>();
|
|
|
|
}
|
|
|
|
|
2020-05-15 19:06:24 +01:00
|
|
|
var base64 = Convert.ToBase64String(Encoding.UTF8.GetBytes($"{clientId}:{clientSecret}"));
|
|
|
|
return new Dictionary<string, string>
|
|
|
|
{
|
|
|
|
{ "Authorization", $"Basic {base64}"}
|
|
|
|
};
|
2020-05-13 22:49:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
private static APIConnector ValidateConfig(SpotifyClientConfig config)
|
|
|
|
{
|
|
|
|
Ensure.ArgumentNotNull(config, nameof(config));
|
2020-05-25 17:00:38 +01:00
|
|
|
|
2020-05-13 22:49:54 +01:00
|
|
|
return new APIConnector(
|
|
|
|
config.BaseAddress,
|
|
|
|
config.Authenticator,
|
|
|
|
config.JSONSerializer,
|
|
|
|
config.HTTPClient,
|
|
|
|
config.RetryHandler,
|
|
|
|
config.HTTPLogger
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|