6cb2fbe610
PR Checks / test-and-build (pull_request) Successful in 7m28s
Implements Issue #110: game masters can now publish sessions exclusively to a club's private showcase, gated behind a member application and approval flow. Adds a 4-state publication_mode (None/Catalog/ClubOnly/Both) replacing the binary is_public, plus a club_memberships table with Pending/Active/Rejected/Left lifecycle and partial unique index ensuring a single Active row per (group, player). Highlights - V030 migration: club_memberships, publication_mode, drop is_public, recreate partial indexes, portfolio_games gains publication_mode. - PublicationMode enum + extensions in GmRelay.Shared. - ISessionStore gains 12 membership/showcase methods; AuthorizedMembershipService owns the membership flow with GM-only approve/reject authorization. - PublicClub / PublicMasterProfile / PublicSession: member- aware queries (ClubOnly visible only to Active members). - New pages: MyClubMemberships (/profile/memberships) and ClubApplications (/group/{id}/applications). - GroupDetails and EditSession switch from a bool toggle to a 4-state publication_mode selector. - NavMenu adds Moji kluby, PublicLayout adds Kluby. Tests: 4 new test files (PublicationMode, ClubMemberships, AuthorizedMembershipService, ClubShowcaseSource) + updates to PublicClubPages, AuthorizedSessionService/Portfolio service FakeSessionStore, CampaignTemplatesNavigation. 493 tests pass. Bump version 3.6.0 -> 3.7.0 across Directory.Build.props, compose.yaml, deploy.yml, NavMenu.razor. Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
105 lines
4.4 KiB
C#
105 lines
4.4 KiB
C#
namespace GmRelay.Bot.Tests.Web;
|
|
|
|
public sealed class ClubShowcaseSourceTests
|
|
{
|
|
[Fact]
|
|
public async Task PublicClubPage_ShouldRenderMembersOnlyBlock()
|
|
{
|
|
var page = await ReadRepositoryFileAsync("src/GmRelay.Web/Components/Pages/PublicClub.razor");
|
|
|
|
Assert.Contains("Игры для участников клуба", page, StringComparison.Ordinal);
|
|
Assert.Contains("viewerIsActiveMember", page, StringComparison.Ordinal);
|
|
Assert.Contains("members-only-section", page, StringComparison.Ordinal);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task PublicClubPage_ShouldRenderApplyAndLoginCtas()
|
|
{
|
|
var page = await ReadRepositoryFileAsync("src/GmRelay.Web/Components/Pages/PublicClub.razor");
|
|
|
|
Assert.Contains("Подать заявку", page, StringComparison.Ordinal);
|
|
Assert.Contains("Войти как участник", page, StringComparison.Ordinal);
|
|
Assert.Contains("applicationMessage", page, StringComparison.Ordinal);
|
|
Assert.Contains("ApplyForCurrentUserAsync", page, StringComparison.Ordinal);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task PublicClubPage_ShouldHideMembersOnlyBlockForAnonymous()
|
|
{
|
|
var page = await ReadRepositoryFileAsync("src/GmRelay.Web/Components/Pages/PublicClub.razor");
|
|
|
|
// Anonymous users must not see the members-only block content
|
|
Assert.Contains("viewerIsActiveMember", page, StringComparison.Ordinal);
|
|
// Login CTA appears when viewerPlayerId is null
|
|
Assert.Contains("viewerPlayerId is null", page, StringComparison.Ordinal);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task PublicLayout_ShouldExposeClubsLink()
|
|
{
|
|
var layout = await ReadRepositoryFileAsync("src/GmRelay.Web/Components/Layout/PublicLayout.razor");
|
|
|
|
Assert.Contains("href=\"/showcase\"", layout, StringComparison.Ordinal);
|
|
Assert.Contains("Клубы", layout, StringComparison.Ordinal);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task NavMenu_ShouldExposeMyClubsLink()
|
|
{
|
|
var menu = await ReadRepositoryFileAsync("src/GmRelay.Web/Components/Layout/NavMenu.razor");
|
|
|
|
Assert.Contains("href=\"profile/memberships\"", menu, StringComparison.Ordinal);
|
|
Assert.Contains("Мои клубы", menu, StringComparison.Ordinal);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task GroupDetails_ShouldExposeApplicationsLink()
|
|
{
|
|
var page = await ReadRepositoryFileAsync("src/GmRelay.Web/Components/Pages/GroupDetails.razor");
|
|
|
|
Assert.Contains("/applications", page, StringComparison.Ordinal);
|
|
Assert.Contains("Заявки участников", page, StringComparison.Ordinal);
|
|
Assert.Contains("pendingApplicationsCount", page, StringComparison.Ordinal);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task GroupDetails_ShouldUsePublicationModeSelectorNotBooleanToggle()
|
|
{
|
|
var page = await ReadRepositoryFileAsync("src/GmRelay.Web/Components/Pages/GroupDetails.razor");
|
|
|
|
Assert.DoesNotContain("SetSessionPublic(session.Id, !session.IsPublic)", page, StringComparison.Ordinal);
|
|
Assert.DoesNotContain("SetBatchPublic(batch, !batch.AllSessionsPublic)", page, StringComparison.Ordinal);
|
|
Assert.Contains("SetSessionPublicationMode", page, StringComparison.Ordinal);
|
|
Assert.Contains("SetBatchPublicationMode", page, StringComparison.Ordinal);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task EditSession_ShouldExposePublicationModeSelector()
|
|
{
|
|
var page = await ReadRepositoryFileAsync("src/GmRelay.Web/Components/Pages/EditSession.razor");
|
|
|
|
Assert.Contains("PublicationMode", page, StringComparison.Ordinal);
|
|
Assert.Contains("Режим публикации", page, StringComparison.Ordinal);
|
|
Assert.Contains("Catalog", page, StringComparison.Ordinal);
|
|
Assert.Contains("ClubOnly", page, StringComparison.Ordinal);
|
|
Assert.Contains("Both", page, StringComparison.Ordinal);
|
|
}
|
|
|
|
private static async Task<string> ReadRepositoryFileAsync(string relativePath)
|
|
{
|
|
var directory = new DirectoryInfo(AppContext.BaseDirectory);
|
|
while (directory is not null)
|
|
{
|
|
var candidate = Path.Combine(directory.FullName, relativePath);
|
|
if (File.Exists(candidate))
|
|
{
|
|
return await File.ReadAllTextAsync(candidate);
|
|
}
|
|
|
|
directory = directory.Parent;
|
|
}
|
|
|
|
throw new FileNotFoundException($"Could not locate repository file '{relativePath}'.");
|
|
}
|
|
}
|