Newer
Older
#!/usr/bin/env perl
use Mojolicious::Lite;
use List::MoreUtils qw();
use Travel::Status::DE::DeutscheBahn;
use Travel::Status::DE::IRIS::Stations;
no if $] >= 5.018, warnings => "experimental::smartmatch";
our $VERSION = qx{git describe --dirty} || '0.05';
my $refresh_interval = 180;
my ( $backend, $station, %opt ) = @_;
default_expires => $refresh_interval . ' sec',
lock_level => Cache::File::LOCK_LOCAL(),
# Cache::File has UTF-8 problems, so strip it (and any other potentially
# problematic chars).
my $cstation = $station;
$cstation =~ tr{[0-9a-zA-Z -]}{}cd;
my $cache_str = "${backend}_${cstation}";
my $results = $cache->thaw($cache_str);
# requests with DS100 codes should be preferred (they avoid
# encoding problems on the IRIS server). However, only use them
# if we have an exact match. Ask the backend otherwise.
my @station_matches
= Travel::Status::DE::IRIS::Stations::get_station($station);
if ( @station_matches == 1 ) {
$station = $station_matches[0][0];
}
my $status = Travel::Status::DE::IRIS->new(
station => $station,
serializable => 1,
%opt
);
$results = [ $status->results ];
$cache->freeze( $cache_str, $results );
}
else {
my $status = Travel::Status::DE::DeutscheBahn->new(
station => $station,
%opt
);
$results = [ $status->results ];
$cache->freeze( $cache_str, $results );
}
my $station = $self->stash('station');
my @platforms = split( /,/, $self->param('platforms') // q{} );
my @lines = split( /,/, $self->param('lines') // q{} );
my $template = $self->param('mode') // 'multi';
my $hide_low_delay = $self->param('hidelowdelay') // 0;
my $hide_opts = $self->param('hide_opts') // 0;
my $show_realtime = $self->param('show_realtime') // 0;
my $backend = $self->param('backend') // 'ris';
my $admode = $self->param('admode') // 'deparr';
my $apiver = $self->param('version') // 0;
my $api_version
= $backend eq 'iris'
? $Travel::Status::DE::IRIS::VERSION
: $Travel::Status::DE::DeutscheBahn::VERSION;
$self->stash( departures => [] );
$self->stash( title => 'db-fakedisplay' );
$self->stash( version => $VERSION );
if ( not( $template ~~ [qw[clean json marudor_v1 marudor multi single]] ) )
{
Birte Kristina Friesel
committed
$template = 'multi';
}
$self->render(
$template,
hide_opts => 0,
show_intro => 1
);
if ( $template eq 'marudor' and $backend eq 'iris' ) {
$opt{lookahead} = 120;
}
Birte Kristina Friesel
committed
my @departures;
my @results = get_results_for( $backend, $station, %opt );
if ( not @results and $template ~~ [qw[json marudor_v1 marudor]] ) {
if ( $backend eq 'iris' ) {
my @candidates = map { { code => $_->[0], name => $_->[1] } }
Travel::Status::DE::IRIS::Stations::get_station($station);
api_version => $api_version,
version => $VERSION,
error => 'ambiguous station code/name',
candidates => \@candidates,
}
);
}
else {
$json = $self->render_to_string(
json => {
api_version => $api_version,
version => $VERSION,
error => 'unknown station code/name',
}
);
}
if ($callback) {
$self->render(
data => "$callback($json);",
format => 'json'
);
}
else {
$self->render(
data => $json,
format => 'json'
);
}
return;
}
if ( $backend eq 'iris' ) {
my @candidates = map { [ "$_->[1] ($_->[0])", $_->[0] ] }
Travel::Status::DE::IRIS::Stations::get_station($station);
if (@candidates) {
$self->render(
'multi',
stationlist => \@candidates,
hide_opts => 0
);
}
}
$self->render(
'multi',
error => "Got no results for '$station'",
hide_opts => 0
);
if ( $template eq 'single' ) {
if ( not @platforms ) {
for my $result (@results) {
if ( not( $result->platform ~~ \@platforms ) ) {
push( @platforms, $result->platform );
}
}
@platforms = sort { $a <=> $b } @platforms;
}
my %pcnt;
@results = grep { $pcnt{ $_->platform }++ < 1 } @results;
@results = sort { $a->platform <=> $b->platform } @results;
}
if ( $backend eq 'iris' and $show_realtime ) {
if ( $admode eq 'arr' ) {
@results = sort {
( $a->arrival // $a->departure )
<=> ( $b->arrival // $b->depearture )
} @results;
}
else {
@results = sort {
( $a->departure // $a->arrival )
<=> ( $b->departure // $b->arrival )
} @results;
}
}
my $platform = ( split( / /, $result->platform ) )[0];
my $line = $result->line;
if ( $result->isa('Travel::Status::DE::IRIS::Result') ) {
@route = $result->route_post;
}
if ( not( List::MoreUtils::any { $_ =~ m{$via}io } @route ) ) {
if ( @platforms
and not( List::MoreUtils::any { $_ eq $platform } @platforms ) )
{
if ( @lines and not( List::MoreUtils::any { $line =~ m{^$_} } @lines ) )
{
if ( $backend eq 'iris' and $admode eq 'arr' and not $result->arrival )
{
next;
}
if ( $backend eq 'iris'
and $admode eq 'dep'
and not $result->departure )
{
next;
}
Birte Kristina Friesel
committed
my ( $info, $moreinfo );
my $delaymsg
= join( ', ', map { $_->[1] } $result->delay_messages );
my $qosmsg = join( ' +++ ', map { $_->[1] } $result->qos_messages );
$info = "Fahrt fällt aus: ${delaymsg}";
elsif ( $result->delay and $result->delay > 0 ) {
if ( $template eq 'clean' ) {
$info = $delaymsg;
$delay = $result->delay;
}
else {
$info = sprintf( 'Verspätung ca. %d Min.%s%s',
$result->delay, $delaymsg ? q{: } : q{}, $delaymsg );
if ( $result->replacement_for and $template ne 'clean' ) {
for my $rep ( $result->replacement_for ) {
$info = sprintf(
'Ersatzzug für %s %s %s%s',
$rep->type, $rep->train_no,
$info ? '+++ ' : q{}, $info // q{}
Birte Kristina Friesel
committed
);
}
}
if ( $info and $qosmsg ) {
$info .= ' +++ ';
}
$info .= $qosmsg;
Birte Kristina Friesel
committed
if ( $result->additional_stops and not $result->is_cancelled ) {
my $additional_line = join( q{, }, $result->additional_stops );
$info
= 'Zusätzliche Halte: '
. $additional_line
. ( $info ? ' +++ ' : q{} )
. $info;
if ( $template ne 'marudor_v1' and $template ne 'marudor' ) {
push(
@{$moreinfo},
[ 'Zusätzliche Halte', $additional_line ]
);
}
if ( $result->canceled_stops and not $result->is_cancelled ) {
my $cancel_line = join( q{, }, $result->canceled_stops );
$info
= 'Ohne Halt in: '
. $cancel_line
. ( $info ? ' +++ ' : q{} )
. $info;
if ( $template ne 'marudor_v1' and $template ne 'marudor' ) {
push( @{$moreinfo}, [ 'Ohne Halt in', $cancel_line ] );
}
}
push( @{$moreinfo}, $result->messages );
Birte Kristina Friesel
committed
if ($info) {
$moreinfo = [ [ 'RIS', $info ] ];
}
my $time = $result->time;
if ( $backend eq 'iris' ) {
# ->time defaults to dep, so we only need to overwrite $time
# if we want arrival times
if ( $admode eq 'arr' ) {
$time = $result->sched_arrival->strftime('%H:%M');
}
if ($show_realtime) {
if ( ( $admode eq 'arr' and $result->arrival )
or not $result->departure )
{
$time = $result->arrival->strftime('%H:%M');
}
else {
$time = $result->departure->strftime('%H:%M');
}
}
if ( $info eq '+0' ) {
if ( $template eq 'clean'
and $info
and $info =~ s{ (?: ca \. \s* )? \+ (\d+) :? \s* }{}x )
{
$delay = $1;
}
if ( $hide_low_delay and $info ) {
Birte Kristina Friesel
committed
$info =~ s{ (?: ca\. \s* )? \+ [ 1 2 3 4 ] $ }{}x;
Birte Kristina Friesel
committed
$info =~ s{ (?: ca\. \s* )? \+ (\d+) }{Verspätung ca $1 Min.}x;
}
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
if ( $template eq 'marudor' ) {
my ( $route_idx, $sched_idx ) = ( 0, 0 );
my @json_route;
my @route = $result->route;
my @sched_route = $result->sched_route;
while ( $route_idx <= $#route and $sched_idx <= $#sched_route ) {
if ( $route[$route_idx] eq $sched_route[$sched_idx] ) {
push( @json_route, { name => $route[$route_idx] } );
$route_idx++;
$sched_idx++;
}
# this branch is inefficient, but won't be taken frequently
elsif ( not( $route[$route_idx] ~~ \@sched_route ) ) {
push(
@json_route,
{
name => $route[$route_idx],
isAdditional => 1
}
);
$route_idx++;
}
else {
push(
@json_route,
{
name => $sched_route[$sched_idx],
isCancelled => 1
}
);
$sched_idx++;
}
}
while ( $route_idx++ < $#route ) {
push(
@json_route,
{
name => $route[ $route_idx++ ],
isAdditional => 1,
isCancelled => 0
}
);
}
while ( $sched_idx++ < $#sched_route ) {
push(
@json_route,
{
name => $route[ $route_idx++ ],
isAdditional => 0,
isCancelled => 1
}
);
}
push(
@departures,
{
delay => $delay,
destination => $result->destination,
isCancelled => $result->can('is_cancelled')
? $result->is_cancelled
: undef,
messages => {
delay => [
map { { timestamp => $_->[0], text => $_->[1] } }
$result->delay_messages
],
qos => [
map { { timestamp => $_->[0], text => $_->[1] } }
$result->qos_messages
],
},
platform => $result->platform,
route => \@json_route,
scheduledPlatform => $result->sched_platform,
time => $time,
train => $result->train,
via => [ $result->route_interesting(3) ],
}
);
}
elsif ( $backend eq 'iris' ) {
time => $time,
sched_arrival => $result->sched_arrival
? $result->sched_arrival->strftime('%H:%M')
: undef,
sched_departure => $result->sched_departure
? $result->sched_departure->strftime('%H:%M')
: undef,
arrival => $result->arrival
? $result->arrival->strftime('%H:%M')
: undef,
departure => $result->departure
? $result->departure->strftime('%H:%M')
: undef,
train_type => $result->type,
train_line => $result->line_no,
train_no => $result->train_no,
scheduled_route => [ $result->sched_route ],
route_post => [ $result->route_post ],
platform => $platform,
info => $info,
is_cancelled => $result->can('is_cancelled')
? $result->is_cancelled
: undef,
messages => {
delay => [
map { { timestamp => $_->[0], text => $_->[1] } }
],
qos => [
map { { timestamp => $_->[0], text => $_->[1] } }
],
},
moreinfo => $moreinfo,
delay => $delay,
additional_stops => [ $result->additional_stops ],
canceled_stops => [ $result->canceled_stops ],
replaced_by => $result->can('replaced_by')
? [
map { $_->type . q{ } . $_->train_no }
$result->replaced_by
]
: [],
replacement_for => $result->can('replacement_for')
? [
map { $_->type . q{ } . $_->train_no }
$result->replacement_for
]
: [],
}
);
}
else {
push(
@departures,
{
time => $time,
train => $result->train,
via => [ $result->route_interesting(3) ],
destination => $result->destination,
platform => $platform,
info => $info,
is_cancelled => $result->can('is_cancelled')
? $result->is_cancelled
: undef,
messages => {
},
moreinfo => $moreinfo,
delay => $delay,
replaced_by => [],
replacement_for => [],
preformatted => \@departures,
version => $VERSION,
raw => \@results,
}
);
if ($callback) {
$self->render(
data => "$callback($json);",
format => 'json'
);
}
else {
$self->render(
data => $json,
format => 'json'
);
}
elsif ( $template eq 'marudor' ) {
my $json = $self->render_to_string(
json => {
departures => \@departures,
}
);
if ($callback) {
$self->render(
data => "$callback($json);",
format => 'json'
);
}
else {
$self->render(
data => $json,
format => 'json'
);
}
}
my $json = $self->render_to_string(
json => {
api_version => $api_version,
preformatted => \@departures,
version => $VERSION,
}
);
if ($callback) {
$self->render(
data => "$callback($json);",
format => 'json'
);
}
else {
$self->render(
data => $json,
format => 'json'
);
}
else {
$self->render(
$template,
departures => \@departures,
version => $VERSION,
title => "departures for ${station}",
refresh_interval => $refresh_interval + 3,
hide_opts => $hide_opts,
show_realtime => $show_realtime,
get '/_redirect' => sub {
my $station = $self->param('station');
my $params = $self->req->params;
$params->remove('station');
$params->remove('via');
if ( $params->param('mode') and $params->param('mode') eq 'multi' ) {
$params->remove($param);
}
}
$params = $params->to_string;
$self->redirect_to("/${station}/${via}?${params}");
$self->redirect_to("/${station}?${params}");
app->defaults( layout => 'default' );
get '/' => \&handle_request;
get '/:station' => \&handle_request;
get '/multi/:station' => \&handle_request;
accepts => 10,
listen => ['http://*:8092'],
workers => $ENV{VRRFAKEDISPLAY_WORKERS} // 2,
app->types->type( json => 'application/json; charset=utf-8' );