Compare commits
5 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 92d5d9c2d3 | |||
| 47d106e288 | |||
| a5624897e9 | |||
| 11e75d036a | |||
| 2942da0c35 |
@@ -6,7 +6,7 @@ on:
|
|||||||
- main
|
- main
|
||||||
|
|
||||||
env:
|
env:
|
||||||
VERSION: 3.0.6
|
VERSION: 3.0.8
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
# ЧАСТЬ 1: Собираем образы и кладем в Gitea (чтобы делиться с ребятами)
|
# ЧАСТЬ 1: Собираем образы и кладем в Gitea (чтобы делиться с ребятами)
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
<Project>
|
<Project>
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
<Version>3.0.6</Version>
|
<Version>3.0.8</Version>
|
||||||
<TargetFramework>net10.0</TargetFramework>
|
<TargetFramework>net10.0</TargetFramework>
|
||||||
<LangVersion>preview</LangVersion>
|
<LangVersion>preview</LangVersion>
|
||||||
<Nullable>enable</Nullable>
|
<Nullable>enable</Nullable>
|
||||||
|
|||||||
+3
-3
@@ -49,7 +49,7 @@ services:
|
|||||||
crond -f
|
crond -f
|
||||||
|
|
||||||
bot:
|
bot:
|
||||||
image: git.codeanddice.ru/toutsu/gmrelay-bot:3.0.6
|
image: git.codeanddice.ru/toutsu/gmrelay-bot:3.0.8
|
||||||
restart: always
|
restart: always
|
||||||
depends_on:
|
depends_on:
|
||||||
db:
|
db:
|
||||||
@@ -67,7 +67,7 @@ services:
|
|||||||
retries: 3
|
retries: 3
|
||||||
|
|
||||||
discord:
|
discord:
|
||||||
image: git.codeanddice.ru/toutsu/gmrelay-discord-bot:3.0.6
|
image: git.codeanddice.ru/toutsu/gmrelay-discord-bot:3.0.8
|
||||||
restart: always
|
restart: always
|
||||||
depends_on:
|
depends_on:
|
||||||
db:
|
db:
|
||||||
@@ -84,7 +84,7 @@ services:
|
|||||||
retries: 3
|
retries: 3
|
||||||
|
|
||||||
web:
|
web:
|
||||||
image: git.codeanddice.ru/toutsu/gmrelay-web:3.0.6
|
image: git.codeanddice.ru/toutsu/gmrelay-web:3.0.8
|
||||||
restart: always
|
restart: always
|
||||||
depends_on:
|
depends_on:
|
||||||
db:
|
db:
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
using GmRelay.DiscordBot.Rendering;
|
using GmRelay.DiscordBot.Rendering;
|
||||||
|
using NetCord;
|
||||||
using NetCord.Rest;
|
using NetCord.Rest;
|
||||||
using NetCord.Services.ApplicationCommands;
|
using NetCord.Services.ApplicationCommands;
|
||||||
|
|
||||||
@@ -22,10 +23,44 @@ public class DiscordNewSessionCommand : ApplicationCommandModule<SlashCommandCon
|
|||||||
[SlashCommandParameter(Name = "seats", Description = "Maximum number of players")] long? seats = null,
|
[SlashCommandParameter(Name = "seats", Description = "Maximum number of players")] long? seats = null,
|
||||||
[SlashCommandParameter(Name = "link", Description = "Join link")] string? link = null)
|
[SlashCommandParameter(Name = "link", Description = "Join link")] string? link = null)
|
||||||
{
|
{
|
||||||
|
_logger.LogInformation(
|
||||||
|
"newsession called by user {UserId} ({UserType}) in guild {GuildId}, channel {ChannelId}",
|
||||||
|
Context.User.Id,
|
||||||
|
Context.User.GetType().Name,
|
||||||
|
Context.Interaction.GuildId,
|
||||||
|
Context.Channel?.Id);
|
||||||
|
|
||||||
var guildId = Context.Interaction.GuildId
|
var guildId = Context.Interaction.GuildId
|
||||||
?? throw new InvalidOperationException("This command can only be used in a guild.");
|
?? throw new InvalidOperationException("This command can only be used in a guild.");
|
||||||
var guild = await Context.Client.Rest.GetGuildAsync(guildId);
|
|
||||||
var member = await Context.Client.Rest.GetGuildUserAsync(guildId, Context.User.Id);
|
var member = Context.User as GuildInteractionUser;
|
||||||
|
if (member is null)
|
||||||
|
{
|
||||||
|
_logger.LogError("Context.User is not GuildInteractionUser. Actual type: {ActualType}", Context.User.GetType().Name);
|
||||||
|
throw new InvalidOperationException("Guild member data not available in interaction.");
|
||||||
|
}
|
||||||
|
|
||||||
|
var resolvedPermissions = (ulong)member.Permissions;
|
||||||
|
_logger.LogInformation("Resolved permissions for user {UserId}: {Permissions}", Context.User.Id, resolvedPermissions);
|
||||||
|
|
||||||
|
ulong guildOwnerId = 0;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var guild = await Context.Client.Rest.GetGuildAsync(guildId);
|
||||||
|
guildOwnerId = guild.OwnerId;
|
||||||
|
_logger.LogInformation("Guild owner id: {OwnerId}", guildOwnerId);
|
||||||
|
}
|
||||||
|
catch (RestException ex) when (ex.StatusCode == System.Net.HttpStatusCode.NotFound)
|
||||||
|
{
|
||||||
|
_logger.LogWarning(
|
||||||
|
ex,
|
||||||
|
"Bot is not a REST member of guild {GuildId}; using resolved permissions from interaction payload",
|
||||||
|
guildId);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "Unexpected error fetching guild {GuildId}", guildId);
|
||||||
|
}
|
||||||
|
|
||||||
var timeResult = DiscordNewSessionHandler.ParseTimeInput(time);
|
var timeResult = DiscordNewSessionHandler.ParseTimeInput(time);
|
||||||
if (!timeResult.IsSuccess)
|
if (!timeResult.IsSuccess)
|
||||||
@@ -35,54 +70,56 @@ public class DiscordNewSessionCommand : ApplicationCommandModule<SlashCommandCon
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
var resolvedPermissions = GetResolvedPermissions(guild, member);
|
// Defer the response to avoid Discord 3-second interaction timeout
|
||||||
|
await Context.Interaction.SendResponseAsync(InteractionCallback.DeferredMessage());
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
_logger.LogInformation("Creating session for guild {GuildId}, user {UserId}", guildId, Context.User.Id);
|
||||||
|
|
||||||
var view = await _handler.HandleAsync(
|
var view = await _handler.HandleAsync(
|
||||||
guildId: guild.Id.ToString(),
|
guildId: guildId.ToString(),
|
||||||
channelId: Context.Channel.Id.ToString(),
|
channelId: Context.Channel!.Id.ToString(),
|
||||||
userId: Context.User.Id,
|
userId: Context.User.Id,
|
||||||
userDisplayName: Context.User.GlobalName ?? Context.User.Username,
|
userDisplayName: Context.User.GlobalName ?? Context.User.Username,
|
||||||
resolvedPermissions: resolvedPermissions,
|
resolvedPermissions: resolvedPermissions,
|
||||||
guildOwnerId: guild.OwnerId,
|
guildOwnerId: guildOwnerId,
|
||||||
title: title,
|
title: title,
|
||||||
scheduledAt: timeResult.Value,
|
scheduledAt: timeResult.Value,
|
||||||
maxPlayers: seats is null ? null : (int)seats.Value,
|
maxPlayers: seats is null ? null : (int)seats.Value,
|
||||||
joinLink: link,
|
joinLink: link,
|
||||||
CancellationToken.None);
|
CancellationToken.None);
|
||||||
|
|
||||||
|
_logger.LogInformation("Session created successfully. Building render.");
|
||||||
|
|
||||||
var (embeds, actionRows) = DiscordSessionBatchRenderer.Render(view);
|
var (embeds, actionRows) = DiscordSessionBatchRenderer.Render(view);
|
||||||
await Context.Interaction.SendResponseAsync(
|
|
||||||
InteractionCallback.Message(new InteractionMessageProperties()
|
_logger.LogInformation("Sending success response.");
|
||||||
.WithContent(":white_check_mark: **Session created successfully!**")
|
|
||||||
.WithEmbeds(embeds)
|
await Context.Interaction.ModifyResponseAsync(message =>
|
||||||
.WithComponents(actionRows)));
|
{
|
||||||
|
message.Content = ":white_check_mark: **Session created successfully!**";
|
||||||
|
message.Embeds = embeds;
|
||||||
|
message.Components = actionRows;
|
||||||
|
});
|
||||||
|
|
||||||
|
_logger.LogInformation("Success response sent.");
|
||||||
}
|
}
|
||||||
catch (UnauthorizedAccessException ex)
|
catch (UnauthorizedAccessException ex)
|
||||||
{
|
{
|
||||||
await Context.Interaction.SendResponseAsync(
|
_logger.LogWarning(ex, "Unauthorized session creation attempt by user {UserId}", Context.User.Id);
|
||||||
InteractionCallback.Message($":no_entry: {ex.Message}"));
|
await Context.Interaction.ModifyResponseAsync(message =>
|
||||||
|
{
|
||||||
|
message.Content = $":no_entry: {ex.Message}";
|
||||||
|
});
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
_logger.LogError(ex, "Failed to create session for user {UserId} in guild {GuildId}", Context.User.Id, guild.Id);
|
_logger.LogError(ex, "Failed to create session for user {UserId} in guild {GuildId}", Context.User.Id, guildId);
|
||||||
await Context.Interaction.SendResponseAsync(
|
await Context.Interaction.ModifyResponseAsync(message =>
|
||||||
InteractionCallback.Message(":boom: An error occurred while creating the session."));
|
{
|
||||||
|
message.Content = ":boom: An error occurred while creating the session.";
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static ulong GetResolvedPermissions(NetCord.Rest.RestGuild guild, NetCord.GuildUser member)
|
|
||||||
{
|
|
||||||
if (member is null)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
ulong resolved = 0;
|
|
||||||
foreach (var roleId in member.RoleIds)
|
|
||||||
{
|
|
||||||
if (guild.Roles.TryGetValue(roleId, out var role))
|
|
||||||
resolved |= (ulong)role.Permissions;
|
|
||||||
}
|
|
||||||
return resolved;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
namespace GmRelay.DiscordBot.Features.Sessions;
|
namespace GmRelay.DiscordBot.Features.Sessions;
|
||||||
|
|
||||||
|
using NetCord;
|
||||||
using NetCord.Rest;
|
using NetCord.Rest;
|
||||||
using NetCord.Services.ApplicationCommands;
|
using NetCord.Services.ApplicationCommands;
|
||||||
|
|
||||||
@@ -22,10 +23,43 @@ public class DiscordRescheduleCommand : ApplicationCommandModule<SlashCommandCon
|
|||||||
[SlashCommandParameter(Name = "option3", Description = "Third time option (optional)")] string? option3 = null,
|
[SlashCommandParameter(Name = "option3", Description = "Third time option (optional)")] string? option3 = null,
|
||||||
[SlashCommandParameter(Name = "deadline", Description = "Voting deadline (YYYY-MM-DD HH:mm)")] string deadline = "")
|
[SlashCommandParameter(Name = "deadline", Description = "Voting deadline (YYYY-MM-DD HH:mm)")] string deadline = "")
|
||||||
{
|
{
|
||||||
|
_logger.LogInformation(
|
||||||
|
"reschedule called by user {UserId} ({UserType}) in guild {GuildId}",
|
||||||
|
Context.User.Id,
|
||||||
|
Context.User.GetType().Name,
|
||||||
|
Context.Interaction.GuildId);
|
||||||
|
|
||||||
var guildId = Context.Interaction.GuildId
|
var guildId = Context.Interaction.GuildId
|
||||||
?? throw new InvalidOperationException("This command can only be used in a guild.");
|
?? throw new InvalidOperationException("This command can only be used in a guild.");
|
||||||
var guild = await Context.Client.Rest.GetGuildAsync(guildId);
|
|
||||||
var member = await Context.Client.Rest.GetGuildUserAsync(guildId, Context.User.Id);
|
var member = Context.User as GuildInteractionUser;
|
||||||
|
if (member is null)
|
||||||
|
{
|
||||||
|
_logger.LogError("Context.User is not GuildInteractionUser. Actual type: {ActualType}", Context.User.GetType().Name);
|
||||||
|
throw new InvalidOperationException("Guild member data not available in interaction.");
|
||||||
|
}
|
||||||
|
|
||||||
|
var resolvedPermissions = (ulong)member.Permissions;
|
||||||
|
_logger.LogInformation("Resolved permissions for user {UserId}: {Permissions}", Context.User.Id, resolvedPermissions);
|
||||||
|
|
||||||
|
ulong guildOwnerId = 0;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var guild = await Context.Client.Rest.GetGuildAsync(guildId);
|
||||||
|
guildOwnerId = guild.OwnerId;
|
||||||
|
_logger.LogInformation("Guild owner id: {OwnerId}", guildOwnerId);
|
||||||
|
}
|
||||||
|
catch (RestException ex) when (ex.StatusCode == System.Net.HttpStatusCode.NotFound)
|
||||||
|
{
|
||||||
|
_logger.LogWarning(
|
||||||
|
ex,
|
||||||
|
"Bot is not a REST member of guild {GuildId}; using resolved permissions from interaction payload",
|
||||||
|
guildId);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.LogError(ex, "Unexpected error fetching guild {GuildId}", guildId);
|
||||||
|
}
|
||||||
|
|
||||||
if (!Guid.TryParse(sessionIdText, out var sessionId))
|
if (!Guid.TryParse(sessionIdText, out var sessionId))
|
||||||
{
|
{
|
||||||
@@ -66,55 +100,55 @@ public class DiscordRescheduleCommand : ApplicationCommandModule<SlashCommandCon
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
var resolvedPermissions = GetResolvedPermissions(guild, member);
|
// Defer the response to avoid Discord 3-second interaction timeout
|
||||||
|
await Context.Interaction.SendResponseAsync(InteractionCallback.DeferredMessage());
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
_logger.LogInformation("Initiating reschedule for session {SessionId} in guild {GuildId}", sessionId, guildId);
|
||||||
|
|
||||||
var result = await _handler.HandleAsync(
|
var result = await _handler.HandleAsync(
|
||||||
guildId: guild.Id.ToString(),
|
guildId: guildId.ToString(),
|
||||||
channelId: Context.Channel.Id.ToString(),
|
channelId: Context.Channel!.Id.ToString(),
|
||||||
userId: Context.User.Id,
|
userId: Context.User.Id,
|
||||||
userDisplayName: Context.User.GlobalName ?? Context.User.Username,
|
userDisplayName: Context.User.GlobalName ?? Context.User.Username,
|
||||||
resolvedPermissions: resolvedPermissions,
|
resolvedPermissions: resolvedPermissions,
|
||||||
guildOwnerId: guild.OwnerId,
|
guildOwnerId: guildOwnerId,
|
||||||
sessionId: sessionId,
|
sessionId: sessionId,
|
||||||
options: parsedOptions,
|
options: parsedOptions,
|
||||||
deadline: deadlineResult.Value,
|
deadline: deadlineResult.Value,
|
||||||
CancellationToken.None);
|
CancellationToken.None);
|
||||||
|
|
||||||
await Context.Interaction.SendResponseAsync(
|
_logger.LogInformation("Reschedule voting started for session {SessionId}, proposal {ProposalId}", sessionId, result.ProposalId);
|
||||||
InteractionCallback.Message(
|
|
||||||
$"🗳 Голосование за перенос запущено! Дедлайн: {deadlineResult.Value:yyyy-MM-dd HH:mm} UTC."));
|
await Context.Interaction.ModifyResponseAsync(message =>
|
||||||
|
{
|
||||||
|
message.Content = $"🗳 Голосование за перенос запущено! Дедлайн: {deadlineResult.Value:yyyy-MM-dd HH:mm} UTC.";
|
||||||
|
});
|
||||||
}
|
}
|
||||||
catch (UnauthorizedAccessException ex)
|
catch (UnauthorizedAccessException ex)
|
||||||
{
|
{
|
||||||
await Context.Interaction.SendResponseAsync(
|
_logger.LogWarning(ex, "Unauthorized reschedule attempt by user {UserId}", Context.User.Id);
|
||||||
InteractionCallback.Message($":no_entry: {ex.Message}"));
|
await Context.Interaction.ModifyResponseAsync(message =>
|
||||||
|
{
|
||||||
|
message.Content = $":no_entry: {ex.Message}";
|
||||||
|
});
|
||||||
}
|
}
|
||||||
catch (InvalidOperationException ex)
|
catch (InvalidOperationException ex)
|
||||||
{
|
{
|
||||||
await Context.Interaction.SendResponseAsync(
|
_logger.LogWarning(ex, "Invalid reschedule request by user {UserId}", Context.User.Id);
|
||||||
InteractionCallback.Message($":warning: {ex.Message}"));
|
await Context.Interaction.ModifyResponseAsync(message =>
|
||||||
|
{
|
||||||
|
message.Content = $":warning: {ex.Message}";
|
||||||
|
});
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
_logger.LogError(ex, "Failed to initiate reschedule for session {SessionId}", sessionId);
|
_logger.LogError(ex, "Failed to initiate reschedule for session {SessionId}", sessionId);
|
||||||
await Context.Interaction.SendResponseAsync(
|
await Context.Interaction.ModifyResponseAsync(message =>
|
||||||
InteractionCallback.Message(":boom: Ошибка при запуске голосования."));
|
{
|
||||||
|
message.Content = ":boom: Ошибка при запуске голосования.";
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static ulong GetResolvedPermissions(NetCord.Rest.RestGuild guild, NetCord.GuildUser member)
|
|
||||||
{
|
|
||||||
if (member is null)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
ulong resolved = 0;
|
|
||||||
foreach (var roleId in member.RoleIds)
|
|
||||||
{
|
|
||||||
if (guild.Roles.TryGetValue(roleId, out var role))
|
|
||||||
resolved |= (ulong)role.Permissions;
|
|
||||||
}
|
|
||||||
return resolved;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -36,6 +36,8 @@ discordOptions.Validate();
|
|||||||
|
|
||||||
builder.Services.AddSingleton(discordOptions);
|
builder.Services.AddSingleton(discordOptions);
|
||||||
|
|
||||||
|
builder.Logging.AddConsole();
|
||||||
|
|
||||||
builder.Services.AddSingleton<NpgsqlDataSource>(sp =>
|
builder.Services.AddSingleton<NpgsqlDataSource>(sp =>
|
||||||
{
|
{
|
||||||
var config = sp.GetRequiredService<IConfiguration>();
|
var config = sp.GetRequiredService<IConfiguration>();
|
||||||
|
|||||||
@@ -73,7 +73,7 @@
|
|||||||
</button>
|
</button>
|
||||||
</form>
|
</form>
|
||||||
|
|
||||||
<div class="nav-version">v3.0.6</div>
|
<div class="nav-version">v3.0.8</div>
|
||||||
</div>
|
</div>
|
||||||
</Authorized>
|
</Authorized>
|
||||||
<NotAuthorized>
|
<NotAuthorized>
|
||||||
|
|||||||
@@ -145,7 +145,7 @@ public sealed class DiscordNewSessionHandlerTests
|
|||||||
var source = File.ReadAllText(commandPath);
|
var source = File.ReadAllText(commandPath);
|
||||||
|
|
||||||
Assert.Contains("DiscordSessionBatchRenderer.Render", source, StringComparison.Ordinal);
|
Assert.Contains("DiscordSessionBatchRenderer.Render", source, StringComparison.Ordinal);
|
||||||
Assert.Contains("WithEmbeds", source, StringComparison.Ordinal);
|
Assert.Contains("message.Embeds = embeds", source, StringComparison.Ordinal);
|
||||||
}
|
}
|
||||||
|
|
||||||
private static DateTimeOffset FutureDateAt1930()
|
private static DateTimeOffset FutureDateAt1930()
|
||||||
|
|||||||
@@ -61,7 +61,7 @@ public sealed class DiscordProjectStructureTests
|
|||||||
var prChecks = File.ReadAllText(Path.Combine(repoRoot, ".gitea", "workflows", "pr-checks.yml"));
|
var prChecks = File.ReadAllText(Path.Combine(repoRoot, ".gitea", "workflows", "pr-checks.yml"));
|
||||||
var deploy = File.ReadAllText(Path.Combine(repoRoot, ".gitea", "workflows", "deploy.yml"));
|
var deploy = File.ReadAllText(Path.Combine(repoRoot, ".gitea", "workflows", "deploy.yml"));
|
||||||
|
|
||||||
Assert.Contains("gmrelay-discord-bot:3.0.6", compose);
|
Assert.Contains("gmrelay-discord-bot:3.0.8", compose);
|
||||||
Assert.Contains("Discord__Token=${DISCORD_BOT_TOKEN:?Set DISCORD_BOT_TOKEN in .env}", compose);
|
Assert.Contains("Discord__Token=${DISCORD_BOT_TOKEN:?Set DISCORD_BOT_TOKEN in .env}", compose);
|
||||||
Assert.Contains("src/GmRelay.DiscordBot/Dockerfile", deploy);
|
Assert.Contains("src/GmRelay.DiscordBot/Dockerfile", deploy);
|
||||||
Assert.Contains("DISCORD_BOT_TOKEN", deploy);
|
Assert.Contains("DISCORD_BOT_TOKEN", deploy);
|
||||||
@@ -75,13 +75,13 @@ public sealed class DiscordProjectStructureTests
|
|||||||
{
|
{
|
||||||
var repoRoot = GetRepoRoot();
|
var repoRoot = GetRepoRoot();
|
||||||
|
|
||||||
Assert.Contains("<Version>3.0.6</Version>", File.ReadAllText(Path.Combine(repoRoot, "Directory.Build.props")));
|
Assert.Contains("<Version>3.0.8</Version>", File.ReadAllText(Path.Combine(repoRoot, "Directory.Build.props")));
|
||||||
Assert.Contains("VERSION: 3.0.6", File.ReadAllText(Path.Combine(repoRoot, ".gitea", "workflows", "deploy.yml")));
|
Assert.Contains("VERSION: 3.0.8", File.ReadAllText(Path.Combine(repoRoot, ".gitea", "workflows", "deploy.yml")));
|
||||||
Assert.Contains("gmrelay-bot:3.0.6", File.ReadAllText(Path.Combine(repoRoot, "compose.yaml")));
|
Assert.Contains("gmrelay-bot:3.0.8", File.ReadAllText(Path.Combine(repoRoot, "compose.yaml")));
|
||||||
Assert.Contains("gmrelay-web:3.0.6", File.ReadAllText(Path.Combine(repoRoot, "compose.yaml")));
|
Assert.Contains("gmrelay-web:3.0.8", File.ReadAllText(Path.Combine(repoRoot, "compose.yaml")));
|
||||||
Assert.Contains("gmrelay-discord-bot:3.0.6", File.ReadAllText(Path.Combine(repoRoot, "compose.yaml")));
|
Assert.Contains("gmrelay-discord-bot:3.0.8", File.ReadAllText(Path.Combine(repoRoot, "compose.yaml")));
|
||||||
Assert.Contains(
|
Assert.Contains(
|
||||||
"v3.0.6",
|
"v3.0.8",
|
||||||
File.ReadAllText(Path.Combine(repoRoot, "src", "GmRelay.Web", "Components", "Layout", "NavMenu.razor")));
|
File.ReadAllText(Path.Combine(repoRoot, "src", "GmRelay.Web", "Components", "Layout", "NavMenu.razor")));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user