Compare commits
2 Commits
9331519649
...
a970012209
Author | SHA1 | Date | |
---|---|---|---|
a970012209 | |||
1d36c8d165 |
@ -7,7 +7,7 @@ using Overflow.SouthernWater;
|
|||||||
|
|
||||||
var driver = new MongoClient("mongodb://localhost");
|
var driver = new MongoClient("mongodb://localhost");
|
||||||
|
|
||||||
var api = new SouthernWaterApi(new HttpClient());
|
var api = new SouthernWaterApi(new HttpClient(), NullLogger<SouthernWaterApi>.Instance);
|
||||||
await api.LoadApiUrl();
|
await api.LoadApiUrl();
|
||||||
|
|
||||||
var runner = new SouthernWaterApiJobRunnerPersisting(api, NullLogger<SouthernWaterApiJobRunner>.Instance, driver.GetDatabase(Static.DatabaseName));
|
var runner = new SouthernWaterApiJobRunnerPersisting(api, NullLogger<SouthernWaterApiJobRunner>.Instance, driver.GetDatabase(Static.DatabaseName));
|
||||||
|
@ -1,3 +1,6 @@
|
|||||||
|
using Microsoft.Extensions.Logging.Abstractions;
|
||||||
|
using Overflow.SouthernWater;
|
||||||
|
|
||||||
namespace Overflow.Test;
|
namespace Overflow.Test;
|
||||||
|
|
||||||
public class Tests
|
public class Tests
|
||||||
@ -10,7 +13,7 @@ public class Tests
|
|||||||
[Test]
|
[Test]
|
||||||
public async Task Test1()
|
public async Task Test1()
|
||||||
{
|
{
|
||||||
var southern = new SouthernWater.SouthernWaterApi(new HttpClient());
|
var southern = new SouthernWater.SouthernWaterApi(new HttpClient(), NullLogger<SouthernWaterApi>.Instance);
|
||||||
await southern.LoadApiUrl();
|
await southern.LoadApiUrl();
|
||||||
var spills = await southern.GetSpills();
|
var spills = await southern.GetSpills();
|
||||||
}
|
}
|
||||||
|
47
Overflow.Web.Client/Components/SpillViewDialog.razor
Normal file
47
Overflow.Web.Client/Components/SpillViewDialog.razor
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
@using Overflow.SouthernWater
|
||||||
|
@rendermode RenderMode.InteractiveAuto
|
||||||
|
|
||||||
|
<RadzenStack>
|
||||||
|
<RadzenStack Gap="0">
|
||||||
|
<RadzenText TextStyle="TextStyle.Overline" class="rz-display-flex rz-mt-2 rz-my-0">Bathing Site</RadzenText>
|
||||||
|
<RadzenText TextStyle="TextStyle.Body1"><b>@Spill.bathingSite</b></RadzenText>
|
||||||
|
<RadzenText TextStyle="TextStyle.Overline" class="rz-display-flex rz-mt-4 rz-mb-0">Outfall</RadzenText>
|
||||||
|
<RadzenText TextStyle="TextStyle.Body1"><b>@Spill.outfallName</b></RadzenText>
|
||||||
|
</RadzenStack>
|
||||||
|
<RadzenStack Gap="0">
|
||||||
|
<RadzenText TextStyle="TextStyle.Overline">Status: </RadzenText>
|
||||||
|
<RadzenStack Orientation="Orientation.Horizontal">
|
||||||
|
<RadzenText TextStyle="TextStyle.Body1"><b>@Spill.status</b></RadzenText>
|
||||||
|
@if (Spill.status == "Genuine")
|
||||||
|
{
|
||||||
|
<RadzenIcon Icon="verified" />
|
||||||
|
}
|
||||||
|
</RadzenStack>
|
||||||
|
<RadzenText TextStyle="TextStyle.Overline">Is Impacting</RadzenText>
|
||||||
|
<RadzenStack Orientation="Orientation.Horizontal">
|
||||||
|
@if (Spill.isImpacting)
|
||||||
|
{
|
||||||
|
<RadzenIcon Icon="check_circle" />
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
<RadzenIcon Icon="cancel" />
|
||||||
|
}
|
||||||
|
</RadzenStack>
|
||||||
|
</RadzenStack>
|
||||||
|
<RadzenStack Gap="0">
|
||||||
|
<RadzenText TextStyle="TextStyle.Overline" class="rz-display-flex rz-mt-2 rz-my-0">Start</RadzenText>
|
||||||
|
<RadzenText TextStyle="TextStyle.Body1"><b>@Spill.eventStart.ToLongDateString()</b> @Spill.eventStart.ToLongTimeString()</RadzenText>
|
||||||
|
<RadzenText TextStyle="TextStyle.Overline" class="rz-display-flex rz-mt-4 rz-mb-0">End</RadzenText>
|
||||||
|
<RadzenText TextStyle="TextStyle.Body1"><b>@Spill.eventStop.ToLongDateString()</b> @Spill.eventStop.ToLongTimeString()</RadzenText>
|
||||||
|
|
||||||
|
<RadzenStack Orientation="Orientation.Vertical">
|
||||||
|
<RadzenText TextStyle="TextStyle.Overline" class="rz-display-flex rz-mt-4 rz-mb-0">Duration</RadzenText>
|
||||||
|
<RadzenText TextStyle="TextStyle.Body1">@TimeSpan.FromMinutes(Spill.duration).ToString()</RadzenText>
|
||||||
|
</RadzenStack>
|
||||||
|
</RadzenStack>
|
||||||
|
</RadzenStack>
|
||||||
|
|
||||||
|
@code {
|
||||||
|
[Parameter] public Spill Spill { get; set; }
|
||||||
|
}
|
@ -10,7 +10,7 @@
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
<RadzenScheduler @ref=@scheduler SlotRender=@OnSlotRender style="height: 768px;" TItem="Spill" Data=@Job.Spills StartProperty="eventStart" EndProperty="eventStop"
|
<RadzenScheduler @ref=@scheduler SlotRender=@OnSlotRender style="height: 768px;" TItem="Spill" Data=@Spills StartProperty="eventStart" EndProperty="eventStop"
|
||||||
TextProperty="bathingSite" SelectedIndex="2"
|
TextProperty="bathingSite" SelectedIndex="2"
|
||||||
SlotSelect=@OnSlotSelect AppointmentSelect=@OnAppointmentSelect AppointmentRender=@OnAppointmentRender
|
SlotSelect=@OnSlotSelect AppointmentSelect=@OnAppointmentSelect AppointmentRender=@OnAppointmentRender
|
||||||
AppointmentMove=@OnAppointmentMove >
|
AppointmentMove=@OnAppointmentMove >
|
||||||
@ -26,9 +26,21 @@ else
|
|||||||
@code {
|
@code {
|
||||||
RadzenScheduler<Spill> scheduler;
|
RadzenScheduler<Spill> scheduler;
|
||||||
[Parameter] public SouthernWaterApiJob? Job { get; set; }
|
[Parameter] public SouthernWaterApiJob? Job { get; set; }
|
||||||
|
[Parameter] public bool GenuineOnly { get; set; } = true;
|
||||||
|
|
||||||
Dictionary<DateTime, string> events = new Dictionary<DateTime, string>();
|
private IEnumerable<Spill> Spills {
|
||||||
|
get
|
||||||
|
{
|
||||||
|
if (GenuineOnly)
|
||||||
|
{
|
||||||
|
return Job.Spills.Where(j => j.status == "Genuine");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return Job.Spills;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void OnSlotRender(SchedulerSlotRenderEventArgs args)
|
void OnSlotRender(SchedulerSlotRenderEventArgs args)
|
||||||
{
|
{
|
||||||
@ -63,24 +75,9 @@ else
|
|||||||
|
|
||||||
async Task OnAppointmentSelect(SchedulerAppointmentSelectEventArgs<Spill> args)
|
async Task OnAppointmentSelect(SchedulerAppointmentSelectEventArgs<Spill> args)
|
||||||
{
|
{
|
||||||
// var copy = new Appointment
|
_ = await DialogService.OpenAsync<SpillViewDialog>("View Spill", new Dictionary<string, object> { { "Spill", args.Data } });
|
||||||
// {
|
|
||||||
// Start = args.Data.Start,
|
await scheduler.Reload();
|
||||||
// End = args.Data.End,
|
|
||||||
// Text = args.Data.Text
|
|
||||||
// };
|
|
||||||
//
|
|
||||||
// var data = await DialogService.OpenAsync<EditAppointmentPage>("Edit Appointment", new Dictionary<string, object> { { "Appointment", copy } });
|
|
||||||
//
|
|
||||||
// if (data != null)
|
|
||||||
// {
|
|
||||||
// // Update the appointment
|
|
||||||
// args.Data.Start = data.Start;
|
|
||||||
// args.Data.End = data.End;
|
|
||||||
// args.Data.Text = data.Text;
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// await scheduler.Reload();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void OnAppointmentRender(SchedulerAppointmentRenderEventArgs<Spill> args)
|
void OnAppointmentRender(SchedulerAppointmentRenderEventArgs<Spill> args)
|
||||||
|
@ -45,4 +45,4 @@ else
|
|||||||
@code {
|
@code {
|
||||||
RadzenDataGrid<Spill> spillsGrid;
|
RadzenDataGrid<Spill> spillsGrid;
|
||||||
[Parameter] public SouthernWaterApiJob? Job { get; set; }
|
[Parameter] public SouthernWaterApiJob? Job { get; set; }
|
||||||
}
|
}
|
||||||
|
@ -16,4 +16,4 @@
|
|||||||
<a class="dismiss">🗙</a>
|
<a class="dismiss">🗙</a>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<RadzenComponents/>
|
<RadzenComponents @rendermode="InteractiveServer"/>
|
@ -15,15 +15,20 @@
|
|||||||
{
|
{
|
||||||
<RadzenText TextStyle="TextStyle.Body1">Last updated at <b>@job.EndTime</b></RadzenText>
|
<RadzenText TextStyle="TextStyle.Body1">Last updated at <b>@job.EndTime</b></RadzenText>
|
||||||
}
|
}
|
||||||
|
<RadzenStack Orientation="Orientation.Horizontal" AlignItems="AlignItems.Start" Wrap="FlexWrap.Wrap">
|
||||||
|
<RadzenCheckBox @bind-Value=@genuineOnly Name="genuineOnly" />
|
||||||
|
<RadzenLabel Text="Genuine Events Only" Component="genuineOnly" Style="margin-left: 8px; vertical-align: middle;" />
|
||||||
|
</RadzenStack>
|
||||||
</RadzenStack>
|
</RadzenStack>
|
||||||
</RadzenCard>
|
</RadzenCard>
|
||||||
|
|
||||||
<SpillsCalendar Job="@job" />
|
<SpillsCalendar Job="@job" GenuineOnly="@genuineOnly" />
|
||||||
|
|
||||||
@code {
|
@code {
|
||||||
private SouthernWaterApiJob? job;
|
private SouthernWaterApiJob? job;
|
||||||
[Inject] private IMongoDatabase database { get; set; }
|
[Inject] private IMongoDatabase database { get; set; }
|
||||||
// private bool showIds;
|
// private bool showIds;
|
||||||
|
private bool genuineOnly = true;
|
||||||
|
|
||||||
protected override async Task OnInitializedAsync()
|
protected override async Task OnInitializedAsync()
|
||||||
{
|
{
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
using System.Net.Http.Json;
|
using System.Net.Http.Json;
|
||||||
using System.Text.Json;
|
using System.Text.Json;
|
||||||
using System.Text.RegularExpressions;
|
using System.Text.RegularExpressions;
|
||||||
|
using Microsoft.Extensions.Logging;
|
||||||
using Quartz.Util;
|
using Quartz.Util;
|
||||||
|
|
||||||
namespace Overflow.SouthernWater;
|
namespace Overflow.SouthernWater;
|
||||||
@ -8,15 +9,17 @@ namespace Overflow.SouthernWater;
|
|||||||
public partial class SouthernWaterApi
|
public partial class SouthernWaterApi
|
||||||
{
|
{
|
||||||
private readonly HttpClient _client;
|
private readonly HttpClient _client;
|
||||||
|
private readonly ILogger<SouthernWaterApi> _logger;
|
||||||
|
|
||||||
private static readonly string spillsEndpoint = "Spills/GetHistoricSpills";
|
private static readonly string spillsEndpoint = "Spills/GetHistoricSpills";
|
||||||
|
|
||||||
private string baseUrl;
|
private string baseUrl;
|
||||||
private string apiKey;
|
private string apiKey;
|
||||||
|
|
||||||
public SouthernWaterApi(HttpClient client)
|
public SouthernWaterApi(HttpClient client, ILogger<SouthernWaterApi> logger)
|
||||||
{
|
{
|
||||||
_client = client;
|
_client = client;
|
||||||
|
_logger = logger;
|
||||||
}
|
}
|
||||||
|
|
||||||
[GeneratedRegex(@".*const APIURL = '(?<APIURL>.*)'.*\n.*const APIGWKEY = '(?<APIKEY>.*)'.*", RegexOptions.IgnoreCase)]
|
[GeneratedRegex(@".*const APIURL = '(?<APIURL>.*)'.*\n.*const APIGWKEY = '(?<APIKEY>.*)'.*", RegexOptions.IgnoreCase)]
|
||||||
@ -24,42 +27,58 @@ public partial class SouthernWaterApi
|
|||||||
|
|
||||||
public async Task LoadApiUrl()
|
public async Task LoadApiUrl()
|
||||||
{
|
{
|
||||||
var request = new HttpRequestMessage
|
var success = false;
|
||||||
|
|
||||||
|
while (!success)
|
||||||
{
|
{
|
||||||
RequestUri = new Uri("https://www.southernwater.co.uk/scripts/beachbuoyhistoricspillstable.js"),
|
try
|
||||||
Method = HttpMethod.Get,
|
|
||||||
Headers =
|
|
||||||
{
|
{
|
||||||
{"Accept", "*/*"},
|
var request = new HttpRequestMessage
|
||||||
{"Accept-Language", "en-GB,en;q=0.5"},
|
{
|
||||||
// {"Accept-Encoding", "gzip, deflate, br, zstd"},
|
RequestUri = new Uri("https://www.southernwater.co.uk/scripts/beachbuoyhistoricspillstable.js"),
|
||||||
{"Cache-Control", "no-cache"},
|
Method = HttpMethod.Get,
|
||||||
{"Connection", "keep-alive"},
|
Headers =
|
||||||
{"DNT", "1"},
|
{
|
||||||
{"User-Agent", "Mozilla/5.0 (Windows NT 10.0; rv:126.0) Gecko/20100101 Firefox/126.0"},
|
{"Accept", "*/*"},
|
||||||
{"Upgrade-Insecure-Requests", "1"},
|
{"Accept-Language", "en-GB,en;q=0.5"},
|
||||||
{"Referer", "https://www.southernwater.co.uk/our-region/clean-rivers-and-seas-task-force/beachbuoy-historic-release-table/"}
|
// {"Accept-Encoding", "gzip, deflate, br, zstd"},
|
||||||
|
{"Cache-Control", "no-cache"},
|
||||||
|
{"Connection", "keep-alive"},
|
||||||
|
{"DNT", "1"},
|
||||||
|
{"User-Agent", "Mozilla/5.0 (Windows NT 10.0; rv:126.0) Gecko/20100101 Firefox/126.0"},
|
||||||
|
{"Upgrade-Insecure-Requests", "1"},
|
||||||
|
{"Referer", "https://www.southernwater.co.uk/our-region/clean-rivers-and-seas-task-force/beachbuoy-historic-release-table/"}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
var content = await _client.SendAsync(request);
|
||||||
|
|
||||||
|
content.EnsureSuccessStatusCode();
|
||||||
|
|
||||||
|
success = true;
|
||||||
|
|
||||||
|
var contentString = await content.Content.ReadAsStringAsync();
|
||||||
|
|
||||||
|
Match m = ApiUrlAndKey().Match(contentString);
|
||||||
|
|
||||||
|
var apiUrlFound = m.Groups.TryGetValue("APIURL", out var apiUrl);
|
||||||
|
var apiKeyFound = m.Groups.TryGetValue("APIKEY", out var apiKey);
|
||||||
|
|
||||||
|
if (apiUrlFound)
|
||||||
|
{
|
||||||
|
baseUrl = apiUrl.Value;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (apiKeyFound)
|
||||||
|
{
|
||||||
|
this.apiKey = apiKey.Value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (HttpRequestException e)
|
||||||
|
{
|
||||||
|
_logger.LogError(e, "HTTP Exception while API details, waiting {} before retrying", Static.Interval);
|
||||||
|
await Task.Delay(Static.Interval);
|
||||||
}
|
}
|
||||||
};
|
|
||||||
|
|
||||||
var content = await _client.SendAsync(request);
|
|
||||||
|
|
||||||
content.EnsureSuccessStatusCode();
|
|
||||||
var contentString = await content.Content.ReadAsStringAsync();
|
|
||||||
|
|
||||||
Match m = ApiUrlAndKey().Match(contentString);
|
|
||||||
|
|
||||||
var apiUrlFound = m.Groups.TryGetValue("APIURL", out var apiUrl);
|
|
||||||
var apiKeyFound = m.Groups.TryGetValue("APIKEY", out var apiKey);
|
|
||||||
|
|
||||||
if (apiUrlFound)
|
|
||||||
{
|
|
||||||
baseUrl = apiUrl.Value;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (apiKeyFound)
|
|
||||||
{
|
|
||||||
this.apiKey = apiKey.Value;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -67,39 +86,59 @@ public partial class SouthernWaterApi
|
|||||||
{
|
{
|
||||||
if (baseUrl.IsNullOrWhiteSpace()) await LoadApiUrl();
|
if (baseUrl.IsNullOrWhiteSpace()) await LoadApiUrl();
|
||||||
|
|
||||||
var request = new HttpRequestMessage()
|
PagedItems<Spill>? parsedPage = null;
|
||||||
|
var success = false;
|
||||||
|
|
||||||
|
while (!success)
|
||||||
{
|
{
|
||||||
RequestUri = new Uri(baseUrl + spillsEndpoint + "?page=" + page),
|
try
|
||||||
Method = HttpMethod.Get,
|
|
||||||
Headers =
|
|
||||||
{
|
{
|
||||||
{"Accept", "*/*"},
|
var request = new HttpRequestMessage()
|
||||||
{"Accept-Language", "en-GB,en;q=0.5"},
|
{
|
||||||
// {"Accept-Encoding", "gzip, deflate, br, zstd"},
|
RequestUri = new Uri(baseUrl + spillsEndpoint + "?page=" + page),
|
||||||
{"Cache-Control", "no-cache"},
|
Method = HttpMethod.Get,
|
||||||
{"Connection", "keep-alive"},
|
Headers =
|
||||||
{"DNT", "1"},
|
{
|
||||||
{"User-Agent", "Mozilla/5.0 (Windows NT 10.0; rv:126.0) Gecko/20100101 Firefox/126.0"},
|
{ "Accept", "*/*" },
|
||||||
{"Upgrade-Insecure-Requests", "1"},
|
{ "Accept-Language", "en-GB,en;q=0.5" },
|
||||||
{"Referer", "https://www.southernwater.co.uk/our-region/clean-rivers-and-seas-task-force/beachbuoy-historic-release-table/"},
|
// {"Accept-Encoding", "gzip, deflate, br, zstd"},
|
||||||
{"x-Gateway-APIKey", apiKey},
|
{ "Cache-Control", "no-cache" },
|
||||||
{"X-Requested-With", "XMLHttpRequest"},
|
{ "Connection", "keep-alive" },
|
||||||
|
{ "DNT", "1" },
|
||||||
|
{ "User-Agent", "Mozilla/5.0 (Windows NT 10.0; rv:126.0) Gecko/20100101 Firefox/126.0" },
|
||||||
|
{ "Upgrade-Insecure-Requests", "1" },
|
||||||
|
{
|
||||||
|
"Referer",
|
||||||
|
"https://www.southernwater.co.uk/our-region/clean-rivers-and-seas-task-force/beachbuoy-historic-release-table/"
|
||||||
|
},
|
||||||
|
{ "x-Gateway-APIKey", apiKey },
|
||||||
|
{ "X-Requested-With", "XMLHttpRequest" },
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
var content = await _client.SendAsync(request);
|
||||||
|
|
||||||
|
content.EnsureSuccessStatusCode();
|
||||||
|
|
||||||
|
success = true;
|
||||||
|
|
||||||
|
parsedPage = (PagedItems<Spill>?)await content.Content.ReadFromJsonAsync(typeof(PagedItems<Spill>),
|
||||||
|
jsonSerialiser ?? new JsonSerialiser());
|
||||||
|
|
||||||
|
if (parsedPage is not null)
|
||||||
|
{
|
||||||
|
parsedPage.items.ForEach(x =>
|
||||||
|
{
|
||||||
|
x.eventStart = x.eventStart.ToUniversalTime();
|
||||||
|
x.eventStop = x.eventStop.ToUniversalTime();
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
};
|
catch (HttpRequestException e)
|
||||||
|
|
||||||
var content = await _client.SendAsync(request);
|
|
||||||
|
|
||||||
content.EnsureSuccessStatusCode();
|
|
||||||
|
|
||||||
var parsedPage = (PagedItems<Spill>?) await content.Content.ReadFromJsonAsync(typeof(PagedItems<Spill>), jsonSerialiser ?? new JsonSerialiser());
|
|
||||||
|
|
||||||
if (parsedPage is not null)
|
|
||||||
{
|
|
||||||
parsedPage.items.ForEach(x =>
|
|
||||||
{
|
{
|
||||||
x.eventStart = x.eventStart.ToUniversalTime();
|
_logger.LogError(e, "HTTP Exception while loading page [{}], waiting {} before retrying", page, Static.Interval);
|
||||||
x.eventStop = x.eventStop.ToUniversalTime();
|
await Task.Delay(Static.Interval);
|
||||||
});
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return parsedPage;
|
return parsedPage;
|
||||||
|
Loading…
Reference in New Issue
Block a user