Compare commits
6 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 3aa8b85c59 | |||
| 869b0b466e | |||
| 430c5fd2b8 | |||
| 28bad6c3c1 | |||
| 7684e8869b | |||
| 99eb077d14 |
@@ -1,7 +1,6 @@
|
|||||||
[Pp]assword.txt
|
[Pp]assword.txt
|
||||||
CVV.txt
|
CVV.txt
|
||||||
CardDetails.txt
|
CardDetails.txt
|
||||||
run.bat
|
|
||||||
|
|
||||||
# ---> VisualStudioCode
|
# ---> VisualStudioCode
|
||||||
.vscode/*
|
.vscode/*
|
||||||
|
|||||||
@@ -7,13 +7,6 @@
|
|||||||
<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" />
|
||||||
|
|||||||
@@ -58,7 +58,6 @@ 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;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
+2
-35
@@ -28,10 +28,7 @@ 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.GotoAsync("https://my.hdc.mv/");
|
||||||
try
|
|
||||||
{
|
|
||||||
await page.GetByRole(AriaRole.Link, new() { Name = "Login" }).ClickAsync();
|
await page.GetByRole(AriaRole.Link, new() { Name = "Login" }).ClickAsync();
|
||||||
await page.GetByRole(AriaRole.Button, new() { Name = "eFaas" }).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.GetByRole(AriaRole.Tab, new() { Name = "Password Login" }).ClickAsync(new () { Delay = 1000 });
|
||||||
@@ -50,14 +47,6 @@ try
|
|||||||
|
|
||||||
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();
|
||||||
@@ -69,13 +58,11 @@ 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: 8, minute: 0, second: 0, millisecond: 0);
|
hour: 7, minute: 59, second: 59, millisecond: 800);
|
||||||
|
|
||||||
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))
|
||||||
{
|
{
|
||||||
@@ -87,22 +74,8 @@ 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();
|
||||||
try
|
await page.Locator("div.grid.bg-gray-100.rounded-lg.anim").Filter(new () { HasText = endTime }).ClickAsync();
|
||||||
{
|
|
||||||
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 });
|
||||||
@@ -120,16 +93,10 @@ 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();
|
|
||||||
@@ -9,5 +9,3 @@ 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 +0,0 @@
|
|||||||
C:\Users\Shaamil\Desktop\BadmintonBooker\Scheduler
|
|
||||||
@@ -1,59 +0,0 @@
|
|||||||
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
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -1,17 +0,0 @@
|
|||||||
<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>
|
|
||||||
Reference in New Issue
Block a user