Na stronie http://policja.pl/pol/form/1,Informacja-dzienna.html udostępniane są dzienne dane dotyczące liczby interwencji, zatrzymanych na gorącym uczynku, zatrzymanych poszukiwanych, pijanych kierujących, wypadków, zabitych w wypadkach, rannych w wypadkach.
Ściągam wszystkie dane:
#!/bin/bash
rm pp.html
for ((i=0; i <= 274; i++)) do
if [ ! -f ${i}.html ] ; then
curl -o ${i}.html "http://policja.pl/pol/form/1,Informacja-dzienna.html?page=${i}" ;
grep 'data-label' ${i}.html >> pp.html
sleep 6
else
grep 'data-label' ${i}.html >> pp.html
echo done
fi
done
zamieniam prostymi skryptami na plik CSV, który ma następującą strukturę:
data;interwencje;zng;zp;znk;wypadki;zabici;ranni 2008-12-01;NA;873;344;447;135;1;1
okazuje się że liczba interwencji jest podawana od roku 2018, wcześniej nie była. Nic to wstawiamy NA.
Na przyszłość dane będą aktualizowane w ten sposób, że codziennie (przez odpowiedni wpis w pliku crontab) będzie pobierany plik http://policja.pl/pol/form/1,Informacja-dzienna.html:
#!/usr/bin/perl
use LWP::Simple;
$PP="http://policja.pl/pol/form/1,Informacja-dzienna.html";
$PPBase="pp.csv";
$content = get("$PP");
$content =~ s/\r//g; # dla pewności usuń
@content = split (/\n/, $content);
foreach (@content) { chomp();
unless ($_ =~ m/data-label=/ ) { next }
if ($_ =~ m/Data statystyki/ ) { $d = clean($_); }
elsif ($_ =~ m/Interwencje/ ) { $i = clean($_); }
elsif ($_ =~ m/Zatrzymani na g/ ) { $zg = clean($_); }
elsif ($_ =~ m/Zatrzymani p/ ) { $zp = clean($_); }
elsif ($_ =~ m/Zatrzymani n/ ) { $zn = clean($_); }
elsif ($_ =~ m/Wypadki d/ ) { $w = clean($_); }
elsif ($_ =~ m/Zabici/ ) { $z = clean($_); }
elsif ($_ =~ m/Ranni/ ) { $r = clean($_);
$l = "$d;$i;$zg;$zp;$zn;$w;$z;$r";
$last_line = "$l"; $last_date = "$d";
## pierwszy wpis powinien zawierać dane dotyczące kolejnego dnia
## więc po pobraniu pierwszego można zakończyć
last;
}
}
### read the database
open (PP, "<$PPBase") || die "cannot open $PPBase/r!\n" ;
while (<PP>) { chomp(); $line = $_; @tmp = split /;/, $line; }
close(PP);
### append the database (if new record)
open (PP, ">>$PPBase") || die "cannot open $PPBase/w!\n" ;
unless ("$tmp[0]" eq "$last_date") { print PP "$last_line\n" }
else {print STDERR "nic nowego nie widzę!\n"}
close(PP);
sub clean {
my $s = shift;
$s =~ s/<[^<>]*>//g;
$s =~ s/[ \t]//g;
return ($s);
}
Zaktualizowana baza jest wysyłana na githuba. Tutaj jest: https://github.com/hrpunio/Nafisa/tree/master/PP
Agregacja do danych tygodniowych okazała się nietrywialna
Niektóra lata zaczynają się od tygodnia numer 0 a inne od 1. Okazuje się, że tak ma być (https://en.wikipedia.org/wiki/ISO_week_date#First_week):
If 1 January is on a Monday, Tuesday, Wednesday or Thursday, it is in W01. If it is on a Friday, it is part of W53 of the previous year. If it is on a Saturday, it is part of the last week of the previous year which is numbered W52 in a common year and W53 in a leap year. If it is on a Sunday, it is part of W52 of the previous year.
Nie bawię się w subtelności tylko tygodnie o numerze zero dodaję do tygodnia z poprzedniego roku.
Sprawdzam czy jest OK i się okazuje że niektóre tygodnie mają 8 dni. W plikach html są błędy:
Błędne daty 2019-10-30 winno być 2019-09-30; podobnie błędne 2019-03-28 (winno być 2019-02-28), 2018-11-01 (2018-12-01), 2018-12-01 (2017-12-01), 2016-04-30 (2016-03-30), 2009-08-31 (2009-07-31). Powtórzone daty: 2016-03-10, 2010-07-25, 2010-01-10 (zdublowane/różne/arbitralnie usuwamy drugi) Ponadto brak danych z następujących dni: 2015-12-04--2015-12-07, 2015-04-17--2015-04-20, 2014-10-02--2014-10-05, 2014-01-23 i jeszcze paru innych (nie chcialo mi się poprawiać starych.)
Teraz jest OK, plik ppw.csv ma nast strukturę:
rok;nrt;interwencje;in;zng;zngn;zp;zpn;znk;znkn;wypadki;wn;zabici;zn;ranni;rn;d1;d7 coś co się kończy na `n' to liczba tego co jest w kolumnie poprzedniej, np zn to liczba dni tygodnia dla kolumny zabici. Generalnie kolumny kończące się na `n' zawierają 7 :-) Kolumna d1 to pierwszy dzień tygodnia a kolumna d7 ostatni.
maxY <- max (d$zabici)
pz <- ggplot(d, aes(x= as.factor(nrt), y=zabici )) +
geom_bar(fill="steelblue", stat="identity") +
xlab(label="") +
ggtitle("Wypadki/zabici (Polska/2020)", subtitle="policja.pl/pol/form/1,Informacja-dzienna.html")
W sumie agregacja jest niepotrzebna, bo można ją zrobić na poziomie R używając funkcji stat_summary:
pw <- ggplot(d, aes(x= week, y=wypadki)) +
stat_summary(fun.y = sum, geom="bar", fill="steelblue") +
scale_x_date( labels = date_format("%y/%m"), breaks = "2 months") +
xlab(label="") +
#theme(plot.subtitle=element_text(size=8, hjust=0, color="black")) +
ggtitle("Wypadki (Polska/2018--2020)", subtitle="policja.pl/pol/form/1,Informacja-dzienna.html")
albo najpierw agregując dane a potem wykreślając wykres szeregu zagregowanego. Drugi sposób pozwala na przykład na dodanie linii oznaczających poziomy zagregowanego zjawiska/etykiety słupków w sposób `inteligentny'. Dodajemy etykiety (z numerem tygodnia) tylko dla słupków poniżej/powyżej Q1/Q3:
## agregowanie do danych tygodniowych kolumn ranni, zabici, wypadki
dw <- d %>% group_by ( YrWeek) %>% summarise_at ( vars(ranni,zabici,wypadki), sum )
## Obliczanie mediany i kwartyli
median.zw <- median(dw$zabici)
q1.zw <- quantile(dw$zabici, 1/4)
q3.zw <- quantile(dw$zabici, 3/4)
## YrWeekZZ to numer tygodnia, dla tygodni w których liczba wypadków jest typowa
## numer jest pusty (żeby nie był drukowany); taki trick może jest lepszy
dw$YrWeekZZ <- substr(dw$YrWeek,4,5)
dw$YrWeekZZ[ (dw$zabici > q1.zw) & (dw$zabici < q3.zw) ] <- ""
pz.2 <- ggplot(dw, aes(x= YrWeek, y=zabici)) +
geom_bar(stat="identity", fill = "steelblue") +
geom_text(data=dw, aes(label=sprintf("%s", YrWeekZZ), x=YrWeek, y= zabici), vjust=-0.9, size=3 ) +
geom_hline(yintercept=median.zw, linetype="solid", color = "violet", size=.4) +
geom_hline(yintercept=q3.zw, linetype="solid", color = "red", size=.4) +
geom_hline(yintercept=q1.zw, linetype="solid", color = "red", size=.4) +
xlab(label="rok/tydzień") +
ylab(label="zabici") +
scale_x_discrete(breaks=c("18/01", "18/10", "18/20", "18/30", "18/40",
"19/01", "19/10", "19/20", "19/30", "19/40", "20/01", "20/10")) +
# labels = c("/18/01", "18/10", "18/20", "")) ## tutaj niepotrzebne
ggtitle("Wypadki/zabici (Polska/2018--2020)",
subtitle="Linie poziomie: q1/me/q3 (źródło: policja.pl/pol/form/1,Informacja-dzienna.html)")
Brak komentarzy:
Prześlij komentarz