| | | 1 | | using System.Net.Http.Json; |
| | | 2 | | using Chronicis.Shared.DTOs; |
| | | 3 | | |
| | | 4 | | namespace Chronicis.Client.Services; |
| | | 5 | | |
| | | 6 | | /// <summary> |
| | | 7 | | /// Service for anonymous public API operations. |
| | | 8 | | /// Uses a separate HttpClient without authentication headers. |
| | | 9 | | /// </summary> |
| | | 10 | | public class PublicApiService : IPublicApiService |
| | | 11 | | { |
| | | 12 | | private readonly HttpClient _http; |
| | | 13 | | private readonly ILogger<PublicApiService> _logger; |
| | | 14 | | |
| | 0 | 15 | | public PublicApiService(HttpClient http, ILogger<PublicApiService> logger) |
| | | 16 | | { |
| | 0 | 17 | | _http = http; |
| | 0 | 18 | | _logger = logger; |
| | 0 | 19 | | } |
| | | 20 | | |
| | | 21 | | public async Task<WorldDetailDto?> GetPublicWorldAsync(string publicSlug) |
| | | 22 | | { |
| | | 23 | | try |
| | | 24 | | { |
| | 0 | 25 | | var response = await _http.GetAsync($"public/worlds/{publicSlug}"); |
| | | 26 | | |
| | 0 | 27 | | if (response.IsSuccessStatusCode) |
| | | 28 | | { |
| | 0 | 29 | | return await response.Content.ReadFromJsonAsync<WorldDetailDto>(); |
| | | 30 | | } |
| | | 31 | | |
| | 0 | 32 | | if (response.StatusCode == System.Net.HttpStatusCode.NotFound) |
| | | 33 | | { |
| | 0 | 34 | | _logger.LogDebug("Public world not found: {PublicSlug}", publicSlug); |
| | 0 | 35 | | return null; |
| | | 36 | | } |
| | | 37 | | |
| | 0 | 38 | | _logger.LogWarning("Failed to get public world {PublicSlug}: {StatusCode}", |
| | 0 | 39 | | publicSlug, response.StatusCode); |
| | 0 | 40 | | return null; |
| | | 41 | | } |
| | 0 | 42 | | catch (Exception ex) |
| | | 43 | | { |
| | 0 | 44 | | _logger.LogError(ex, "Error getting public world {PublicSlug}", publicSlug); |
| | 0 | 45 | | return null; |
| | | 46 | | } |
| | 0 | 47 | | } |
| | | 48 | | |
| | | 49 | | public async Task<List<ArticleTreeDto>> GetPublicArticleTreeAsync(string publicSlug) |
| | | 50 | | { |
| | | 51 | | try |
| | | 52 | | { |
| | 0 | 53 | | var response = await _http.GetAsync($"public/worlds/{publicSlug}/articles"); |
| | | 54 | | |
| | 0 | 55 | | if (response.IsSuccessStatusCode) |
| | | 56 | | { |
| | 0 | 57 | | return await response.Content.ReadFromJsonAsync<List<ArticleTreeDto>>() |
| | 0 | 58 | | ?? new List<ArticleTreeDto>(); |
| | | 59 | | } |
| | | 60 | | |
| | 0 | 61 | | _logger.LogWarning("Failed to get public article tree for {PublicSlug}: {StatusCode}", |
| | 0 | 62 | | publicSlug, response.StatusCode); |
| | 0 | 63 | | return new List<ArticleTreeDto>(); |
| | | 64 | | } |
| | 0 | 65 | | catch (Exception ex) |
| | | 66 | | { |
| | 0 | 67 | | _logger.LogError(ex, "Error getting public article tree for {PublicSlug}", publicSlug); |
| | 0 | 68 | | return new List<ArticleTreeDto>(); |
| | | 69 | | } |
| | 0 | 70 | | } |
| | | 71 | | |
| | | 72 | | public async Task<ArticleDto?> GetPublicArticleAsync(string publicSlug, string articlePath) |
| | | 73 | | { |
| | | 74 | | try |
| | | 75 | | { |
| | 0 | 76 | | var response = await _http.GetAsync($"public/worlds/{publicSlug}/articles/{articlePath}"); |
| | | 77 | | |
| | 0 | 78 | | if (response.IsSuccessStatusCode) |
| | | 79 | | { |
| | 0 | 80 | | return await response.Content.ReadFromJsonAsync<ArticleDto>(); |
| | | 81 | | } |
| | | 82 | | |
| | 0 | 83 | | if (response.StatusCode == System.Net.HttpStatusCode.NotFound) |
| | | 84 | | { |
| | 0 | 85 | | _logger.LogDebug("Public article not found: {PublicSlug}/{ArticlePath}", publicSlug, articlePath); |
| | 0 | 86 | | return null; |
| | | 87 | | } |
| | | 88 | | |
| | 0 | 89 | | _logger.LogWarning("Failed to get public article {PublicSlug}/{ArticlePath}: {StatusCode}", |
| | 0 | 90 | | publicSlug, articlePath, response.StatusCode); |
| | 0 | 91 | | return null; |
| | | 92 | | } |
| | 0 | 93 | | catch (Exception ex) |
| | | 94 | | { |
| | 0 | 95 | | _logger.LogError(ex, "Error getting public article {PublicSlug}/{ArticlePath}", publicSlug, articlePath); |
| | 0 | 96 | | return null; |
| | | 97 | | } |
| | 0 | 98 | | } |
| | | 99 | | |
| | | 100 | | public async Task<string?> ResolvePublicArticlePathAsync(string publicSlug, Guid articleId) |
| | | 101 | | { |
| | | 102 | | try |
| | | 103 | | { |
| | 0 | 104 | | var response = await _http.GetAsync($"public/worlds/{publicSlug}/articles/resolve/{articleId}"); |
| | | 105 | | |
| | 0 | 106 | | if (response.IsSuccessStatusCode) |
| | | 107 | | { |
| | 0 | 108 | | return await response.Content.ReadAsStringAsync(); |
| | | 109 | | } |
| | | 110 | | |
| | 0 | 111 | | if (response.StatusCode == System.Net.HttpStatusCode.NotFound) |
| | | 112 | | { |
| | 0 | 113 | | _logger.LogDebug("Public article path not found: {PublicSlug}/{ArticleId}", publicSlug, articleId); |
| | 0 | 114 | | return null; |
| | | 115 | | } |
| | | 116 | | |
| | 0 | 117 | | _logger.LogWarning("Failed to resolve public article path {PublicSlug}/{ArticleId}: {StatusCode}", |
| | 0 | 118 | | publicSlug, articleId, response.StatusCode); |
| | 0 | 119 | | return null; |
| | | 120 | | } |
| | 0 | 121 | | catch (Exception ex) |
| | | 122 | | { |
| | 0 | 123 | | _logger.LogError(ex, "Error resolving public article path {PublicSlug}/{ArticleId}", publicSlug, articleId); |
| | 0 | 124 | | return null; |
| | | 125 | | } |
| | 0 | 126 | | } |
| | | 127 | | } |