Files
GmRelayBot/tests/GmRelay.Bot.Tests/Web/PublicationModeTests.cs
Toutsu 6cb2fbe610
PR Checks / test-and-build (pull_request) Successful in 7m28s
feat(web): add private club showcases with membership flow (v3.7.0)
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>
2026-06-03 11:09:22 +03:00

80 lines
3.7 KiB
C#

namespace GmRelay.Bot.Tests.Web;
public sealed class PublicationModeTests
{
[Fact]
public void PublicationMode_ShouldHaveFourValues()
{
var values = Enum.GetValues<GmRelay.Shared.Domain.PublicationMode>();
Assert.Equal(4, values.Length);
Assert.Contains(GmRelay.Shared.Domain.PublicationMode.None, values);
Assert.Contains(GmRelay.Shared.Domain.PublicationMode.Catalog, values);
Assert.Contains(GmRelay.Shared.Domain.PublicationMode.ClubOnly, values);
Assert.Contains(GmRelay.Shared.Domain.PublicationMode.Both, values);
}
[Fact]
public async Task MigrationV030_ShouldAddClubMembershipsTable()
{
var migration = await ReadRepositoryFileAsync("src/GmRelay.Bot/Migrations/V030__add_club_memberships_and_publication_mode.sql");
Assert.Contains("CREATE TABLE club_memberships", migration, StringComparison.Ordinal);
Assert.Contains("status", migration, StringComparison.Ordinal);
Assert.Contains("role", migration, StringComparison.Ordinal);
Assert.Contains("Pending", migration, StringComparison.Ordinal);
Assert.Contains("Active", migration, StringComparison.Ordinal);
Assert.Contains("Rejected", migration, StringComparison.Ordinal);
Assert.Contains("Left", migration, StringComparison.Ordinal);
Assert.Contains("ux_club_memberships_one_active", migration, StringComparison.Ordinal);
Assert.Contains("Member", migration, StringComparison.Ordinal);
}
[Fact]
public async Task MigrationV030_ShouldReplaceIsPublicWithPublicationModeEnum()
{
var migration = await ReadRepositoryFileAsync("src/GmRelay.Bot/Migrations/V030__add_club_memberships_and_publication_mode.sql");
Assert.Contains("ADD COLUMN publication_mode", migration, StringComparison.Ordinal);
Assert.Contains("ck_sessions_publication_mode", migration, StringComparison.Ordinal);
Assert.Contains("'None', 'Catalog', 'ClubOnly', 'Both'", migration, StringComparison.Ordinal);
Assert.Contains("UPDATE sessions SET publication_mode = 'Both' WHERE is_public = true", migration, StringComparison.Ordinal);
Assert.Contains("UPDATE sessions SET publication_mode = 'None' WHERE is_public = false", migration, StringComparison.Ordinal);
Assert.Contains("DROP COLUMN is_public", migration, StringComparison.Ordinal);
}
[Fact]
public async Task MigrationV030_ShouldRecreatePartialIndexUsingPublicationMode()
{
var migration = await ReadRepositoryFileAsync("src/GmRelay.Bot/Migrations/V030__add_club_memberships_and_publication_mode.sql");
Assert.Contains("ix_sessions_public_schedule", migration, StringComparison.Ordinal);
Assert.Contains("publication_mode IN ('Catalog', 'Both')", migration, StringComparison.Ordinal);
}
[Fact]
public async Task MigrationV030_ShouldAddPortfolioPublicationModeColumn()
{
var migration = await ReadRepositoryFileAsync("src/GmRelay.Bot/Migrations/V030__add_club_memberships_and_publication_mode.sql");
Assert.Contains("portfolio_games", migration, StringComparison.Ordinal);
Assert.Contains("ix_portfolio_games_showcase", migration, 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}'.");
}
}