diff --git a/BotNet.CommandHandlers/SQL/SQLCommandHandler.cs b/BotNet.CommandHandlers/SQL/SQLCommandHandler.cs
index 0b79a25..b207a2f 100644
--- a/BotNet.CommandHandlers/SQL/SQLCommandHandler.cs
+++ b/BotNet.CommandHandlers/SQL/SQLCommandHandler.cs
@@ -57,7 +57,8 @@ await _telegramBotClient.SendTextMessageAsync(
chatId: command.Chat.Id,
text: $$"""
Table '{{table}}' not found. Available tables are:
- - pileg_dpr
+ - pileg_dpr_dapil
+ - pileg_dpr_provinsi
- pilpres
- vps
diff --git a/BotNet.Services/Pemilu2024/PilegDPRPerDapilDataSource.cs b/BotNet.Services/Pemilu2024/PilegDPRPerDapilDataSource.cs
new file mode 100644
index 0000000..e13965c
--- /dev/null
+++ b/BotNet.Services/Pemilu2024/PilegDPRPerDapilDataSource.cs
@@ -0,0 +1,158 @@
+using System.Collections.Generic;
+using System.Linq;
+using System.Threading;
+using System.Threading.Tasks;
+using BotNet.Services.SQL;
+using BotNet.Services.Sqlite;
+
+namespace BotNet.Services.Pemilu2024 {
+ public sealed class PilegDPRPerDapilDataSource(
+ ScopedDatabase scopedDatabase,
+ SirekapClient sirekapClient
+ ) : IScopedDataSource {
+ private const string PKB = "1";
+ private const string GERINDRA = "2";
+ private const string PDIP = "3";
+ private const string GOLKAR = "4";
+ private const string NASDEM = "5";
+ private const string PARTAI_BURUH = "6";
+ private const string GELORA = "7";
+ private const string PKS = "8";
+ private const string PKN = "9";
+ private const string HANURA = "10";
+ private const string GARUDA = "11";
+ private const string PAN = "12";
+ private const string PBB = "13";
+ private const string DEMOKRAT = "14";
+ private const string PSI = "15";
+ private const string PERINDO = "16";
+ private const string PPP = "17";
+ private const string PNA = "18";
+ private const string GABTHAT = "19";
+ private const string PDA = "20";
+ private const string PARTAI_ACEH = "21";
+ private const string PAS_ACEH = "22";
+ private const string PARTAI_SIRA = "23";
+ private const string PARTAI_UMMAT = "24";
+ private readonly ScopedDatabase _scopedDatabase = scopedDatabase;
+ private readonly SirekapClient _sirekapClient = sirekapClient;
+
+ public async Task LoadTableAsync(CancellationToken cancellationToken) {
+ _scopedDatabase.ExecuteNonQuery("""
+ CREATE TABLE pileg_dpr_dapil (
+ kode_dapil VARCHAR(5) PRIMARY KEY,
+ dapil VARCHAR(50),
+ progress REAL,
+ pkb INTEGER,
+ gerindra INTEGER,
+ pdip INTEGER,
+ golkar INTEGER,
+ nasdem INTEGER,
+ partai_buruh INTEGER,
+ gelora INTEGER,
+ pks INTEGER,
+ pkn INTEGER,
+ hanura INTEGER,
+ garuda INTEGER,
+ pan INTEGER,
+ pbb INTEGER,
+ demokrat INTEGER,
+ psi INTEGER,
+ perindo INTEGER,
+ ppp INTEGER,
+ pna INTEGER,
+ gabthat INTEGER,
+ pda INTEGER,
+ partai_aceh INTEGER,
+ pas_aceh INTEGER,
+ partai_sira INTEGER,
+ partai_ummat INTEGER,
+ total INTEGER
+ )
+ """);
+
+ IList listDapilDPR = await _sirekapClient.GetDapilDPRListAsync(cancellationToken);
+ Dictionary dapilByKode = listDapilDPR.ToDictionary(
+ keySelector: dapil => dapil.Kode
+ );
+
+ ReportPilegDPRByDapil report = await _sirekapClient.GetReportPilegDPRByDapilAsync(cancellationToken);
+
+ foreach ((string kodeDapil, ReportPilegDPRByDapil.Row? row) in report.RowByKodeDapil.OrderBy(pair => pair.Key)) {
+ if (row == null) {
+ _scopedDatabase.ExecuteNonQuery("""
+ INSERT INTO pileg_dpr_dapil (kode_dapil, dapil, progress, pkb, gerindra, pdip, golkar, nasdem, partai_buruh, gelora, pks, pkn, hanura, garuda, pan, pbb, demokrat, psi, perindo, ppp, pna, gabthat, pda, partai_aceh, pas_aceh, partai_sira, partai_ummat, total)
+ VALUES (@kode_dapil, @dapil, null, null, null, null, null, null, null, null, null, null, null, null, null, null, null, null, null, null, null, null, null, null, null, null, null, null)
+ """,
+ [
+ ( "@kode_dapil", kodeDapil ),
+ ( "@dapil", dapilByKode[kodeDapil].Nama )
+ ]
+ );
+ continue;
+ }
+
+ int? pkb = row.VotesByKodePartai!.TryGetValue(PKB, out int p) ? p : null;
+ int? gerindra = row.VotesByKodePartai!.TryGetValue(GERINDRA, out int g) ? g : null;
+ int? pdip = row.VotesByKodePartai!.TryGetValue(PDIP, out int pd) ? pd : null;
+ int? golkar = row.VotesByKodePartai!.TryGetValue(GOLKAR, out int go) ? go : null;
+ int? nasdem = row.VotesByKodePartai!.TryGetValue(NASDEM, out int n) ? n : null;
+ int? partai_buruh = row.VotesByKodePartai!.TryGetValue(PARTAI_BURUH, out int pb) ? pb : null;
+ int? gelora = row.VotesByKodePartai!.TryGetValue(GELORA, out int ge) ? ge : null;
+ int? pks = row.VotesByKodePartai!.TryGetValue(PKS, out int pk) ? pk : null;
+ int? pkn = row.VotesByKodePartai!.TryGetValue(PKN, out int pn) ? pn : null;
+ int? hanura = row.VotesByKodePartai!.TryGetValue(HANURA, out int h) ? h : null;
+ int? garuda = row.VotesByKodePartai!.TryGetValue(GARUDA, out int ga) ? ga : null;
+ int? pan = row.VotesByKodePartai!.TryGetValue(PAN, out int pa) ? pa : null;
+ int? pbb = row.VotesByKodePartai!.TryGetValue(PBB, out int pb2) ? pb2 : null;
+ int? demokrat = row.VotesByKodePartai!.TryGetValue(DEMOKRAT, out int d) ? d : null;
+ int? psi = row.VotesByKodePartai!.TryGetValue(PSI, out int ps) ? ps : null;
+ int? perindo = row.VotesByKodePartai!.TryGetValue(PERINDO, out int pe) ? pe : null;
+ int? ppp = row.VotesByKodePartai!.TryGetValue(PPP, out int pp) ? pp : null;
+ int? pna = row.VotesByKodePartai!.TryGetValue(PNA, out int pn2) ? pn2 : null;
+ int? gabthat = row.VotesByKodePartai!.TryGetValue(GABTHAT, out int gab) ? gab : null;
+ int? pda = row.VotesByKodePartai!.TryGetValue(PDA, out int pd2) ? pd2 : null;
+ int? partai_aceh = row.VotesByKodePartai!.TryGetValue(PARTAI_ACEH, out int pa2) ? pa2 : null;
+ int? pas_aceh = row.VotesByKodePartai!.TryGetValue(PAS_ACEH, out int pas) ? pas : null;
+ int? partai_sira = row.VotesByKodePartai!.TryGetValue(PARTAI_SIRA, out int s) ? s : null;
+ int? partai_ummat = row.VotesByKodePartai!.TryGetValue(PARTAI_UMMAT, out int u) ? u : null;
+ int total = (pkb ?? 0) + (gerindra ?? 0) + (pdip ?? 0) + (golkar ?? 0) + (nasdem ?? 0) + (partai_buruh ?? 0) + (gelora ?? 0) + (pks ?? 0) + (pkn ?? 0) + (hanura ?? 0) + (garuda ?? 0) + (pan ?? 0) + (pbb ?? 0) + (demokrat ?? 0) + (psi ?? 0) + (perindo ?? 0) + (ppp ?? 0) + (pna ?? 0) + (gabthat ?? 0) + (pda ?? 0) + (partai_aceh ?? 0) + (pas_aceh ?? 0) + (partai_sira ?? 0) + (partai_ummat ?? 0);
+ _scopedDatabase.ExecuteNonQuery("""
+ INSERT INTO pileg_dpr_dapil (kode_dapil, dapil, progress, pkb, gerindra, pdip, golkar, nasdem, partai_buruh, gelora, pks, pkn, hanura, garuda, pan, pbb, demokrat, psi, perindo, ppp, pna, gabthat, pda, partai_aceh, pas_aceh, partai_sira, partai_ummat, total)
+ VALUES (@kode_dapil, @dapil, @progress, @pkb, @gerindra, @pdip, @golkar, @nasdem, @partai_buruh, @gelora, @pks, @pkn, @hanura, @garuda, @pan, @pbb, @demokrat, @psi, @perindo, @ppp, @pna, @gabthat, @pda, @partai_aceh, @pas_aceh, @partai_sira, @partai_ummat, @total)
+ """,
+ [
+ ( "@kode_dapil", kodeDapil ),
+ ( "@dapil", dapilByKode[kodeDapil].Nama ),
+ ( "@progress", row.Persen ),
+ ( "@pkb", pkb),
+ ( "@gerindra", gerindra),
+ ( "@pdip", pdip),
+ ( "@golkar", golkar),
+ ( "@nasdem", nasdem),
+ ( "@partai_buruh", partai_buruh),
+ ( "@gelora", gelora),
+ ( "@pks", pks),
+ ( "@pkn", pkn),
+ ( "@hanura", hanura),
+ ( "@garuda", garuda),
+ ( "@pan", pan),
+ ( "@pbb", pbb),
+ ( "@demokrat", demokrat),
+ ( "@psi", psi),
+ ( "@perindo", perindo),
+ ( "@ppp", ppp),
+ ( "@pna", pna),
+ ( "@gabthat", gabthat),
+ ( "@pda", pda),
+ ( "@partai_aceh", partai_aceh),
+ ( "@pas_aceh", pas_aceh),
+ ( "@partai_sira", partai_sira),
+ ( "@partai_ummat", partai_ummat),
+ ( "@total", total)
+ ]
+ );
+ }
+ }
+ }
+}
diff --git a/BotNet.Services/Pemilu2024/PilegDPRDataSource.cs b/BotNet.Services/Pemilu2024/PilegDPRPerProvinsiDataSource.cs
similarity index 91%
rename from BotNet.Services/Pemilu2024/PilegDPRDataSource.cs
rename to BotNet.Services/Pemilu2024/PilegDPRPerProvinsiDataSource.cs
index be1f956..386e723 100644
--- a/BotNet.Services/Pemilu2024/PilegDPRDataSource.cs
+++ b/BotNet.Services/Pemilu2024/PilegDPRPerProvinsiDataSource.cs
@@ -6,7 +6,7 @@
using BotNet.Services.Sqlite;
namespace BotNet.Services.Pemilu2024 {
- public sealed class PilegDPRDataSource(
+ public sealed class PilegDPRPerProvinsiDataSource(
ScopedDatabase scopedDatabase,
SirekapClient sirekapClient
) : IScopedDataSource {
@@ -39,7 +39,7 @@ SirekapClient sirekapClient
public async Task LoadTableAsync(CancellationToken cancellationToken) {
_scopedDatabase.ExecuteNonQuery("""
- CREATE TABLE pileg_dpr (
+ CREATE TABLE pileg_dpr_provinsi (
provinsi VARCHAR(50) PRIMARY KEY,
progress REAL,
pkb INTEGER,
@@ -75,9 +75,9 @@ total INTEGER
keySelector: provinsi => provinsi.Kode
);
- ReportPilegDPR report = await _sirekapClient.GetReportPilegDPRAsync(cancellationToken);
+ ReportPilegDPRByWilayah report = await _sirekapClient.GetReportPilegDPRByProvinsiAsync(cancellationToken);
- foreach ((string kodeWilayah, ReportPilegDPR.Row row) in report.RowByKodeWilayah.OrderBy(pair => pair.Key)) {
+ foreach ((string kodeWilayah, ReportPilegDPRByWilayah.Row row) in report.RowByKodeWilayah.OrderBy(pair => pair.Key)) {
int? pkb = row.VotesByKodePartai!.TryGetValue(PKB, out int p) ? p : null;
int? gerindra = row.VotesByKodePartai!.TryGetValue(GERINDRA, out int g) ? g : null;
int? pdip = row.VotesByKodePartai!.TryGetValue(PDIP, out int pd) ? pd : null;
@@ -104,7 +104,7 @@ total INTEGER
int? partai_ummat = row.VotesByKodePartai!.TryGetValue(PARTAI_UMMAT, out int u) ? u: null;
int total = (pkb ?? 0) + (gerindra ?? 0) + (pdip ?? 0) + (golkar ?? 0) + (nasdem ?? 0) + (partai_buruh ?? 0) + (gelora ?? 0) + (pks ?? 0) + (pkn ?? 0) + (hanura ?? 0) + (garuda ?? 0) + (pan ?? 0) + (pbb ?? 0) + (demokrat ?? 0) + (psi ?? 0) + (perindo ?? 0) + (ppp ?? 0) + (pna ?? 0) + (gabthat ?? 0) + (pda ?? 0) + (partai_aceh ?? 0) + (pas_aceh ?? 0) + (partai_sira ?? 0) + (partai_ummat ?? 0);
_scopedDatabase.ExecuteNonQuery("""
- INSERT INTO pileg_dpr (provinsi, progress, pkb, gerindra, pdip, golkar, nasdem, partai_buruh, gelora, pks, pkn, hanura, garuda, pan, pbb, demokrat, psi, perindo, ppp, pna, gabthat, pda, partai_aceh, pas_aceh, partai_sira, partai_ummat, total)
+ INSERT INTO pileg_dpr_provinsi (provinsi, progress, pkb, gerindra, pdip, golkar, nasdem, partai_buruh, gelora, pks, pkn, hanura, garuda, pan, pbb, demokrat, psi, perindo, ppp, pna, gabthat, pda, partai_aceh, pas_aceh, partai_sira, partai_ummat, total)
VALUES (@provinsi, @progress, @pkb, @gerindra, @pdip, @golkar, @nasdem, @partai_buruh, @gelora, @pks, @pkn, @hanura, @garuda, @pan, @pbb, @demokrat, @psi, @perindo, @ppp, @pna, @gabthat, @pda, @partai_aceh, @pas_aceh, @partai_sira, @partai_ummat, @total)
""",
[
diff --git a/BotNet.Services/Pemilu2024/ServiceCollectionExtensions.cs b/BotNet.Services/Pemilu2024/ServiceCollectionExtensions.cs
index 7cbb438..1fd7af0 100644
--- a/BotNet.Services/Pemilu2024/ServiceCollectionExtensions.cs
+++ b/BotNet.Services/Pemilu2024/ServiceCollectionExtensions.cs
@@ -6,7 +6,8 @@ public static class ServiceCollectionExtensions {
public static IServiceCollection AddPemilu2024(this IServiceCollection services) {
services.AddTransient();
services.AddKeyedTransient("pilpres");
- services.AddKeyedTransient("pileg_dpr");
+ services.AddKeyedTransient("pileg_dpr_provinsi");
+ services.AddKeyedTransient("pileg_dpr_dapil");
return services;
}
}
diff --git a/BotNet.Services/Pemilu2024/SirekapClient.cs b/BotNet.Services/Pemilu2024/SirekapClient.cs
index cce7410..6e988c2 100644
--- a/BotNet.Services/Pemilu2024/SirekapClient.cs
+++ b/BotNet.Services/Pemilu2024/SirekapClient.cs
@@ -64,11 +64,18 @@ public async Task GetReportPilpresByWilayahAsync(string kodeWilay
) ?? throw new JsonException("Unexpected response");
}
- public async Task GetReportPilegDPRAsync(CancellationToken cancellationToken) {
- return await _httpClient.GetFromJsonAsync(
+ public async Task GetReportPilegDPRByProvinsiAsync(CancellationToken cancellationToken) {
+ return await _httpClient.GetFromJsonAsync(
requestUri: "https://sirekap-obj-data.kpu.go.id/pemilu/hhcw/pdpr.json",
cancellationToken: cancellationToken
) ?? throw new JsonException("Unexpected response");
}
+
+ public async Task GetReportPilegDPRByDapilAsync(CancellationToken cancellationToken) {
+ return await _httpClient.GetFromJsonAsync(
+ requestUri: "https://sirekap-obj-data.kpu.go.id/pemilu/hhcd/pdpr/0.json",
+ cancellationToken: cancellationToken
+ ) ?? throw new JsonException("Unexpected response");
+ }
}
}
diff --git a/BotNet.Services/Pemilu2024/Types.cs b/BotNet.Services/Pemilu2024/Types.cs
index c5f99e6..d8a8b7e 100644
--- a/BotNet.Services/Pemilu2024/Types.cs
+++ b/BotNet.Services/Pemilu2024/Types.cs
@@ -68,13 +68,13 @@ int Progres
);
}
- public sealed record ReportPilegDPR(
+ public sealed record ReportPilegDPRByWilayah(
[property: JsonPropertyName("ts")] string Timestamp,
string Psu,
string Mode,
IDictionary Chart,
- [property: JsonPropertyName("table")] IDictionary RowByKodeWilayah,
- ReportPilegDPR.Progress Progres
+ [property: JsonPropertyName("table")] IDictionary RowByKodeWilayah,
+ ReportPilegDPRByWilayah.Progress Progres
) {
public sealed record Row {
public string? Psu { get; set; }
@@ -91,14 +91,52 @@ public IDictionary? VotesByKodePartai {
return null;
}
- Dictionary votesByKodeCalon = [];
+ Dictionary votesByKodePartai = [];
foreach (KeyValuePair kvp in VotesByKodePartaiJson) {
if (kvp.Value.ValueKind == JsonValueKind.Number
&& kvp.Value.TryGetInt32(out int votes)) {
- votesByKodeCalon[kvp.Key] = votes;
+ votesByKodePartai[kvp.Key] = votes;
}
}
- return votesByKodeCalon;
+ return votesByKodePartai;
+ }
+ }
+ }
+
+ public sealed record Progress(
+ int Total,
+ int Progres
+ );
+ }
+
+ public sealed record ReportPilegDPRByDapil(
+ [property: JsonPropertyName("ts")] string Timestamp,
+ string Mode,
+ IDictionary Chart,
+ [property: JsonPropertyName("table")] IDictionary RowByKodeDapil,
+ ReportPilegDPRByDapil.Progress Progres
+ ) {
+ public sealed record Row {
+ public decimal Persen { get; set; }
+
+ [JsonExtensionData]
+ public IDictionary? VotesByKodePartaiJson { get; set; }
+
+ [JsonIgnore]
+ public IDictionary? VotesByKodePartai {
+ get {
+ if (VotesByKodePartaiJson is null) {
+ return null;
+ }
+
+ Dictionary votesByKodePartai = [];
+ foreach (KeyValuePair kvp in VotesByKodePartaiJson) {
+ if (kvp.Value.ValueKind == JsonValueKind.Number
+ && kvp.Value.TryGetInt32(out int votes)) {
+ votesByKodePartai[kvp.Key] = votes;
+ }
+ }
+ return votesByKodePartai;
}
}
}