feat(e2e): #147 group creation and bot invitation scenario

- Add GroupSetupScenario: create supergroup, invite GmRelay bot, send /start,
  wait for reply, then delete the group
- Extend TelegramUserClient with DeleteGroupAsync and channel cache
- Update Program.cs to run the scenario with cleanup in finally
- Update README status table and runner documentation

Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
This commit is contained in:
2026-06-16 12:17:58 +03:00
parent 4b0f328f2e
commit f4a61269c2
4 changed files with 106 additions and 15 deletions
+6 -3
View File
@@ -11,8 +11,8 @@ Tracked as a Gitea milestone: [E2E Automation](https://git.codeanddice.ru/toutsu
|-------|-------|--------|
| #144 | initData / Login Widget helper for mock Telegram auth | ✅ Done |
| #145 | Playwright tests for Blazor dashboard with mocked Telegram auth | ✅ Done |
| #146 | Telegram user client (MTProto) | 🚧 In progress |
| #147 | Automate group creation and bot invitation | ⏳ Planned |
| #146 | Telegram user client (MTProto) | ✅ Done |
| #147 | Automate group creation and bot invitation | 🚧 In progress |
| #148 | Scenario: /newsession from creation to publication | ⏳ Planned |
| #149 | Join/leave, waitlist, reschedule and notification scenarios | ⏳ Planned |
| #150 | Dashboard display and editing verification | ⏳ Planned |
@@ -36,6 +36,7 @@ tests/e2e/
├── GmRelay.E2E.Runner.csproj # C# console runner using WTelegramClient (MTProto)
├── Program.cs # Entry point for quick manual checks
├── TelegramUserClient.cs # Reusable MTProto user client wrapper
├── GroupSetupScenario.cs # Create group + invite bot + verify /start
├── RunnerConfig.cs # Configuration model
├── .env.example # Required environment variables
├── .gitignore # Ignore .env and session files
@@ -85,7 +86,7 @@ python tests/e2e/helpers/test_telegram_init_data.py
## Run the MTProto user client runner
The runner logs in to a real Telegram user account, creates a supergroup, and invites the test bot.
The runner logs in to a real Telegram user account, creates a supergroup, invites the test bot, sends `/start`, waits for a reply, and deletes the group.
1. Copy the example environment file and fill in real values:
```bash
@@ -115,6 +116,8 @@ The runner logs in to a real Telegram user account, creates a supergroup, and in
- Login as a Telegram user.
- Create a supergroup (`Channels_CreateChannel` with `megagroup: true`).
- Resolve a bot by username and invite it to the group.
- Send `/start` to the bot inside the group and wait for any reply.
- Delete the test supergroup after the scenario (cleanup).
- Send messages/commands and read recent messages.
- Wait for a bot reply.
+59
View File
@@ -0,0 +1,59 @@
namespace GmRelay.E2E.Runner;
/// <summary>
/// Automates the first step of every Telegram E2E scenario:
/// create a supergroup, invite the GmRelay bot, and verify the bot responds to /start.
/// </summary>
public sealed class GroupSetupScenario
{
private readonly TelegramUserClient _client;
private readonly RunnerConfig _config;
public GroupSetupScenario(TelegramUserClient client, RunnerConfig config)
{
_client = client;
_config = config;
}
public async Task<ScenarioResult> RunAsync(CancellationToken cancellationToken = default)
{
var group = await _client.CreateGroupAsync(
$"GmRelay E2E {DateTime.UtcNow:yyyyMMdd-HHmmss}",
"Automated test group for GmRelay E2E suite.",
cancellationToken);
Console.WriteLine($"[scenario] created group id={group.Id} title='{group.Title}'");
await _client.InviteBotToGroupAsync(group, _config.BotUsername, cancellationToken);
Console.WriteLine($"[scenario] invited @{_config.BotUsername}");
await _client.SendCommandAsync(group, "start", cancellationToken);
var reply = await _client.WaitForBotReplyAsync(
group,
containsText: null,
timeout: TimeSpan.FromSeconds(30),
cancellationToken);
if (reply is null)
throw new InvalidOperationException("Bot did not reply to /start in the group.");
Console.WriteLine($"[scenario] bot replied to /start (msg id={reply.id})");
return new ScenarioResult(group, reply.id);
}
public async Task CleanupAsync(ScenarioResult result, CancellationToken cancellationToken = default)
{
try
{
await _client.DeleteGroupAsync(result.Group, cancellationToken);
Console.WriteLine($"[scenario] deleted group id={result.Group.Id}");
}
catch (Exception ex)
{
Console.WriteLine($"[scenario] warning: failed to delete group id={result.Group.Id}: {ex.Message}");
}
}
}
public sealed record ScenarioResult(ChatGroup Group, int LastBotMessageId);
+23 -10
View File
@@ -12,16 +12,29 @@ var config = new RunnerConfig
BotToken = Environment.GetEnvironmentVariable("TELEGRAM_BOT_TOKEN")!,
};
using var runner = new TelegramUserClient(config);
await runner.ConnectAsync();
using var client = new TelegramUserClient(config);
await client.ConnectAsync();
Console.WriteLine("Connected as Telegram user. Creating test group...");
var group = await runner.CreateGroupAsync("GmRelay E2E Test Group");
Console.WriteLine($"Created group id={group.Id} title='{group.Title}'");
var scenario = new GroupSetupScenario(client, config);
ScenarioResult? result = null;
await runner.InviteBotToGroupAsync(group, config.BotUsername);
Console.WriteLine($"Invited @{config.BotUsername} to the group");
try
{
result = await scenario.RunAsync();
Console.WriteLine("Scenario completed successfully.");
}
catch (Exception ex)
{
Console.WriteLine($"Scenario failed: {ex.Message}");
throw;
}
finally
{
if (result is not null)
{
await scenario.CleanupAsync(result);
}
// Keep the process alive briefly so the session is persisted.
await Task.Delay(TimeSpan.FromSeconds(2));
Console.WriteLine("Done.");
// Keep the session file written to disk.
await Task.Delay(TimeSpan.FromSeconds(2));
}
+18 -2
View File
@@ -1,3 +1,4 @@
using System.Collections.Concurrent;
using TL;
using WTelegram;
@@ -7,6 +8,7 @@ public sealed class TelegramUserClient : IDisposable
{
private readonly Client _client;
private readonly RunnerConfig _config;
private readonly ConcurrentDictionary<long, Channel> _knownChannels = new();
public TelegramUserClient(RunnerConfig config)
{
@@ -22,11 +24,11 @@ public sealed class TelegramUserClient : IDisposable
public async Task<ChatGroup> CreateGroupAsync(string title, string about = "", CancellationToken cancellationToken = default)
{
// Creating a megagroup gives us an UpdatesBase with Chats immediately.
var updates = await _client.Channels_CreateChannel(title, about, megagroup: true);
var channel = updates.Chats.Values.OfType<Channel>().FirstOrDefault()
?? throw new InvalidOperationException("Failed to create a supergroup.");
_knownChannels[channel.id] = channel;
return new ChatGroup(channel.id, channel.title);
}
@@ -44,6 +46,13 @@ public sealed class TelegramUserClient : IDisposable
await _client.Channels_InviteToChannel(channel, new InputUserBase[] { inputUser });
}
public async Task DeleteGroupAsync(ChatGroup group, CancellationToken cancellationToken = default)
{
var channel = await ResolveChannelAsync(group.Id, cancellationToken);
await _client.Channels_DeleteChannel(channel);
_knownChannels.TryRemove(group.Id, out _);
}
public async Task SendMessageAsync(ChatGroup group, string text, CancellationToken cancellationToken = default)
{
var channel = await ResolveChannelAsync(group.Id, cancellationToken);
@@ -102,12 +111,19 @@ public sealed class TelegramUserClient : IDisposable
private async Task<Channel> ResolveChannelAsync(long channelId, CancellationToken cancellationToken = default)
{
if (_knownChannels.TryGetValue(channelId, out var cached))
return cached;
var dialogs = await _client.Messages_GetAllDialogs();
var channel = dialogs.chats.Values
.OfType<Channel>()
.FirstOrDefault(c => c.id == channelId);
return channel ?? throw new InvalidOperationException($"Could not resolve channel {channelId}.");
if (channel is null)
throw new InvalidOperationException($"Could not resolve channel {channelId}.");
_knownChannels[channelId] = channel;
return channel;
}
}