year in review: further delay stats

This commit is contained in:
Daniel Friesel 2022-12-27 17:29:34 +01:00
parent 9058a0be9d
commit ed63cae012
No known key found for this signature in database
GPG key ID: 100D5BFB5166E005
2 changed files with 179 additions and 28 deletions

View file

@ -1044,13 +1044,17 @@ sub compute_review {
my $longest_t; my $longest_t;
my $shortest_km; my $shortest_km;
my $shortest_t; my $shortest_t;
my $message_count my $most_delayed;
; # anzahl fahrten bei denen irgendeine nachricht vermerkt war -> irgendwas war anders als geplant my $most_delay;
my %num_by_message; # für jede nachricht my $most_undelay;
my %num_by_wrtype my $num_cancelled = 0;
; # zugtyp, sofern wagenreihung verfügbar. 'none' für nicht verfügbar. my $num_fgr = 0;
my %num_by_linetype; # zugtyp nach "ICE 123" / "RE 127". my $num_punctual = 0;
my %num_by_stop; # arr/dep name my $message_count = 0;
my %num_by_message;
my %num_by_wrtype;
my %num_by_linetype;
my %num_by_stop;
if ( not $stats or not @journeys or $stats->{num_trains} == 0 ) { if ( not $stats or not @journeys or $stats->{num_trains} == 0 ) {
return; return;
@ -1063,7 +1067,13 @@ sub compute_review {
my $min_total = $stats->{min_travel_real} + $stats->{min_interchange_real}; my $min_total = $stats->{min_travel_real} + $stats->{min_interchange_real};
for my $journey (@journeys) { for my $journey (@journeys) {
if ( $journey->{cancelled} ) {
$num_cancelled += 1;
next;
}
my %seen; my %seen;
if ( $journey->{rt_duration} ) { if ( $journey->{rt_duration} ) {
if ( not $longest_t if ( not $longest_t
or $journey->{rt_duration} > $longest_t->{rt_duration} ) or $journey->{rt_duration} > $longest_t->{rt_duration} )
@ -1076,6 +1086,7 @@ sub compute_review {
$shortest_t = $journey; $shortest_t = $journey;
} }
} }
if ( $journey->{km_route} ) { if ( $journey->{km_route} ) {
if ( not $longest_km if ( not $longest_km
or $journey->{km_route} > $longest_km->{km_route} ) or $journey->{km_route} > $longest_km->{km_route} )
@ -1088,6 +1099,7 @@ sub compute_review {
$shortest_km = $journey; $shortest_km = $journey;
} }
} }
if ( $journey->{messages} and @{ $journey->{messages} } ) { if ( $journey->{messages} and @{ $journey->{messages} } ) {
$message_count += 1; $message_count += 1;
for my $message ( @{ $journey->{messages} } ) { for my $message ( @{ $journey->{messages} } ) {
@ -1097,15 +1109,65 @@ sub compute_review {
} }
} }
} }
if ( $journey->{type} ) { if ( $journey->{type} ) {
$num_by_linetype{ $journey->{type} } += 1; $num_by_linetype{ $journey->{type} } += 1;
} }
if ( $journey->{from_name} ) { if ( $journey->{from_name} ) {
$num_by_stop{ $journey->{from_name} } += 1; $num_by_stop{ $journey->{from_name} } += 1;
} }
if ( $journey->{to_name} ) { if ( $journey->{to_name} ) {
$num_by_stop{ $journey->{to_name} } += 1; $num_by_stop{ $journey->{to_name} } += 1;
} }
if ( $journey->{sched_dep_ts} and $journey->{rt_dep_ts} ) {
$journey->{delay_dep}
= ( $journey->{rt_dep_ts} - $journey->{sched_dep_ts} ) / 60;
}
if ( $journey->{sched_arr_ts} and $journey->{rt_arr_ts} ) {
$journey->{delay_arr}
= ( $journey->{rt_arr_ts} - $journey->{sched_arr_ts} ) / 60;
}
if ( $journey->{delay_arr} and $journey->{delay_arr} >= 60 ) {
$num_fgr += 1;
}
if ( not $journey->{delay_arr} and not $journey->{delay_dep} ) {
$num_punctual += 1;
}
if ( $journey->{delay_arr} and $journey->{delay_arr} > 0 ) {
if ( not $most_delayed
or $journey->{delay_arr} > $most_delayed->{delay_arr} )
{
$most_delayed = $journey;
}
}
if ( $journey->{rt_duration} and $journey->{sched_duration} ) {
my $slowdown = $journey->{rt_duration} - $journey->{sched_duration};
my $speedup = -$slowdown;
if (
not $most_delay
or $slowdown > (
$most_delay->{rt_duration} - $most_delay->{sched_duration}
)
)
{
$most_delay = $journey;
}
if (
not $most_undelay
or $speedup > (
$most_undelay->{sched_duration}
- $most_undelay->{rt_duration}
)
)
{
$most_undelay = $journey;
}
}
} }
my @linetypes = sort { $b->[1] <=> $a->[1] } my @linetypes = sort { $b->[1] <=> $a->[1] }
@ -1117,15 +1179,18 @@ sub compute_review {
map { [ $_, $num_by_message{$_} ] } keys %num_by_message; map { [ $_, $num_by_message{$_} ] } keys %num_by_message;
$review{num_stops} = scalar @stops; $review{num_stops} = scalar @stops;
$review{km_circle} = $stats->{km_route} / 40030;
$review{km_diag} = $stats->{km_route} / 12742;
$review{trains_per_day} = sprintf( '%.1f', $stats->{num_trains} / 365 ); $review{trains_per_day} = sprintf( '%.1f', $stats->{num_trains} / 365 );
$review{km_route} = sprintf( '%.0f', $stats->{km_route} ); $review{km_route} = sprintf( '%.0f', $stats->{km_route} );
$review{km_beeline} = sprintf( '%.0f', $stats->{km_beeline} ); $review{km_beeline} = sprintf( '%.0f', $stats->{km_beeline} );
$review{km_circle} = sprintf( '%.1f', $stats->{km_route} / 40030 ); $review{km_circle_h} = sprintf( '%.1f', $review{km_circle} );
$review{km_diag} = sprintf( '%.1f', $stats->{km_route} / 12742 ); $review{km_diag_h} = sprintf( '%.1f', $review{km_diag} );
$review{trains_per_day} =~ tr{.}{,}; $review{trains_per_day} =~ tr{.}{,};
$review{km_circle} =~ tr{.}{,}; $review{km_circle_h} =~ tr{.}{,};
$review{km_diag} =~ tr{.}{,}; $review{km_diag_h} =~ tr{.}{,};
$review{traveling_min_total} = $min_total; $review{traveling_min_total} = $min_total;
$review{traveling_percentage_year} $review{traveling_percentage_year}
@ -1134,7 +1199,10 @@ sub compute_review {
$review{traveling_time_year} = min_to_human($min_total); $review{traveling_time_year} = min_to_human($min_total);
if (@linetypes) { if (@linetypes) {
$review{typical_type} = $linetypes[0][0]; $review{typical_type_1} = $linetypes[0][0];
}
if ( @linetypes > 1 ) {
$review{typical_type_2} = $linetypes[1][0];
} }
if ( @stops >= 3 ) { if ( @stops >= 3 ) {
my $desc = q{}; my $desc = q{};
@ -1184,6 +1252,46 @@ sub compute_review {
$review{shortest_km_to} = $shortest_km->{to_name}; $review{shortest_km_to} = $shortest_km->{to_name};
$review{shortest_km_id} = $shortest_km->{id}; $review{shortest_km_id} = $shortest_km->{id};
$review{most_delayed_type} = $most_delayed->{type};
$review{most_delayed_delay_dep}
= min_to_human( $most_delayed->{delay_dep} );
$review{most_delayed_delay_arr}
= min_to_human( $most_delayed->{delay_arr} );
$review{most_delayed_lineno} = $most_delayed->{line} // $most_delayed->{no};
$review{most_delayed_from} = $most_delayed->{from_name};
$review{most_delayed_to} = $most_delayed->{to_name};
$review{most_delayed_id} = $most_delayed->{id};
$review{most_delay_type} = $most_delay->{type};
$review{most_delay_delay_dep} = $most_delay->{delay_dep};
$review{most_delay_delay_arr} = $most_delay->{delay_arr};
$review{most_delay_sched_time}
= min_to_human( $most_delay->{sched_duration} / 60 );
$review{most_delay_real_time}
= min_to_human( $most_delay->{rt_duration} / 60 );
$review{most_delay_delta} = min_to_human(
( $most_delay->{rt_duration} - $most_delay->{sched_duration} ) / 60 );
$review{most_delay_lineno} = $most_delay->{line} // $most_delay->{no};
$review{most_delay_from} = $most_delay->{from_name};
$review{most_delay_to} = $most_delay->{to_name};
$review{most_delay_id} = $most_delay->{id};
$review{most_undelay_type} = $most_undelay->{type};
$review{most_undelay_delay_dep} = $most_undelay->{delay_dep};
$review{most_undelay_delay_arr} = $most_undelay->{delay_arr};
$review{most_undelay_sched_time}
= min_to_human( $most_undelay->{sched_duration} / 60 );
$review{most_undelay_real_time}
= min_to_human( $most_undelay->{rt_duration} / 60 );
$review{most_undelay_delta}
= min_to_human(
( $most_undelay->{sched_duration} - $most_undelay->{rt_duration} )
/ 60 );
$review{most_undelay_lineno} = $most_undelay->{line} // $most_undelay->{no};
$review{most_undelay_from} = $most_undelay->{from_name};
$review{most_undelay_to} = $most_undelay->{to_name};
$review{most_undelay_id} = $most_undelay->{id};
$review{issue_percent} $review{issue_percent}
= sprintf( '%.0f%%', $message_count * 100 / $stats->{num_trains} ); = sprintf( '%.0f%%', $message_count * 100 / $stats->{num_trains} );
for my $i ( 0 .. 2 ) { for my $i ( 0 .. 2 ) {
@ -1194,14 +1302,14 @@ sub compute_review {
} }
} }
printf( "In %.0f%% der Fahrten war irgendetwas nicht wie vorgesehen\n", $review{cancel_count} = $num_cancelled;
$message_count * 100 / $stats->{num_trains} ); $review{fgr_percent} = $num_fgr * 100 / $stats->{num_trains};
say "Die drei häufigsten Anmerkungen waren:"; $review{fgr_percent_h} = sprintf( '%.1f%%', $review{fgr_percent} );
for my $i ( 0 .. 2 ) { $review{fgr_percent_h} =~ tr{.}{,};
if ( $reasons[$i] ) { $review{punctual_percent} = $num_punctual * 100 / $stats->{num_trains};
printf( "%d× %s\n", $reasons[$i][1], $reasons[$i][0] ); $review{punctual_percent_h}
} = sprintf( '%.1f%%', $review{punctual_percent} );
} $review{punctual_percent_h} =~ tr{.}{,};
return \%review; return \%review;
} }
@ -1348,7 +1456,7 @@ sub get_stats {
my @journeys = $self->get( my @journeys = $self->get(
uid => $uid, uid => $uid,
cancelled => $opt{cancelled} ? 1 : 0, cancelled => 0,
verbose => 1, verbose => 1,
with_polyline => 1, with_polyline => 1,
after => $interval_start, after => $interval_start,
@ -1365,7 +1473,15 @@ sub get_stats {
); );
if ( $opt{review} ) { if ( $opt{review} ) {
return ( $stats, $self->compute_review( $stats, @journeys ) ); my @cancelled_journeys = $self->get(
uid => $uid,
cancelled => 1,
verbose => 1,
after => $interval_start,
before => $interval_end
);
return ( $stats,
$self->compute_review( $stats, @journeys, @cancelled_journeys ) );
} }
return $stats; return $stats;

View file

@ -21,10 +21,10 @@
<p> <p>
Dabei hast du ca. <strong><%= $review->{km_route} %> km</strong> (Luftlinie: <%= $review->{km_beeline} %> km) auf Schienen zurückgelegt. Dabei hast du ca. <strong><%= $review->{km_route} %> km</strong> (Luftlinie: <%= $review->{km_beeline} %> km) auf Schienen zurückgelegt.
% if ($review->{km_circle} > 1) { % if ($review->{km_circle} > 1) {
Das entspricht <%= $review->{km_circle} %> Fahrten um die Erde. Das entspricht <%= $review->{km_circle_h} %> Fahrten um die Erde.
% } % }
% elsif ($review->{km_diag} > 1) { % elsif ($review->{km_diag} > 1) {
Das entspricht <%= $review->{km_diag} %> Reisen zum Mittelpunkt der Erde und zurück. Das entspricht <%= $review->{km_diag_h} %> Reisen zum Mittelpunkt der Erde und zurück.
% } % }
</p> </p>
<p> <p>
@ -34,13 +34,13 @@
<div class="carousel-item" href="#two"> <div class="carousel-item" href="#two">
<h2>Eine typische Zugfahrt</h2> <h2>Eine typische Zugfahrt</h2>
<p> <p>
% if ($review->{typical_stops_3} and $review->{typical_type}) { % if ($review->{typical_stops_3} and $review->{typical_type_1}) {
… führte dich mit … führte dich mit
% if ($review->{typical_type} eq 'S') { % if ($review->{typical_type_1} eq 'S') {
einer <strong>S-Bahn</strong> einer <strong>S-Bahn</strong>
% } % }
% else { % else {
einem <strong><%= $review->{typical_type} %></strong> einem <strong><%= $review->{typical_type_1} %></strong>
% } % }
durch das Dreieck <%= join(' / ', @{$review->{typical_stops_3}}) %>. durch das Dreieck <%= join(' / ', @{$review->{typical_stops_3}}) %>.
% } % }
@ -99,8 +99,43 @@
<p><strong><%= $review->{"issue${i}_count"} %>×</strong> „<%= $review->{"issue${i}_text"} %>“</p> <p><strong><%= $review->{"issue${i}_count"} %>×</strong> „<%= $review->{"issue${i}_text"} %>“</p>
% } % }
% } % }
<p>Lediglich <strong><%= $review->{punctual_percent_h} %></strong> der Züge waren pünktlich auf die Minute.</p>
</div> </div>
% } % }
<div class="carousel-item" href="#five">
<h2>De trein is stukkie wukkie</h2>
<p>
% if ($review->{fgr_percent} >= 0.1) {
<strong><%= $review->{fgr_percent_h} %></strong> deiner Fahrten hatten mindestens eine Stunde Verspätung
% }
% if ($review->{cancel_count}) {
% if ($review->{fgr_percent} >= 0.1) {
und <strong><%= $review->{cancel_count} %></strong> Züge kamen gar nicht erst am Ziel an.
% }
% else {
<strong><%= $review->{cancel_count} %></strong> deiner geplanten Fahrten sind ausgefallen.
% }
% }
</p>
<p>
Mit <strong><%= $review->{most_delayed_delay_arr} %></strong> hatte <a href="/journey/<%= $review->{most_delayed_id} %>"><%= $review->{most_delayed_type} %> <%= $review->{most_delayed_lineno} %></a> <%= $review->{most_delayed_from} %> → <%= $review->{most_delayed_to} %> die größte Verspätung.
</p>
<p>
Die Fahrt mit <a href="/journey/<%= $review->{most_delay_id} %>"><%= $review->{most_delay_type} %> <%= $review->{most_delay_lineno} %></a>
von <%= $review->{most_delay_from} %> nach <%= $review->{most_delay_to} %> verlief besonders gemächlich:
sie dauerte <strong><%= $review->{most_delay_delta} %></strong> länger als geplant.
</p>
<p>
In <a href="/journey/<%= $review->{most_undelay_id} %>"><%= $review->{most_undelay_type} %> <%= $review->{most_undelay_lineno} %></a>
wurde hingegen Vmax ausgereizt und die Strecke von
<%= $review->{most_undelay_from} %> nach <%= $review->{most_undelay_to} %>
<strong><%= $review->{most_undelay_delta} %></strong> schneller absolviert als vorgesehen.
</p>
</div>
<div class="carousel-item" href="#six">
<h2>Last, but not least</h2>
<p><em>Thank you for traveling with travelynx</em></p>
</div>
</div> </div>
</div> </div>
</div> </div>