Blazor & WASM in combination to get statistics from Spotify API for performing the song analysis. With separate microservices for auth, Spotify, user data tracking, and application, connected through gRPC with Polly.
Nelze vybrat více než 25 témat Téma musí začínat písmenem nebo číslem, může obsahovat pomlčky („-“) a může být dlouhé až 35 znaků.

AuthService.cs 5.8KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152
  1. //using IdentityProvider.Protos.AuthService;
  2. using Blazored.LocalStorage;
  3. using Grpc.Core;
  4. using Grpc.Net.Client;
  5. using GrpcShared;
  6. using GrpcShared.DTO;
  7. using GrpcShared.DTO.Auth;
  8. using GrpcShared.DTO.Db;
  9. using GrpcShared.DTO.User;
  10. using GrpcShared.Interfaces;
  11. using Microsoft.Extensions.Options;
  12. using Microsoft.Net.Http.Headers;
  13. using Newtonsoft.Json;
  14. using System.Diagnostics;
  15. using System.IO;
  16. using System.Net;
  17. using System.Net.Http.Headers;
  18. using System.Text;
  19. using System.Text.Json;
  20. namespace SpotifyService.Services
  21. {
  22. public class AuthService : IAuthService
  23. {
  24. private readonly ILogger<AuthService> _logger;
  25. private readonly CodeRequest _params;
  26. private readonly IHttpClientFactory _httpClientFactory;
  27. //private ISessionStorageService _sessionStorageService;
  28. public AuthService(ILogger<AuthService> logger, IOptions<CodeRequest> options, IHttpClientFactory httpClientFactory)
  29. {
  30. _logger = logger;
  31. _params = options.Value;
  32. _httpClientFactory = httpClientFactory;
  33. }
  34. public async Task<TokenResponse> GetAccessToken(TokenRequest tokenRequest)
  35. {
  36. var http = _httpClientFactory.CreateClient("HttpClient");
  37. string url = "https://accounts.spotify.com/api/token";
  38. http.BaseAddress = new Uri(url);
  39. //get client id and secret, and redirect uri from appsettings, convert to base64 and set as header
  40. //var secrets = await GetAuthParams();
  41. var secrets = new CodeRequest {
  42. ClientId = GLOBALS.CLIENT_ID ,
  43. ClientSecret = GLOBALS.SECRET,
  44. RedirectURI = GLOBALS.REDIRECT_URI
  45. };
  46. byte[] contentType = Encoding.UTF8.GetBytes($"{secrets.ClientId}:{secrets.ClientSecret}");
  47. tokenRequest.RedirectUri = secrets.RedirectURI;
  48. //AUTHORIZATION HEADER
  49. http.DefaultRequestHeaders.Add(HeaderNames.Authorization, "Basic " + Convert.ToBase64String(contentType));
  50. //BODY PARAMS
  51. var requestBody = new Dictionary<string, string>();
  52. requestBody["grant_type"] = tokenRequest.GrantType;
  53. requestBody["code"] = tokenRequest.Code!;
  54. requestBody["redirect_uri"] = tokenRequest.RedirectUri!;
  55. try
  56. {
  57. //REQUEST
  58. var response = await http.PostAsync(url, new FormUrlEncodedContent(requestBody));
  59. var contents = JsonConvert.DeserializeObject<TokenResponse>(await response.Content.ReadAsStringAsync())!;
  60. return contents;
  61. }
  62. catch (RpcException e)
  63. {
  64. if (e.StatusCode == StatusCode.Cancelled)
  65. {
  66. return new TokenResponse();
  67. }
  68. throw;
  69. }
  70. }
  71. public async Task<CodeRequest> GetAuthParams()
  72. {
  73. var authParams = new CodeRequest
  74. {
  75. ClientId = _params.ClientId,
  76. RedirectURI = _params.RedirectURI,
  77. Scope = _params.Scope,
  78. ClientSecret = _params.ClientSecret
  79. };
  80. return await Task.FromResult(authParams);
  81. }
  82. public async Task<UserInfoResponse> GetUserInfo(UserResponse tokenM)
  83. {
  84. // expired token example "BQBMgFm6jnFNWWeZEMGIRP_f-ENPid7Kw8JubAyuWAe4JK0S1DPFGlaAdZ_Fey6ePkCnz8-cqC0oyRmrciWUy5ISUTQKDe8PTQn4iBRMYCgM0n4GnS1xAErHJcm4Vpu2TAngk-4vQUOfTQRcedNTfCaHKP4uFJgTlTI7JHGrtB-_EZLnFcZ2OQe31oFQIJ1wM3ZtvwnN"
  85. var http = _httpClientFactory.CreateClient("HttpClient");
  86. http.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Bearer", tokenM.Token!);
  87. var response = await http.GetAsync("me");
  88. //make this a method in http utils
  89. if (response.StatusCode == HttpStatusCode.Unauthorized)
  90. {
  91. //refresh the token
  92. var refreshResponse = await RefreshAccessToken(tokenM);
  93. //if response is invalid redirect to login
  94. http.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Bearer", refreshResponse.AccessToken);
  95. response = await http.GetAsync("me");
  96. }
  97. //var headerError = response.Headers.WwwAuthenticate.;
  98. var userInfo = JsonConvert.DeserializeObject<UserInfoResponse>(await response.Content.ReadAsStringAsync())!;
  99. userInfo.ResponseMsg = response.StatusCode;
  100. return userInfo;
  101. }
  102. public async Task<RefreshTokenResponse> RefreshAccessToken(UserResponse tokenM)
  103. {
  104. var client = _httpClientFactory.CreateClient("HttpClient");
  105. client.BaseAddress = new Uri("https://accounts.spotify.com/api/token");
  106. //BODY PARAMS
  107. var requestBody = new Dictionary<string, string>();
  108. requestBody["refresh_token"] = tokenM.RefreshToken!;
  109. requestBody["grant_type"] = "refresh_token";
  110. //var secrets = await GetAuthParams();
  111. var secrets = new CodeRequest
  112. {
  113. ClientId = GLOBALS.CLIENT_ID,
  114. ClientSecret = GLOBALS.SECRET,
  115. RedirectURI = GLOBALS.REDIRECT_URI
  116. };
  117. byte[] contentType = Encoding.UTF8.GetBytes($"{secrets.ClientId}:{secrets.ClientSecret}");
  118. //AUTHORIZATION HEADER
  119. client.DefaultRequestHeaders.Add(HeaderNames.Authorization, "Basic " + Convert.ToBase64String(contentType));
  120. //REQUEST
  121. var response = await client.PostAsync("https://accounts.spotify.com/api/token", new FormUrlEncodedContent(requestBody));
  122. var contents = JsonConvert.DeserializeObject<RefreshTokenResponse>(await response.Content.ReadAsStringAsync())!;
  123. return contents;
  124. }
  125. }
  126. }