#!/usr/bin/env perl use strict; use warnings; use 5.010; use utf8; our $VERSION = '2.00'; binmode( STDOUT, ':encoding(utf-8)' ); use Encode qw(decode); use Getopt::Long qw(:config no_ignore_case bundling); use List::Util qw(first max none); use Travel::Status::DE::EFA; my $efa_url = 'https://efa.vrr.de/vrr/XSLT_DM_REQUEST'; my $efa_encoding; my ( $date, $time, $input_type, $list_lines, $offset, $relative_times ); my ( $full_routes, $filter_via ); my ( $timeout, $developer_mode ); my ( @grep_lines, @grep_platforms, @grep_mots ); my ( %edata, @edata_pre ); my ( $list_services, $service, $discover_and_print, $discover ); my $efa; @ARGV = map { decode( 'UTF-8', $_ ) } @ARGV; GetOptions( 'A|auto-url|discover-and-print' => \$discover_and_print, 'd|date=s' => \$date, 'D|discover' => \$discover, 'h|help' => sub { show_help(0) }, 'l|line=s@' => \@grep_lines, 'L|linelist' => \$list_lines, 'list' => \$list_services, 'm|mot=s@' => \@grep_mots, 'o|offset=i' => \$offset, 'O|output=s@' => \@edata_pre, 'p|platform=s@' => \@grep_platforms, 'r|relative' => \$relative_times, 's|service=s' => \$service, 't|time=s' => \$time, 'timeout=i' => \$timeout, 'u|efa-url=s' => \$efa_url, 'v|via=s' => \$filter_via, 'V|track-via=s' => \$filter_via, 'version' => \&show_version, 'devmode' => \$developer_mode, ) or show_help(1); if ($list_services) { show_services(); } if ( @ARGV < 1 or @ARGV > 2 ) { show_help(1); } # --line=foo,bar support @edata_pre = split( qr{,}, join( q{,}, @edata_pre ) ); @grep_lines = split( qr{,}, join( q{,}, @grep_lines ) ); @grep_mots = split( qr{,}, join( q{,}, @grep_mots ) ); @grep_platforms = split( qr{,}, join( q{,}, @grep_platforms ) ); my ( $place, $input ); if ( @ARGV == 1 ) { $input = $ARGV[0]; } else { ( $place, $input ) = @ARGV; } if ( $input =~ s{ ^ (? address|poi|stop|stopID) : }{}x ) { $input_type = $+{type}; } for my $efield (@edata_pre) { if ( $efield eq 'a' ) { $edata{route_after} = 1; $full_routes = 1 } elsif ( $efield eq 'b' ) { $edata{route_before} = 1; $full_routes = 1 } elsif ( $efield eq 'f' ) { $edata{fullroute} = 1; $full_routes = 1 } elsif ( $efield eq 'r' ) { $edata{route} = 1; $full_routes = 1 } elsif ( $efield eq 'm' ) { $edata{messages} = 1 } else { $edata{$efield} = 1 } } if ($filter_via) { $full_routes = 1; } if ($service) { my $service_ref = first { lc( $_->{shortname} ) eq lc($service) } Travel::Status::DE::EFA::get_efa_urls(); if ( not $service_ref ) { printf STDERR ( "Error: Unknown service '%s'. See 'efa-m --list' for a " . "list of supported service names\n", $service ); exit 1; } $efa_url = $service_ref->{url}; $efa_encoding = $service_ref->{encoding}; } sub new_efa_by_url { my ($url) = @_; my $res = Travel::Status::DE::EFA->new( date => $date, developer_mode => $developer_mode, efa_url => $url, efa_encoding => $efa_encoding, full_routes => $full_routes, place => $place, name => $input, time => $time, type => $input_type, timeout => $timeout, ); return $res; } sub show_help { my ($code) = @_; print "Usage: efa-m [-d ] [-t ] [place] \n" . "See also: man efa-m\n"; exit $code; } sub show_services { printf( "%-45s %-14s %s\n\n", 'service', 'abbr. (-s)', 'url (-u)' ); for my $service ( Travel::Status::DE::EFA::get_efa_urls() ) { printf( "%-45s %-14s %s\n", @{$service}{qw(name shortname url)} ); } exit 0; } sub show_version { say "efa-m version ${VERSION}"; exit 0; } sub format_delay { my ( $delay, $len ) = @_; if ( $delay and $len ) { return sprintf( "(%+${len}d)", $delay ); } return q{}; } sub format_route { my (@route) = @_; my $output = q{}; for my $stop (@route) { if ( not $stop ) { say 'BUG'; next; } if ( defined $stop->arr and defined $stop->dep ) { if ( $stop->arr->epoch == $stop->dep->epoch ) { $output .= sprintf( " %5s %40s %s\n", $stop->arr->strftime('%H:%M'), $stop->name, $stop->platform, ); } else { $output .= sprintf( "%5s → %5s %40s %s\n", $stop->arr->strftime('%H:%M'), $stop->dep->strftime('%H:%M'), $stop->name, $stop->platform, ); } } elsif ( defined $stop->arr ) { $output .= sprintf( "%5s %40s %s\n", $stop->arr->strftime('%H:%M'), $stop->name, $stop->platform, ); } elsif ( defined $stop->dep ) { $output .= sprintf( " %5s %40s %s\n", $stop->dep->strftime('%H:%M'), $stop->name, $stop->platform, ); } } return $output; } sub display_result { my (@lines) = @_; my @line_length; if ( not @lines ) { die("Nothing to show\n"); } for my $i ( 0 .. 4 ) { $line_length[$i] = max map { length( $_->[$i] ) } @lines; } for my $line (@lines) { if ( $edata{messages} and length( $line->[5] ) ) { $line->[5] =~ tr{\n\x0d}{ }s; chomp $line->[5]; print "\n"; for my $info_line ( split( qr{\n}, $line->[5] ) ) { say "# ${info_line}"; } } printf( join( q{ }, ( map { "%-${_}s" } @line_length ) ) . "\n", @{$line}[ 0 .. 4 ] ); if ( $line->[6] ) { say $line->[6]; } } return; } sub show_lines { my @output; for my $l ( $efa->lines ) { if ( ( @grep_lines and none { $l->name eq $_ } @grep_lines ) or ( @grep_mots and none { $l->mot_name eq $_ } @grep_mots ) ) { next; } if ( @grep_mots and none { $l->mot_name eq $_ } @grep_mots ) { next; } push( @output, [ $l->type, $l->name, $l->direction // q{}, q{}, $l->route // q{} ] ); } display_result(@output); return; } sub show_results { my @output; my $delay_len = 0; my $delay_fmt = 0; for my $d ( $efa->results ) { if ( $d->delay ) { $delay_len = max( $delay_len, length( $d->delay ) + 1 ); } } if ($delay_len) { $delay_fmt = $delay_len + 3; } for my $d ( $efa->results ) { my @output_line; my $platform = $d->platform; my $dtime = ( $relative_times ? sprintf( '%2d min', $d->countdown ) : $d->datetime->strftime('%H:%M') ); if ( $d->platform_db ) { $platform .= ' (DB)'; } if ( ( @grep_lines and none { $d->line eq $_ } @grep_lines ) or ( @grep_mots and none { $d->mot_name eq $_ } @grep_mots ) or ( @grep_platforms and none { $platform eq $_ } @grep_platforms ) or ( $offset and $d->countdown < $offset ) or ( $filter_via and not( first { $_->name =~ m{$filter_via}io } $d->route_post ) ) ) { next; } if ( $d->is_cancelled ) { if ($relative_times) { next; } else { $dtime = '--:--'; } } elsif ($filter_via) { my $via = first { $_->name =~ m{$filter_via}io } $d->route_post; $dtime .= ' → ' . $via->arr->clone->add( minutes => $d->delay // 0 ) ->strftime('%H:%M'); } if ( $d->delay ) { $dtime .= ' ' . format_delay( $d->delay, $delay_len ); } my $line = $d->line; if ( length($line) > 10 and $d->train_type and $d->train_no ) { $line = $d->train_type . ' ' . $d->train_no; } @output_line = ( $dtime, $platform, $line, q{}, $d->destination, $d->info ); if ( $edata{route} ) { $output_line[3] = join( q{ }, map { $_->name_suf } $d->route_interesting ); } elsif ( $d->occupancy ) { if ( $d->occupancy eq 'MANY_SEATS' ) { $output_line[3] = '_'; } elsif ( $d->occupancy eq 'FEW_SEATS' ) { $output_line[3] = '*'; } elsif ( $d->occupancy eq 'STANDING_ONLY' ) { $output_line[3] = '!'; } else { $output_line[3] = '?'; } } if ( $edata{fullroute} ) { $output_line[6] = format_route( $d->route_pre ) . ' -' x 30 . "\n" . format_route( $d->route_post ); } elsif ( $edata{route_after} ) { $output_line[6] = format_route( $d->route_post ); } elsif ( $edata{route_before} ) { $output_line[6] = format_route( reverse $d->route_pre ); } push( @output, \@output_line ); } display_result(@output); return; } if ( $discover or $discover_and_print ) { for my $service_ref ( Travel::Status::DE::EFA::get_efa_urls() ) { $efa = new_efa_by_url( $service_ref->{url} ); if ( $efa and not $efa->errstr ) { if ($discover_and_print) { last; } printf( "%s / %s (%s)\n -> efa-m -s %s %s\n\n", @{$service_ref}{qw(name shortname url shortname)}, join( q{ }, map { "'$_'" } @ARGV ), ); } } if ($discover) { exit 0; } } $efa = new_efa_by_url($efa_url); if ( my $err = $efa->errstr ) { say STDERR "Request error: ${err}"; if ( $efa->place_candidates ) { say 'You might want to try one of the following places:'; say join( "\n", $efa->place_candidates ); } elsif ( $efa->name_candidates ) { say 'You might want to try one of the following names:'; say join( "\n", $efa->name_candidates ); } exit 2; } if ($list_lines) { show_lines(); } else { show_results(); } __END__ =head1 NAME efa-m - Unofficial interface to the efa.vrr.de departure monitor =head1 SYNOPSIS B [B<-Lr>] [B<-d> I] [B<-t> I] [B<-l> I] [B<-p> I] [B<-u> I] [I] [IB<:>]I =head1 VERSION version 2.00 =head1 DESCRIPTION B lists scheduled tram, bus and train departures at the location I. For each departure, it shows =over =item * scheduled departure time, =item * delay in minutes, =item * platform, =item * line, =item * expected occupation (from _ to !, if available), and =item * destination. =back If I is specified, I refers to a location within I. Otherwise, I must be self-contained. I.e., both C<< efa Essen Hbf >> and C<< efa "Essen Hbf" >> are valid. Note, however, than C<< efa E Hbf >> works, but C<< efa "E Hbf" >> does not. By default, I refers to a stop, this can be changed by specifying I. Supported types are B
and B (point of interest). =head1 OPTIONS =over =item B<-A>, B<--auto-url>, B<--discover-and-print> Probe all known EFA entry points for the specified stop. Print the first result which was not an error. Note that this may take a while and will not necessarily return the best result. Also, using thi option by default is not recommended, as it puts EFA services under considerable additional load. =item B<-d>, B<--date> I Show departures for I instead of today. May also be specified as I =item B<-D>, B<--discover> Probe all known EFA entry points for the specified stop. Print the URLs and names of all entry points which did not return an error. =item B<-L>, B<--linelist> Do not show departures. Instead, list all lines serving the specified place. Note that this information may be incomplete -- only lines which are in service either at the time of the B call or at the time specifed using B<--date> and B<--time> are guaranteed to be included. =item B<-l>, B<--line> I Only show departures of I (comma-separatad list, option may be repeated) =item B<--list> List supported EFA services with their URLs (see B<-u>) and abbreviations (see B<-s>). =item B<-m>, B<--mot> I Only show departures whose type appears in I (comma-separated list, this option may be repeated). The following departure types ("modes of transport") are supported: zug, s-bahn, u-bahn, stadtbahn, tram, stadtbus, regionalbus, schnellbus, seilbahn, schiff, ast, sonstige =item B<-o>, B<--offset> I Ignore departures which are less than I from now. =item B<-O>, B<--output> I For each result, show additional information as specified by I. I is a comma-separated list, the B<-O>/B<--output> option may also be repeated. Each output type has both a short and long form, for instance both "-Or" and "--output=route" are valid. The following output types are supported: =over =item a / route_after Show each departure's full route (timestamps and stop names) after the requested station. =item b / route_before Show each departure's full route (timestamps and stop names) before the requested station. =item f / fullroute Show each departure's full route (timestamps and stop names) before and after the requested station. =item r / route Show up to three stops between the requested station and the departure's destination. B tries to display the three most important stops, however these are heuristically determined and may not be optimal. =item m / messages Show free-text messages associated with individual departures. These can include generic information such is bicycle transportation options or Wi-Fi availability, delay reasons, and more. =back =item B<-p>, B<--platform> I Only show departures at I (comma-separated list, option may be repeated). Note that the C<< Bstg. >> / C<< Gleis >> prefix must be omitted. =item B<-r>, B<--relative> Show relative departure times in minutes (i.e. the time difference between the departure and the time of the request). In this case, realtime data is already included. =item B<-s>, B<--service> I Short name of the EFA entry point. See Travel::Status::DE::EFA(3pm) and the B<--list> option for a list of services. =item B<-t>, B<--time> I Show departures starting at I