Compare commits

..

10 Commits

8 changed files with 140 additions and 19 deletions
+1
View File
@@ -1,6 +1,7 @@
[Pp]assword.txt [Pp]assword.txt
CVV.txt CVV.txt
CardDetails.txt CardDetails.txt
run.bat
# ---> VisualStudioCode # ---> VisualStudioCode
.vscode/* .vscode/*
+7
View File
@@ -7,6 +7,13 @@
<Nullable>enable</Nullable> <Nullable>enable</Nullable>
</PropertyGroup> </PropertyGroup>
<ItemGroup>
<Compile Remove="Scheduler/**" />
<Content Remove="Scheduler/**" />
<EmbeddedResource Remove="Scheduler/**" />
<None Remove="Scheduler/**" />
</ItemGroup>
<ItemGroup> <ItemGroup>
<PackageReference Include="Microsoft.Playwright" Version="1.59.0" /> <PackageReference Include="Microsoft.Playwright" Version="1.59.0" />
<PackageReference Include="Soenneker.Playwrights.Extensions.Stealth" Version="4.0.62" /> <PackageReference Include="Soenneker.Playwrights.Extensions.Stealth" Version="4.0.62" />
+1
View File
@@ -58,6 +58,7 @@ public static class GoogleMessages
string messageText = await messageElement.InnerTextAsync(); string messageText = await messageElement.InnerTextAsync();
await originalPage.BringToFrontAsync(); await originalPage.BringToFrontAsync();
await msgPage.CloseAsync();
return messageText; return messageText;
} }
} }
+50 -17
View File
@@ -28,25 +28,36 @@ if (argsLength == 2)
using var playwright = await Playwright.CreateAsync(); using var playwright = await Playwright.CreateAsync();
var context = await BrowserManager.GetContextAsync(playwright); var context = await BrowserManager.GetContextAsync(playwright);
var page = context.Pages.Count > 0 ? context.Pages[0] : await context.NewPageAsync(); var page = context.Pages.Count > 0 ? context.Pages[0] : await context.NewPageAsync();
await page.GotoAsync("https://my.hdc.mv/");
await page.GetByRole(AriaRole.Link, new() { Name = "Login" }).ClickAsync();
await page.GetByRole(AriaRole.Button, new() { Name = "eFaas" }).ClickAsync();
await page.GetByRole(AriaRole.Tab, new() { Name = "Password Login" }).ClickAsync(new () { Delay = 1000 });
await page.GetByPlaceholder("Username").FillAsync("A384347");
var password = File.ReadAllText("password.txt");
await page.GetByPlaceholder("Password").FillAsync(password);
await page.GetByRole(AriaRole.Button, new() { Name = "LOGIN" }).ClickAsync();
await page.GetByRole(AriaRole.Button, new() { Name = "CONTINUE" }).ClickAsync(new () { Delay = 1000 });
await Task.Delay(5000);
var egovOtp = await GoogleMessages.GetEgovOTP(context, page); await page.GotoAsync("https://my.hdc.mv/");
if (egovOtp is not null) try
{
await page.GetByRole(AriaRole.Link, new() { Name = "Login" }).ClickAsync();
await page.GetByRole(AriaRole.Button, new() { Name = "eFaas" }).ClickAsync();
await page.GetByRole(AriaRole.Tab, new() { Name = "Password Login" }).ClickAsync(new () { Delay = 1000 });
await page.GetByPlaceholder("Username").FillAsync("A384347");
var password = File.ReadAllText("password.txt");
await page.GetByPlaceholder("Password").FillAsync(password);
await page.GetByRole(AriaRole.Button, new() { Name = "LOGIN" }).ClickAsync();
await page.GetByRole(AriaRole.Button, new() { Name = "CONTINUE" }).ClickAsync(new () { Delay = 1000 });
await Task.Delay(5000);
var egovOtp = await GoogleMessages.GetEgovOTP(context, page);
if (egovOtp is not null)
Console.WriteLine($"Extracted eGov OTP: {egovOtp}"); Console.WriteLine($"Extracted eGov OTP: {egovOtp}");
else else
return; return;
await page.GetByPlaceholder("OTP").FillAsync(egovOtp); await page.GetByPlaceholder("OTP").FillAsync(egovOtp);
await page.GetByRole(AriaRole.Button, new() { Name = "CONTINUE" }).ClickAsync(); await page.GetByRole(AriaRole.Button, new() { Name = "CONTINUE" }).ClickAsync();
}
catch (Exception ex)
{
Console.WriteLine($"Error during login: {ex.Message}\nAssuming already logged in, proceeding with booking...");
await page.CloseAsync();
return;
}
await page.GetByText("Make a booking").ClickAsync(); await page.GetByText("Make a booking").ClickAsync();
await page.GetByText("Fahiveni").ClickAsync(); await page.GetByText("Fahiveni").ClickAsync();
await page.GetByText("Sports and Leisure").ClickAsync(); await page.GetByText("Sports and Leisure").ClickAsync();
@@ -58,11 +69,13 @@ var upperBound = new DateTime(currentDateTime.Year, currentDateTime.Month, curre
hour: 8, minute: 15, second: 0); hour: 8, minute: 15, second: 0);
var lowerBound = new DateTime(currentDateTime.Year, currentDateTime.Month, currentDateTime.Day, var lowerBound = new DateTime(currentDateTime.Year, currentDateTime.Month, currentDateTime.Day,
hour: 7, minute: 59, second: 59, millisecond: 800); hour: 8, minute: 0, second: 0, millisecond: 0);
var lastReloadTime = DateTime.Now; var lastReloadTime = DateTime.Now;
while (!debug && (currentDateTime < lowerBound || currentDateTime >= upperBound)) while (!debug && (currentDateTime < lowerBound || currentDateTime >= upperBound))
{ {
//Don't reload if 10 sec away from 8AM
//Reload every 2 minutes to avoid logout
if(lowerBound - currentDateTime > TimeSpan.FromSeconds(10) && if(lowerBound - currentDateTime > TimeSpan.FromSeconds(10) &&
lastReloadTime - currentDateTime > TimeSpan.FromMinutes(2)) lastReloadTime - currentDateTime > TimeSpan.FromMinutes(2))
{ {
@@ -74,8 +87,22 @@ while (!debug && (currentDateTime < lowerBound || currentDateTime >= upperBound)
currentDateTime = DateTime.Now; currentDateTime = DateTime.Now;
} }
// Click the next button to load tomorrow's time slots
await page.Locator("div.flex.items-center.justify-end > div > div:nth-child(2) > svg").ClickAsync(); await page.Locator("div.flex.items-center.justify-end > div > div:nth-child(2) > svg").ClickAsync();
await page.Locator("div.grid.bg-gray-100.rounded-lg.anim").Filter(new () { HasText = endTime }).ClickAsync(); try
{
Console.WriteLine($"{DateTime.Now:HH:mm:ss}: Clicking next for tomorrow's time slots...");
await page.Locator("div.grid.bg-gray-100.rounded-lg.anim").Filter(new () { HasText = endTime }).ClickAsync( new() { Timeout = 2000 });
Console.WriteLine($"{DateTime.Now:HH:mm:ss}: Timeslot found and clicked.");
}
catch (PlaywrightException)
{
Console.WriteLine($"{DateTime.Now:HH:mm:ss}: Could not find the specified end time slot: {endTime}");
await page.ScreenshotAsync(new PageScreenshotOptions { Path = "ErrorScreenshot.png" });
await page.CloseAsync();
return;
}
await page.GetByRole(AriaRole.Button, new() { Name = "Submit" }).ClickAsync(); await page.GetByRole(AriaRole.Button, new() { Name = "Submit" }).ClickAsync();
await page.GetByRole(AriaRole.Link, new() { Name = "Proceed to payment" }).ClickAsync(); await page.GetByRole(AriaRole.Link, new() { Name = "Proceed to payment" }).ClickAsync();
await page.GetByRole(AriaRole.Button, new() { Name = "Pay Now" }).ClickAsync(new () { Delay = 3500 }); await page.GetByRole(AriaRole.Button, new() { Name = "Pay Now" }).ClickAsync(new () { Delay = 3500 });
@@ -93,10 +120,16 @@ var bmlOtp = await GoogleMessages.GetBMLOTP(context, page);
if (bmlOtp is not null) if (bmlOtp is not null)
Console.WriteLine($"Extracted BML OTP: {bmlOtp}"); Console.WriteLine($"Extracted BML OTP: {bmlOtp}");
else else
{
await page.CloseAsync();
return; return;
}
await page.Locator("input[name='otpValue']").FillAsync(bmlOtp); await page.Locator("input[name='otpValue']").FillAsync(bmlOtp);
await page.GetByRole(AriaRole.Button, new() { Name = "CONFIRM" }).ClickAsync(); await page.GetByRole(AriaRole.Button, new() { Name = "CONFIRM" }).ClickAsync();
await Task.Delay(15000); await Task.Delay(15000);
var hdcMessage = await GoogleMessages.GetMessageFromHDC(context, page); var hdcMessage = await GoogleMessages.GetMessageFromHDC(context, page);
Console.WriteLine($"Message from HDC: {hdcMessage}"); Console.WriteLine($"Message from HDC: {hdcMessage}");
await page.CloseAsync();
+2
View File
@@ -9,3 +9,5 @@ In the future perhaps it would be much better if I can deploy as a webapp that c
Where `22:00` is the end time of the prefered time. `22:00` is the slot from `21:00 - 22:00`. Where `22:00` is the end time of the prefered time. `22:00` is the slot from `21:00 - 22:00`.
Preferably the application will be run close to 8AM around 7:45AM so that there's need to perpetually keep refreshing the site for a long time. Preferably the application will be run close to 8AM around 7:45AM so that there's need to perpetually keep refreshing the site for a long time.
**``OWO``**
+1
View File
@@ -0,0 +1 @@
C:\Users\Shaamil\Desktop\BadmintonBooker\Scheduler
+59
View File
@@ -0,0 +1,59 @@
using System.Reflection;
using System.Text;
using Microsoft.Win32.TaskScheduler;
if(args.Length != 1)
{
Console.WriteLine("Usage: Scheduler <timeslot>");
return;
}
var timeslot = args[0];
if (!TimeSpan.TryParse(timeslot, out var _))
{
Console.WriteLine("Invalid timeslot format. Use HH:mm");
return;
}
var dir = new DirectoryInfo(File.ReadAllText("AppSettings.txt"))!;
var builder = new StringBuilder();
builder.AppendLine($"cd {dir.Parent!.FullName}");
builder.AppendLine($"dotnet run -- {timeslot}");
File.WriteAllText("run.bat", builder.ToString());
using (TaskService taskService = new TaskService())
{
try
{
var existingTask = taskService.GetTask("BadmintonBooker");
if (existingTask != null)
{
taskService.RootFolder.DeleteTask("BadmintonBooker", false);
}
}
catch (Exception ex)
{
Console.WriteLine($"Error deleting existing task: {ex.Message}");
}
TaskDefinition taskDef = taskService.NewTask();
taskDef.RegistrationInfo.Description = "Scheduled Task with Credentials";
taskDef.Principal.RunLevel = TaskRunLevel.Highest;
taskDef.Triggers.Add(new TimeTrigger
{
StartBoundary = DateTime.Today.AddDays(1) + TimeSpan.Parse("07:40")
});
taskDef.Actions.Add(new ExecAction("cmd", $"/c \"{Path.Combine(dir.FullName, "run.bat")}\" > C:\\Output.log 2>&1"));
var pass = File.ReadAllText("Password.txt");
taskService.RootFolder.RegisterTaskDefinition(
"BadmintonBooker",
taskDef,
TaskCreation.CreateOrUpdate,
"Shaamil",
pass,
TaskLogonType.Password
);
}
+17
View File
@@ -0,0 +1,17 @@
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<OutputType>Exe</OutputType>
<TargetFramework>net10.0</TargetFramework>
<ImplicitUsings>enable</ImplicitUsings>
<Nullable>enable</Nullable>
</PropertyGroup>
<ItemGroup>
<None Include="AppSettings.txt" CopyToOutputDirectory="PreserveNewest" />
</ItemGroup>
<ItemGroup>
<PackageReference Include="TaskScheduler" Version="2.12.0" />
</ItemGroup>
</Project>