CoupangModel.cs 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257
  1. using System.Security.Cryptography;
  2. using System.Text;
  3. using System.Text.Json;
  4. namespace goods.Models.Coupang
  5. {
  6. public class CoupangModel
  7. {
  8. private readonly CoupangPartners _coupangPartners;
  9. private string _ChannelID;
  10. public CoupangModel(CoupangPartners coupangPartners)
  11. {
  12. _coupangPartners = coupangPartners;
  13. _ChannelID = "goods20241230";
  14. }
  15. /// <summary>
  16. /// Generate HMAC signature for Coupang API.
  17. /// </summary>
  18. public static String GenerateHmac(String method, String url, String accessKey, String secretKey)
  19. {
  20. var dateTimeStamp = DateTime.Now.ToUniversalTime().ToString("yyMMddTHHmmssZ");
  21. var message = String.Format("{0}{1}{2}", dateTimeStamp, method, url.Replace("?", String.Empty));
  22. var keyBytes = ASCIIEncoding.Default.GetBytes(secretKey);
  23. var messageBytes = ASCIIEncoding.Default.GetBytes(message);
  24. var hashBytes = default(Byte[]);
  25. using (var hash = new HMACSHA256(keyBytes))
  26. {
  27. hashBytes = hash.ComputeHash(messageBytes);
  28. }
  29. var signature = BitConverter.ToString(hashBytes).Replace("-", String.Empty).ToLower();
  30. return String.Format("CEA algorithm=HmacSHA256, access-key={0}, signed-date={1}, signature={2}", accessKey, dateTimeStamp, signature);
  31. }
  32. // 골드박스
  33. public async Task<Product.Response> GetGoldBoxProduct()
  34. {
  35. Product.Response parseData = new();
  36. try
  37. {
  38. var urlPath = $"/v2/providers/affiliate_open_api/apis/openapi/v1/products/goldbox?subId={_ChannelID}"; // URI 경로만 전달
  39. // Step 1: URI 생성
  40. var baseUri = new Uri(_coupangPartners.APIUrl);
  41. var fullUri = baseUri + urlPath;
  42. // Step 2: HMAC 서명 생성
  43. var method = "GET";
  44. var signature = GenerateHmac(method, urlPath, _coupangPartners.APIKey, _coupangPartners.APISecret);
  45. // Step 3: HttpRequestMessage 생성
  46. using var request = new HttpRequestMessage(HttpMethod.Get, fullUri);
  47. request.Headers.Add("Authorization", signature);
  48. // Content-Type을 설정하려면 HttpContent를 추가
  49. request.Content = new StringContent(string.Empty); // 빈 본문
  50. // Step 4: HTTP 요청 보내기
  51. var response = await _coupangPartners.httpClient.SendAsync(request);
  52. if (response.IsSuccessStatusCode)
  53. {
  54. var jsonString = await response.Content.ReadAsStringAsync();
  55. parseData = JsonSerializer.Deserialize<Product.Response>(jsonString);
  56. if (parseData is null)
  57. {
  58. return new Product.Response();
  59. }
  60. }
  61. response.EnsureSuccessStatusCode();
  62. }
  63. catch (HttpRequestException e)
  64. {
  65. Console.WriteLine($"Request error: {e.Message}");
  66. }
  67. return parseData;
  68. }
  69. // 카테고리 별 상품
  70. public async Task<Product.Response> GetCategoryProduct(int categoryID)
  71. {
  72. Product.Response parseData = new();
  73. try
  74. {
  75. var urlPath = $"/v2/providers/affiliate_open_api/apis/openapi/v1/products/bestcategories/{categoryID}?limit=100&subId={_ChannelID}";
  76. // Step 1: URI 생성
  77. var baseUri = new Uri(_coupangPartners.APIUrl);
  78. var fullUri = baseUri + urlPath;
  79. // Step 2: HMAC 서명 생성
  80. var method = "GET";
  81. var signature = GenerateHmac(method, urlPath, _coupangPartners.APIKey, _coupangPartners.APISecret);
  82. // Step 3: HttpRequestMessage 생성
  83. using var request = new HttpRequestMessage(HttpMethod.Get, fullUri);
  84. request.Headers.Add("Authorization", signature);
  85. // Content-Type을 설정하려면 HttpContent를 추가
  86. request.Content = new StringContent(string.Empty); // 빈 본문
  87. // Step 4: HTTP 요청 보내기
  88. var response = await _coupangPartners.httpClient.SendAsync(request);
  89. var content = await response.Content.ReadAsStringAsync() ?? string.Empty;
  90. if (response.IsSuccessStatusCode)
  91. {
  92. if (!string.IsNullOrWhiteSpace(content))
  93. {
  94. var temp = JsonSerializer.Deserialize<Product.Response?>(content);
  95. if (temp != null)
  96. {
  97. parseData = temp;
  98. }
  99. }
  100. }
  101. else
  102. {
  103. Console.WriteLine($"Coupang API error ({response.StatusCode}): {content}");
  104. }
  105. response.EnsureSuccessStatusCode();
  106. }
  107. catch (HttpRequestException e)
  108. {
  109. Console.WriteLine($"Request error: {e.Message}");
  110. }
  111. return parseData;
  112. }
  113. // 쿠팡PL 상품
  114. public async Task<Product.Response> GetPlProduct(int brandID)
  115. {
  116. Product.Response parseData = new();
  117. try
  118. {
  119. var urlPath = $"/v2/providers/affiliate_open_api/apis/openapi/v1/products/coupangPL/{brandID}?limit=100&subId={_ChannelID}";
  120. // Step 1: URI 생성
  121. var baseUri = new Uri(_coupangPartners.APIUrl);
  122. var fullUri = baseUri + urlPath;
  123. // Step 2: HMAC 서명 생성
  124. var method = "GET";
  125. var signature = GenerateHmac(method, urlPath, _coupangPartners.APIKey, _coupangPartners.APISecret);
  126. // Step 3: HttpRequestMessage 생성
  127. using var request = new HttpRequestMessage(HttpMethod.Get, fullUri);
  128. request.Headers.Add("Authorization", signature);
  129. // Content-Type을 설정하려면 HttpContent를 추가
  130. request.Content = new StringContent(string.Empty); // 빈 본문
  131. // Step 4: HTTP 요청 보내기
  132. var response = await _coupangPartners.httpClient.SendAsync(request);
  133. var content = await response.Content.ReadAsStringAsync() ?? string.Empty;
  134. if (response.IsSuccessStatusCode)
  135. {
  136. if (!string.IsNullOrWhiteSpace(content))
  137. {
  138. var temp = JsonSerializer.Deserialize<Product.Response?>(content);
  139. if (temp != null)
  140. {
  141. parseData = temp;
  142. }
  143. }
  144. }
  145. else
  146. {
  147. Console.WriteLine($"Coupang API error ({response.StatusCode}): {content}");
  148. }
  149. response.EnsureSuccessStatusCode();
  150. }
  151. catch (HttpRequestException e)
  152. {
  153. Console.WriteLine($"Request error: {e.Message}");
  154. }
  155. return parseData;
  156. }
  157. // 검색 상품
  158. public async Task<Search.Response> GetSearchProduct(string keyword)
  159. {
  160. Search.Response parseData = new();
  161. try
  162. {
  163. var urlPath = $"/v2/providers/affiliate_open_api/apis/openapi/v1/products/search";
  164. var encodedKeyword = Uri.EscapeDataString(keyword);
  165. var queryParams = $"?keyword={encodedKeyword}&limit=10&srpLinkOnly=false&subId={_ChannelID}";
  166. var fullPathWithQuery = urlPath + queryParams;
  167. // Step 1: URI 생성
  168. var baseUri = new Uri(_coupangPartners.APIUrl);
  169. var fullUri = new Uri(baseUri, fullPathWithQuery);
  170. // Step 2: HMAC 서명 생성
  171. var method = "GET";
  172. var signature = GenerateHmac(method, fullPathWithQuery, _coupangPartners.APIKey, _coupangPartners.APISecret);
  173. // Step 3: HttpRequestMessage 생성
  174. using var request = new HttpRequestMessage(HttpMethod.Get, fullUri);
  175. request.Headers.Add("Authorization", signature);
  176. // Content-Type을 설정하려면 HttpContent를 추가
  177. request.Content = new StringContent(string.Empty); // 빈 본문
  178. // Step 4: HTTP 요청 보내기
  179. var response = await _coupangPartners.httpClient.SendAsync(request);
  180. var content = await response.Content.ReadAsStringAsync() ?? string.Empty;
  181. if (response.IsSuccessStatusCode)
  182. {
  183. if (!string.IsNullOrWhiteSpace(content))
  184. {
  185. var temp = JsonSerializer.Deserialize<Search.Response?>(content);
  186. if (temp != null)
  187. {
  188. parseData = temp;
  189. }
  190. }
  191. }
  192. else
  193. {
  194. Console.WriteLine($"Coupang API error ({response.StatusCode}): {content}");
  195. }
  196. response.EnsureSuccessStatusCode();
  197. }
  198. catch (HttpRequestException e)
  199. {
  200. Console.WriteLine($"Request error: {e.Message}");
  201. }
  202. return parseData;
  203. }
  204. }
  205. }