diff options
Diffstat (limited to 'lib')
-rw-r--r-- | lib/Travel/Status/DE/DBWagenreihung.pm | 702 | ||||
-rw-r--r-- | lib/Travel/Status/DE/DBWagenreihung/Carriage.pm | 189 | ||||
-rw-r--r-- | lib/Travel/Status/DE/DBWagenreihung/Group.pm | 379 | ||||
-rw-r--r-- | lib/Travel/Status/DE/DBWagenreihung/Section.pm | 34 | ||||
-rw-r--r-- | lib/Travel/Status/DE/DBWagenreihung/Sector.pm | 51 | ||||
-rw-r--r-- | lib/Travel/Status/DE/DBWagenreihung/Wagon.pm | 272 |
6 files changed, 703 insertions, 924 deletions
diff --git a/lib/Travel/Status/DE/DBWagenreihung.pm b/lib/Travel/Status/DE/DBWagenreihung.pm index f3e38f0..7d9be78 100644 --- a/lib/Travel/Status/DE/DBWagenreihung.pm +++ b/lib/Travel/Status/DE/DBWagenreihung.pm @@ -12,102 +12,14 @@ use JSON; use List::Util qw(uniq); use LWP::UserAgent; use Travel::Status::DE::DBWagenreihung::Group; -use Travel::Status::DE::DBWagenreihung::Section; -use Travel::Status::DE::DBWagenreihung::Wagon; +use Travel::Status::DE::DBWagenreihung::Sector; +use Travel::Status::DE::DBWagenreihung::Carriage; our $VERSION = '0.14'; Travel::Status::DE::DBWagenreihung->mk_ro_accessors( - qw(direction platform station train_no train_type)); - -# {{{ Rolling Stock Models - -my %is_redesign = ( - "02" => 1, - "03" => 1, - "06" => 1, - "09" => 1, - "10" => 1, - "13" => 1, - "14" => 1, - "15" => 1, - "16" => 1, - "18" => 1, - "19" => 1, - "20" => 1, - "23" => 1, - "24" => 1, - "27" => 1, - "28" => 1, - "29" => 1, - "31" => 1, - "32" => 1, - "33" => 1, - "34" => 1, - "35" => 1, - "36" => 1, - "37" => 1, - "53" => 1 -); - -my %model_name = ( - '011' => [ 'ICE T', 'ÖBB 4011' ], - '401' => ['ICE 1'], - '402' => ['ICE 2'], - '403.S1' => [ 'ICE 3', 'BR 403, 1. Serie' ], - '403.S2' => [ 'ICE 3', 'BR 403, 2. Serie' ], - '403.R' => [ 'ICE 3', 'BR 403 Redesign' ], - '406' => [ 'ICE 3', 'BR 406' ], - '406.R' => [ 'ICE 3', 'BR 406 Redesign' ], - '407' => [ 'ICE 3 Velaro', 'BR 407' ], - '408' => [ 'ICE 3neo', 'BR 408' ], - '411.S1' => [ 'ICE T', 'BR 411, 1. Serie' ], - '411.S2' => [ 'ICE T', 'BR 411, 2. Serie' ], - '412' => ['ICE 4'], - '415' => [ 'ICE T', 'BR 415' ], - '420' => ['BR 420'], - '422' => ['BR 422'], - '423' => ['BR 423'], - '425' => ['BR 425'], - '427' => [ 'FLIRT', 'BR 427' ], - '428' => [ 'FLIRT', 'BR 428' ], - '429' => [ 'FLIRT', 'BR 429' ], - '430' => ['BR 430'], - '440' => [ 'Coradia Continental', 'BR 440' ], - '442' => [ 'Talent 2', 'BR 442' ], - '445' => [ 'Twindexx Vario', 'BR 445' ], - '446' => [ 'Twindexx Vario', 'BR 446' ], - '462' => [ 'Desiro HC', 'BR 462' ], - '463' => [ 'Mireo', 'BR 463' ], - '475' => [ 'TGV', 'BR 475' ], - '612' => [ 'RegioSwinger', 'BR 612' ], - '620' => [ 'LINT 81', 'BR 620' ], - '622' => [ 'LINT 54', 'BR 622' ], - '631' => [ 'Link I', 'BR 631' ], - '632' => [ 'Link II', 'BR 632' ], - '633' => [ 'Link III', 'BR 633' ], - '640' => [ 'LINT 27', 'BR 640' ], - '642' => [ 'Desiro Classic', 'BR 642' ], - '643' => [ 'TALENT', 'BR 643' ], - '648' => [ 'LINT 41', 'BR 648' ], - 'IC2.TWIN' => ['IC 2 Twindexx'], - 'IC2.KISS' => ['IC 2 KISS'], -); - -my %power_desc = ( - 90 => 'mit sonstigem Antrieb', - 91 => 'mit elektrischer Lokomotive', - 92 => 'mit Diesellokomotive', - 93 => 'Hochgeschwindigkeitszug', - 94 => 'Elektrischer Triebzug', - 95 => 'Diesel-Triebzug', - 96 => 'mit speziellen Beiwagen', - 97 => 'mit elektrischer Rangierlok', - 98 => 'mit Diesel-Rangierlok', - 99 => 'Sonderfahrzeug', -); + qw(direction platform train_type)); -# }}} # {{{ Constructors sub new { @@ -122,13 +34,15 @@ sub new { } my $self = { - api_base => $opt{api_base} - // 'https://ist-wr.noncd.db.de/wagenreihung/1.0', + api_base => +'https://www.bahn.de/web/api/reisebegleitung/wagenreihung/vehicle-sequence', developer_mode => $opt{developer_mode}, cache => $opt{cache}, departure => $opt{departure}, + eva => $opt{eva}, from_json => $opt{from_json}, json => JSON->new, + train_type => $opt{train_type}, train_number => $opt{train_number}, user_agent => $opt{user_agent}, }; @@ -151,26 +65,26 @@ sub get_wagonorder { my $api_base = $self->{api_base}; my $cache = $self->{cache}; + my $eva = $self->{eva}; + my $train_type = $self->{train_type}; my $train_number = $self->{train_number}; - my $datetime = $self->{departure}; - - if ( ref($datetime) eq 'DateTime' ) { - $datetime = $datetime->strftime('%Y%m%d%H%M'); - } + my $datetime = $self->{departure}->clone->set_time_zone('UTC'); my $json = $self->{from_json}; if ( not $json ) { - my ( $content, $err ) - = $self->get_with_cache( $cache, - "${api_base}/${train_number}/${datetime}" ); + my $date = $datetime->strftime('%Y-%m-%d'); + my $time = $datetime->rfc3339 =~ s{(?=Z)}{.000}r; + my ( $content, $err ) = $self->get_with_cache( $cache, +"${api_base}?administrationId=80&category=${train_type}&date=${date}&evaNumber=${eva}&number=${train_number}&time=${time}" + ); if ($err) { $self->{errstr} = "Failed to fetch station data: $err"; return; } - $json = $self->{json}->utf8->decode($content); + $json = $self->{from_json} = $self->{json}->utf8->decode($content); } if ( exists $json->{error} ) { @@ -178,16 +92,11 @@ sub get_wagonorder { return; } - if ( @{ $json->{data}{istformation}{allFahrzeuggruppe} // [] } == 0 - and @{ $json->{data}{istformation}{halt} // [] } == 0 ) - { + if ( not $json->{departureID} ) { $self->{errstr} = 'No carriage formation available'; return; } - $self->{data} = $json->{data}; - $self->{meta} = $json->{meta}; - return $self->parse_wagonorder; } @@ -230,56 +139,15 @@ sub get_with_cache { return ( $content, undef ); } -sub wagongroup_powertype { - my ( $self, @wagons ) = @_; - - if ( not @wagons ) { - @wagons = $self->wagons; - } - - my %ml = map { $_ => 0 } ( 90 .. 99 ); - - for my $wagon (@wagons) { - - if ( not $wagon->uic_id or length( $wagon->uic_id ) != 12 ) { - next; - } - - my $wagon_type = substr( $wagon->uic_id, 0, 2 ); - if ( $wagon_type < 90 ) { - next; - } - - $ml{$wagon_type}++; - } - - my @likelihood = reverse sort { $ml{$a} <=> $ml{$b} } keys %ml; - - if ( $ml{ $likelihood[0] } == 0 ) { - return undef; - } - - return $likelihood[0]; -} - sub parse_wagonorder { my ($self) = @_; - $self->{platform} = $self->{data}{istformation}{halt}{gleisbezeichnung}; + $self->{platform} = $self->{from_json}{departurePlatform}; + $self->{platform_sched} = $self->{from_json}{departurePlatformSchedule}; - $self->{station} = { - ds100 => $self->{data}{istformation}{halt}{rl100}, - eva => $self->{data}{istformation}{halt}{evanummer}, - name => $self->{data}{istformation}{halt}{bahnhofsname}, - }; - - $self->{train_type} = $self->{data}{istformation}{zuggattung}; - $self->{train_no} = $self->{data}{istformation}{zugnummer}; - - $self->parse_wagons; - $self->{origins} = $self->merge_group_attr('origin'); - $self->{destinations} = $self->merge_group_attr('destination'); - $self->{train_nos} = $self->merge_group_attr('train_no'); + $self->parse_carriages; + $self->{destinations} = $self->merge_group_attr('destination'); + $self->{train_numbers} = $self->merge_group_attr('train_no'); } sub merge_group_attr { @@ -287,12 +155,12 @@ sub merge_group_attr { my @attrs; my %attr_to_group; - my %attr_to_sections; + my %attr_to_sectors; for my $group ( $self->groups ) { - push( @attrs, $group->{$attr} ); - push( @{ $attr_to_group{ $group->{$attr} } }, $group ); - push( @{ $attr_to_sections{ $group->{$attr} } }, $group->sections ); + push( @attrs, $group->{$attr} ); + push( @{ $attr_to_group{ $group->{$attr} } }, $group ); + push( @{ $attr_to_sectors{ $group->{$attr} } }, $group->sectors ); } @attrs = uniq @attrs; @@ -300,52 +168,64 @@ sub merge_group_attr { return [ map { { - name => $_, - groups => $attr_to_group{$_}, - sections => $attr_to_sections{$_} + name => $_, + groups => $attr_to_group{$_}, + sectors => $attr_to_sectors{$_} } } @attrs ]; } -sub parse_wagons { +sub parse_carriages { my ($self) = @_; - my @wagon_groups; - - for my $group ( @{ $self->{data}{istformation}{allFahrzeuggruppe} } ) { - my @group_wagons; - for my $wagon ( @{ $group->{allFahrzeug} } ) { - my $wagon_object - = Travel::Status::DE::DBWagenreihung::Wagon->new( %{$wagon}, - train_no => $group->{verkehrlichezugnummer} ); - push( @{ $self->{wagons} }, $wagon_object ); - push( @group_wagons, $wagon_object ); - if ( not $wagon_object->{position}{valid} ) { - $self->{has_bad_wagons} = 1; - } - } - my $group_obj = Travel::Status::DE::DBWagenreihung::Group->new( - id => $group->{fahrzeuggruppebezeichnung}, - train_no => $group->{verkehrlichezugnummer}, - origin => $group->{startbetriebsstellename}, - destination => $group->{zielbetriebsstellename}, - wagons => \@group_wagons, + my $platform_length + = $self->{from_json}{platform}{end} - $self->{from_json}{platform}{start}; + + for my $sector ( @{ $self->{from_json}{platform}{sectors} } ) { + push( + @{ $self->{sectors} }, + Travel::Status::DE::DBWagenreihung::Sector->new( + json => $sector, + platform => { + start => $self->{from_json}{platform}{start}, + end => $self->{from_json}{platform}{end}, + } + ) ); - push( @wagon_groups, $group_obj ); + } - my ( $short, $desc ) - = $self->wagongroup_description( $group_obj->wagons ); - my @sections = uniq map { $_->section } $group_obj->wagons; + my @groups; + my @numbers; - if ( @sections and length( join( q{}, @sections ) ) ) { - $group_obj->set_sections(@sections); + for my $group ( @{ $self->{from_json}{groups} // [] } ) { + my @group_carriages; + for my $carriage ( @{ $group->{vehicles} // [] } ) { + my $carriage_object + = Travel::Status::DE::DBWagenreihung::Carriage->new( + json => $carriage, + platform => { + start => $self->{from_json}{platform}{start}, + end => $self->{from_json}{platform}{end}, + } + ); + push( @group_carriages, $carriage_object ); + push( @{ $self->{carriages} }, $carriage_object ); } - $group_obj->set_description( $desc, $short ); + my $group_obj = Travel::Status::DE::DBWagenreihung::Group->new( + json => $group, + carriages => \@group_carriages, + ); + push( @groups, $group_obj ); + push( @numbers, $group_obj->train_no ); } - if ( @{ $self->{wagons} // [] } > 1 and not $self->has_bad_wagons ) { - if ( $self->{wagons}[0]->{position}{start_percent} - > $self->{wagons}[-1]->{position}{start_percent} ) + + @numbers = uniq @numbers; + $self->{train_numbers} = \@numbers; + + if ( @{ $self->{carriages} // [] } > 1 ) { + if ( $self->{carriages}[0]->{start_percent} + > $self->{carriages}[-1]->{start_percent} ) { $self->{direction} = 100; } @@ -353,27 +233,8 @@ sub parse_wagons { $self->{direction} = 0; } } - if ( not $self->has_bad_wagons ) { - @{ $self->{wagons} } = sort { - $a->{position}->{start_percent} <=> $b->{position}->{start_percent} - } @{ $self->{wagons} }; - for my $group (@wagon_groups) { - $group->sort_wagons; - } - @wagon_groups = sort { - $a->{wagons}[0]{position}{start_percent} - <=> $b->{wagons}[0]{position}{start_percent} - } @wagon_groups; - } - for my $i ( 0 .. $#wagon_groups ) { - my $group = $wagon_groups[$i]; - my $tt = $self->wagongroup_subtype( $group->wagons ); - $group->set_traintype( $i, $tt ); - $group->{type} = $tt; - } - - $self->{wagongroups} = [@wagon_groups]; + $self->{groups} = [@groups]; } # }}} @@ -391,365 +252,31 @@ sub destinations { return @{ $self->{destinations} // [] }; } -sub origins { - my ($self) = @_; - - return @{ $self->{origins} // [] }; -} - -sub train_nos { - my ($self) = @_; - - return @{ $self->{train_nos} // [] }; -} - -sub sections { - my ($self) = @_; - - if ( exists $self->{sections} ) { - return @{ $self->{sections} }; - } - - for my $section ( @{ $self->{data}{istformation}{halt}{allSektor} } ) { - my $pos = $section->{positionamgleis}; - if ( $pos->{startprozent} eq '' or $pos->{endeprozent} eq '' ) { - next; - } - push( - @{ $self->{sections} }, - Travel::Status::DE::DBWagenreihung::Section->new( - name => $section->{sektorbezeichnung}, - start_percent => $pos->{startprozent}, - end_percent => $pos->{endeprozent}, - start_meters => $pos->{startmeter}, - end_meters => $pos->{endemeter}, - ) - ); - } - - return @{ $self->{sections} // [] }; -} - sub train_numbers { my ($self) = @_; - if ( exists $self->{train_numbers} ) { - return @{ $self->{train_numbers} }; - } - - my @numbers; - - for my $group ( @{ $self->{data}{istformation}{allFahrzeuggruppe} } ) { - push( @numbers, $group->{verkehrlichezugnummer} ); - } - - @numbers = uniq @numbers; - - $self->{train_numbers} = \@numbers; - - return @numbers; + return @{ $self->{train_numbers} // [] }; } -sub has_bad_wagons { +sub sectors { my ($self) = @_; - if ( defined $self->{has_bad_wagons} ) { - return $self->{has_bad_wagons}; - } - - for my $group ( @{ $self->{data}{istformation}{allFahrzeuggruppe} } ) { - for my $wagon ( @{ $group->{allFahrzeug} } ) { - my $pos = $wagon->{positionamhalt}; - if ( $pos->{startprozent} eq '' - or $pos->{endeprozent} eq '' - or $pos->{startmeter} eq '' - or $pos->{endemeter} eq '' ) - { - return $self->{has_bad_wagons} = 1; - } - } - } - - return $self->{has_bad_wagons} = 0; -} - -sub wagongroup_description { - my ( $self, @wagons ) = @_; - - my $powertype = $self->wagongroup_powertype(@wagons); - my @model = $self->wagongroup_model(@wagons); - - my $short; - my $ret = q{}; - - if (@model) { - $short = $model[0]; - $ret .= $model[0]; - } - - if ( $powertype and $power_desc{$powertype} ) { - if ( not $ret and $power_desc{$powertype} =~ m{^mit} ) { - $ret = "Zug"; - } - $ret .= " $power_desc{$powertype}"; - $short //= $ret; - $short =~ s{elektrischer }{E-}; - $short =~ s{[Ll]\Kokomotive}{ok}; - } - - if ( @model > 1 ) { - $ret .= " ($model[1])"; - } - - return ( $short, $ret ); -} - -sub wagongroup_model { - my ( $self, @wagons ) = @_; - - my $subtype = $self->wagongroup_subtype(@wagons); - - if ( $subtype and $model_name{$subtype} ) { - return @{ $model_name{$subtype} }; - } - if ($subtype) { - return $subtype; - } - return; -} - -sub wagongroup_subtype { - my ( $self, @wagons ) = @_; - - if ( not @wagons ) { - @wagons = $self->wagons; - } - - my %ml = ( - '011' => 0, - '401' => 0, - '402' => 0, - '403.S1' => 0, - '403.S2' => 0, - '403.R' => 0, - '406' => 0, - '407' => 0, - '408' => 0, - '411.S1' => 0, - '411.S2' => 0, - '412' => 0, - '415' => 0, - '420' => 0, - '422' => 0, - '423' => 0, - '425' => 0, - '427' => 0, - '428' => 0, - '429' => 0, - '430' => 0, - '440' => 0, - '442' => 0, - '445' => 0, - '446' => 0, - '462' => 0, - '463' => 0, - '475' => 0, - '612' => 0, - '620' => 0, - '622' => 0, - '631' => 0, - '632' => 0, - '633' => 0, - '640' => 0, - '642' => 0, - '643' => 0, - '648' => 0, - 'IC2.TWIN' => 0, - 'IC2.KISS' => 0, - ); - - for my $wagon (@wagons) { - if ( not $wagon->model ) { - next; - } - if ( $wagon->model == 401 - or ( $wagon->model >= 801 and $wagon->model <= 804 ) ) - { - $ml{'401'}++; - } - elsif ( $wagon->model == 402 - or ( $wagon->model >= 805 and $wagon->model <= 808 ) ) - { - $ml{'402'}++; - } - elsif ( $wagon->model == 403 - and $is_redesign{ substr( $wagon->uic_id, 9, 2 ) } ) - { - $ml{'403.R'}++; - } - elsif ( $wagon->model == 403 and substr( $wagon->uic_id, 9, 2 ) <= 37 ) - { - $ml{'403.S1'}++; - } - elsif ( $wagon->model == 403 and substr( $wagon->uic_id, 9, 2 ) > 37 ) { - $ml{'403.S2'}++; - } - elsif ( $wagon->model == 406 ) { - $ml{'406'}++; - } - elsif ( $wagon->model == 407 ) { - $ml{'407'}++; - } - elsif ( $wagon->model == 408 ) { - $ml{'408'}++; - } - elsif ( $wagon->model == 412 or $wagon->model == 812 ) { - $ml{'412'}++; - } - elsif ( $wagon->model == 411 and substr( $wagon->uic_id, 9, 2 ) <= 32 ) - { - $ml{'411.S1'}++; - } - elsif ( $wagon->model == 411 and substr( $wagon->uic_id, 9, 2 ) > 32 ) { - $ml{'411.S2'}++; - } - elsif ( $wagon->model == 415 ) { - $ml{'415'}++; - } - elsif ( $wagon->model == 420 or $wagon->model == 421 ) { - $ml{'420'}++; - } - elsif ( $wagon->model == 422 or $wagon->model == 432 ) { - $ml{'422'}++; - } - elsif ( $wagon->model == 423 or $wagon->model == 433 ) { - $ml{'423'}++; - } - elsif ( $wagon->model == 425 or $wagon->model == 435 ) { - $ml{'425'}++; - } - elsif ( $wagon->model == 427 or $wagon->model == 827 ) { - $ml{'427'}++; - } - elsif ( $wagon->model == 428 or $wagon->model == 828 ) { - $ml{'428'}++; - } - elsif ( $wagon->model == 429 or $wagon->model == 829 ) { - $ml{'429'}++; - } - elsif ( $wagon->model == 430 or $wagon->model == 431 ) { - $ml{'430'}++; - } - elsif ($wagon->model == 440 - or $wagon->model == 441 - or $wagon->model == 841 ) - { - $ml{'440'}++; - } - elsif ($wagon->model == 442 - or $wagon->model == 443 ) - { - $ml{'442'}++; - } - elsif ($wagon->model == 462 - or $wagon->model == 862 ) - { - $ml{'462'}++; - } - elsif ($wagon->model == 463 - or $wagon->model == 863 ) - { - $ml{'463'}++; - } - elsif ( $wagon->model == 445 ) { - $ml{'445'}++; - } - elsif ( $wagon->model == 446 ) { - $ml{'446'}++; - } - elsif ( $wagon->model == 475 ) { - $ml{'475'}++; - } - elsif ( $wagon->model == 612 ) { - $ml{'612'}++; - } - elsif ( $wagon->model == 620 or $wagon->model == 621 ) { - $ml{'620'}++; - } - elsif ( $wagon->model == 622 ) { - $ml{'622'}++; - } - elsif ( $wagon->model == 631 ) { - $ml{'631'}++; - } - elsif ( $wagon->model == 632 ) { - $ml{'632'}++; - } - elsif ( $wagon->model == 633 ) { - $ml{'633'}++; - } - elsif ( $wagon->model == 640 ) { - $ml{'640'}++; - } - elsif ( $wagon->model == 642 ) { - $ml{'642'}++; - } - elsif ( $wagon->model == 643 or $wagon->model == 943 ) { - $ml{'643'}++; - } - elsif ( $wagon->model == 648 ) { - $ml{'648'}++; - } - elsif ( $self->train_type eq 'IC' and $wagon->model == 110 ) { - $ml{'IC2.KISS'}++; - } - elsif ( $self->train_type eq 'IC' and $wagon->is_dosto ) { - $ml{'IC2.TWIN'}++; - } - elsif ( substr( $wagon->uic_id, 4, 4 ) eq '4011' ) { - $ml{'011'}++; - } - } - - my @likelihood = reverse sort { $ml{$a} <=> $ml{$b} } keys %ml; - - # Less than two wagons are generally inconclusive. - # Exception: BR 631 (Link I) only has a single wagon - if ( - $ml{ $likelihood[0] } < 2 - and not($likelihood[0] eq '631' - and @wagons == 1 - and substr( $wagons[0]->uic_id, 0, 2 ) eq '95' ) - ) - { - return undef; - } - - return $likelihood[0]; + return @{ $self->{sectors} // [] }; } sub groups { my ($self) = @_; - return @{ $self->{wagongroups} // [] }; + return @{ $self->{groups} // [] }; } sub carriages { my ($self) = @_; - return @{ $self->{wagons} // [] }; -} - -sub wagons { - my ($self) = @_; - return @{ $self->{wagons} // [] }; + return @{ $self->{carriages} // [] }; } sub TO_JSON { my ($self) = @_; - # ensure that all objects are available - $self->train_numbers; - $self->sections; - my %copy = %{$self}; delete $copy{from_json}; @@ -772,12 +299,14 @@ Travel::Status::DE::DBWagenreihung - Interface to Deutsche Bahn carriage formati use Travel::Status::DE::DBWagenreihung; my $wr = Travel::Status::DE::DBWagenreihung->new( - departure => 'DateTime or YYYYMMDDhhmm', - train_number => 1234, + eva => 8000080, + departure => $datetime, + train_type => 'IC', + train_number => 2045, ); for my $carriage ( $wr->carriages ) { - printf("Wagen %s: Abschnitt %s\n", $carriage->number // '?', $carriage->section); + printf("Wagen %s: Abschnitt %s\n", $carriage->number // '?', $carriage->sector); } =head1 VERSION @@ -789,12 +318,11 @@ This is beta software. The API may change without notice. =head1 DESCRIPTION Travel:Status:DE::DBWagenreihung is an unofficial interface to the Deutsche -Bahn carriage formation (API at L<https://ist-wr.noncd.db.de/wagenreihung/1.0>. -It returns station-specific carriage formations (also kwnown as coach -sequences) for a variety of trains in the rail network associated with Deutsche -Bahn. Data includes carriage positions on the platform, train type (e.g. ICE -series), carriage-specific attributes such as first/second class, and the -internal type and number of each carriage. +Bahn carriage formation API. It returns station-specific carriage formations +(also kwnown as coach sequences) for a variety of trains in the rail network +associated with Deutsche Bahn. Data includes carriage positions on the +platform, train type (e.g. ICE series), carriage-specific attributes such as +first/second class, and the internal type and number of each carriage. Positions on the platform are given both in meters and percent (relative to platform length). @@ -817,15 +345,22 @@ Arguments: =over -=item B<departure> => I<datetime-obj> | I<YYYYMMDDhhmm> +=item B<departure> => I<datetime-obj> -Scheduled departure at the station of interest. Must be either a -L<DateTime> object or a string in I<YYYYMMDDhhmm> format. Mandatory. +Scheduled departure at the station of interest. Must be a L<DateTime> object. +Mandatory. + +=item B<eva> => I<number> + +EVA ID of the station of interest. + +=time B<train_type> => I<string> + +Train type, e.g. "S" or "ICE". =item B<train_number> => I<number> -Train number. Do not include the train type: Use "8" for "EC 8" or -"100" for "ICE 100". +Train number. =back @@ -836,15 +371,15 @@ Returns undef otherwise. =item $wr->groups -Returns a list of Travel::Status::DE::DBWagenreihung::Group(3pm) objects -which describe the groups making up the carriage formation. Typically, each -group has a distinct origin, destination, or train number. Each group contains -a set of carriages. +Returns a list of Travel::Status::DE::DBWagenreihung::Group(3pm) objects which +describe the groups making up the carriage formation. Individual groups may +have distinct destinations or train numbers. Each group contains a set of +carriages. =item $wr->carriages Describes the individual carriages the train consists of. Returns a list of -L<Travel::Status::DE::DBWagenreihung::Wagon> objects. +L<Travel::Status::DE::DBWagenreihung::carriage> objects. =item $wr->direction @@ -852,17 +387,10 @@ Gives the train's direction of travel. Returns 0 if the train will depart towards position 0 and 100 if the train will depart towards the other platform end (mnemonic: towards the 100% position). -=item $wr->origins - -Returns a list describing the unique origins of this train's carriage groups. -Each origin is a hashref that contains its B<name>, a B<groups> arrayref to the -corresponding Travel::Status::DE::DBWagenreihung::Group(3pm) objects, and -a B<sections> arrayref to section identifiers (subject to change). - =item $wr->destinations Returns a list describing the unique destinations of this train's carriage -groups. Each origin is a hashref that contains its B<name>, a B<groups> +groups. Each destination is a hashref that contains its B<name>, a B<groups> arrayref to the corresponding Travel::Status::DE::DBWagenreihung::Group(3pm) objects, and a B<sections> arrayref to section identifiers (subject to change). @@ -870,25 +398,13 @@ objects, and a B<sections> arrayref to section identifiers (subject to change). Returns the platform name. -=item $wr->sections - -Describes the sections of the platform this train will depart from. -Returns a list of L<Travel::Status::DE::DBWagenreihung::Section> objects. - -=item $wr->station +=item $wr->sectors -Returns a hashref describing the requested station. The hashref contains three -entries: B<ds100> (DS100 / Ril100 identifier), B<eva> (EVA ID, related to but -not necessarily identical with UIC station ID), and B<name> (station name). +Describes the sectors of the platform this train will depart from. +Returns a list of L<Travel::Status::DE::DBWagenreihung::Sector> objects. =item $wr->train_numbers -Returns the list of train numbers for this departure. In most cases, this is -just one element. For trains consisting of multiple wings (which typically have -different numbers), it contains one element for each wing. - -=item $wr->train_nos - Returns a list describing the unique train numbers associated with this train's carriage groups. Each train number is a hashref that contains its B<name> (i.e., number), a B<groups> arrayref to the corresponding diff --git a/lib/Travel/Status/DE/DBWagenreihung/Carriage.pm b/lib/Travel/Status/DE/DBWagenreihung/Carriage.pm new file mode 100644 index 0000000..ca9d56a --- /dev/null +++ b/lib/Travel/Status/DE/DBWagenreihung/Carriage.pm @@ -0,0 +1,189 @@ +package Travel::Status::DE::DBWagenreihung::Carriage; + +use strict; +use warnings; +use 5.020; +use utf8; + +use parent 'Class::Accessor'; +use Carp qw(cluck); + +our $VERSION = '0.14'; +Travel::Status::DE::DBWagenreihung::Carriage->mk_ro_accessors( + qw(class_type is_closed is_dosto is_locomotive is_powercar + number model section uic_id type + start_meters end_meters length_meters start_percent end_percent length_percent + ) +); + +sub new { + my ( $obj, %opt ) = @_; + my $ref = {}; + + my %json = %{ $opt{json} }; + my $platform = $opt{platform}; + + $ref->{class_type} = 0; + $ref->{has_bistro} = 0; + $ref->{is_locomotive} = 0; + $ref->{is_powercar} = 0; + $ref->{is_closed} = 0; + $ref->{number} = $json{wagonIdentificationNumber}; + $ref->{model} = $json{vehicleID}; + $ref->{uic_id} = $json{vehicleID}; + $ref->{section} = $json{platformPosition}{sector}; + $ref->{type} = $json{type}{constructionType}; + + $ref->{model} =~ s{^.....(...)....(?:-.)?$}{$1} or $ref->{model} = undef; + + my $self = bless( $ref, $obj ); + + $self->parse_type; + + if ( $json{status} and $json{status} eq 'CLOSED' ) { + $ref->{is_closed} = 1; + } + + if ( $json{type}{category} =~ m{DININGCAR} ) { + $ref->{has_bistro} = 1; + } + elsif ( $json{type}{category} eq 'LOCOMOTIVE' ) { + $ref->{is_locomotive} = 1; + } + elsif ( $json{type}{category} eq 'POWERCAR' ) { + $ref->{is_powercar} = 1; + } + + if ( $ref->{type} =~ m{AB} ) { + $ref->{class_type} = 12; + } + elsif ( $ref->{type} =~ m{A} ) { + $ref->{class_type} = 1; + } + elsif ( $ref->{type} =~ m{B|WR} ) { + $ref->{class_type} = 2; + } + + my $pos = $json{platformPosition}; + my $platform_length = $platform->{end} - $platform->{start}; + + $ref->{start_meters} = $pos->{start}; + $ref->{end_meters} = $pos->{end}; + $ref->{start_percent} + = ( $pos->{start} - $platform->{start} ) * 100 / $platform_length, + $ref->{end_percent} + = ( $pos->{end} - $platform->{start} ) * 100 / $platform_length, + $ref->{length_meters} = $pos->{start} - $pos->{end}; + $ref->{length_percent} = $ref->{end_percent} - $ref->{start_percent}; + + if ( $pos->{start} eq '' + or $pos->{end} eq '' ) + { + $ref->{position}{valid} = 0; + } + else { + $ref->{position}{valid} = 1; + } + + return $self; +} + +sub attributes { + my ($self) = @_; + + return @{ $self->{attributes} // [] }; +} + +# See also: +# https://de.wikipedia.org/wiki/UIC-Bauart-Bezeichnungssystem_f%C3%BCr_Reisezugwagen#Kennbuchstaben +# https://www.deutsche-reisezugwagen.de/lexikon/erklarung-der-gattungszeichen/ +sub parse_type { + my ($self) = @_; + + my $type = $self->{type}; + my @desc; + + if ( $type =~ m{^D} ) { + $self->{is_dosto} = 1; + push( @desc, 'Doppelstock' ); + } + + if ( $type =~ m{b} ) { + $self->{has_accessibility} = 1; + push( @desc, 'Behindertengerechte Ausstattung' ); + } + + if ( $type =~ m{d} ) { + $self->{multipurpose} = 1; + push( @desc, 'Mehrzweck' ); + } + + if ( $type =~ m{f} ) { + push( @desc, 'Steuerabteil' ); + } + + if ( $type =~ m{i} ) { + push( @desc, 'Interregio' ); + } + + if ( $type =~ m{mm} ) { + push( @desc, 'modernisiert' ); + } + + if ( $type =~ m{p} ) { + $self->{has_ac} = 1; + push( @desc, 'Großraum' ); + } + + if ( $type =~ m{s} ) { + push( @desc, 'Sonderabteil' ); + } + + if ( $type =~ m{v} ) { + $self->{has_ac} = 1; + $self->{has_compartments} = 1; + push( @desc, 'Abteil' ); + } + + if ( $type =~ m{w} ) { + $self->{has_ac} = 1; + $self->{has_compartments} = 1; + push( @desc, 'Abteil' ); + } + + $self->{attributes} = \@desc; +} + +sub is_first_class { + my ($self) = @_; + + if ( $self->{type} =~ m{^D?A} ) { + return 1; + } + return 0; +} + +sub is_second_class { + my ($self) = @_; + + if ( $self->{type} =~ m{^D?A?B} ) { + return 1; + } + return 0; +} + +sub sections { + my ($self) = @_; + + return @{ $self->{sections} }; +} + +sub TO_JSON { + my ($self) = @_; + + my %copy = %{$self}; + + return {%copy}; +} + +1; diff --git a/lib/Travel/Status/DE/DBWagenreihung/Group.pm b/lib/Travel/Status/DE/DBWagenreihung/Group.pm index dd98550..5cd2b24 100644 --- a/lib/Travel/Status/DE/DBWagenreihung/Group.pm +++ b/lib/Travel/Status/DE/DBWagenreihung/Group.pm @@ -6,61 +6,390 @@ use 5.020; use utf8; use parent 'Class::Accessor'; +use List::Util qw(uniq); our $VERSION = '0.14'; Travel::Status::DE::DBWagenreihung::Group->mk_ro_accessors( - qw(id train_no type description desc_short origin destination has_sections) + qw(train_no train_type description desc_short destination has_sectors model series) ); +# {{{ Rolling Stock Models + +my %model_name = ( + '011' => [ 'ICE T', 'ÖBB 4011' ], + '401' => ['ICE 1'], + '402' => ['ICE 2'], + '403.S1' => [ 'ICE 3', 'BR 403, 1. Serie' ], + '403.S2' => [ 'ICE 3', 'BR 403, 2. Serie' ], + '403.R' => [ 'ICE 3', 'BR 403 Redesign' ], + '406' => [ 'ICE 3', 'BR 406' ], + '406.R' => [ 'ICE 3', 'BR 406 Redesign' ], + '407' => [ 'ICE 3 Velaro', 'BR 407' ], + '408' => [ 'ICE 3neo', 'BR 408' ], + '411.S1' => [ 'ICE T', 'BR 411, 1. Serie' ], + '411.S2' => [ 'ICE T', 'BR 411, 2. Serie' ], + '412' => ['ICE 4'], + '415' => [ 'ICE T', 'BR 415' ], + '420' => ['BR 420'], + '422' => ['BR 422'], + '423' => ['BR 423'], + '425' => ['BR 425'], + '427' => [ 'FLIRT', 'BR 427' ], + '428' => [ 'FLIRT', 'BR 428' ], + '429' => [ 'FLIRT', 'BR 429' ], + '430' => ['BR 430'], + '440' => [ 'Coradia Continental', 'BR 440' ], + '442' => [ 'Talent 2', 'BR 442' ], + '445' => [ 'Twindexx Vario', 'BR 445' ], + '446' => [ 'Twindexx Vario', 'BR 446' ], + '462' => [ 'Desiro HC', 'BR 462' ], + '463' => [ 'Mireo', 'BR 463' ], + '475' => [ 'TGV', 'BR 475' ], + '612' => [ 'RegioSwinger', 'BR 612' ], + '620' => [ 'LINT 81', 'BR 620' ], + '622' => [ 'LINT 54', 'BR 622' ], + '631' => [ 'Link I', 'BR 631' ], + '632' => [ 'Link II', 'BR 632' ], + '633' => [ 'Link III', 'BR 633' ], + '640' => [ 'LINT 27', 'BR 640' ], + '642' => [ 'Desiro Classic', 'BR 642' ], + '643' => [ 'TALENT', 'BR 643' ], + '648' => [ 'LINT 41', 'BR 648' ], + 'IC2.TWIN' => ['IC 2 Twindexx'], + 'IC2.KISS' => ['IC 2 KISS'], +); + +my %power_desc = ( + 90 => 'mit sonstigem Antrieb', + 91 => 'mit elektrischer Lokomotive', + 92 => 'mit Diesellokomotive', + 93 => 'Hochgeschwindigkeitszug', + 94 => 'Elektrischer Triebzug', + 95 => 'Diesel-Triebzug', + 96 => 'mit speziellen Beiwagen', + 97 => 'mit elektrischer Rangierlok', + 98 => 'mit Diesel-Rangierlok', + 99 => 'Sonderfahrzeug', +); + +# }}} + sub new { my ( $obj, %opt ) = @_; - my $ref = \%opt; - return bless( $ref, $obj ); -} + my %json = %{ $opt{json} }; -sub set_description { - my ( $self, $desc, $short ) = @_; + my $ref = { + carriages => $opt{carriages}, + destination => $json{transport}{destination}{name}, + train_type => $json{transport}{category}, + name => $json{transport}{name}, + line => $json{transport}{numberwline}, + train_no => $json{transport}{number}, + }; - $self->{description} = $desc; - $self->{desc_short} = $short; + $ref->{sectors} = [ + uniq grep { defined } + map { $_->{platformPosition}{sector} } @{ $json{vehicles} // [] } + ]; + if ( @{ $ref->{sectors} } ) { + $ref->{has_sectors} = 1; + } + + bless( $ref, $obj ); + + $ref->parse_description; + + return $ref; } -sub set_sections { - my ( $self, @sections ) = @_; +sub parse_powertype { + my ($self) = @_; + + my %ml = map { $_ => 0 } ( 90 .. 99 ); + + for my $carriage ( $self->carriages ) { + + if ( not $carriage->uic_id or length( $carriage->uic_id ) != 12 ) { + next; + } + + my $carriage_type = substr( $carriage->uic_id, 0, 2 ); + if ( $carriage_type < 90 ) { + next; + } + + $ml{$carriage_type}++; + } + + my @likelihood = reverse sort { $ml{$a} <=> $ml{$b} } keys %ml; - $self->{sections} = [@sections]; + if ( $ml{ $likelihood[0] } == 0 ) { + return; + } - $self->{has_sections} = 1; + $self->{powertype} = $likelihood[0]; } -sub set_traintype { - my ( $self, $i, $tt ) = @_; - $self->{type} = $tt; - for my $wagon ( $self->wagons ) { - $wagon->set_traintype( $i, $tt ); +sub parse_model { + my ($self) = @_; + + my %ml = ( + '011' => 0, + '401' => 0, + '402' => 0, + '403.S1' => 0, + '403.S2' => 0, + '403.R' => 0, + '406' => 0, + '407' => 0, + '408' => 0, + '411.S1' => 0, + '411.S2' => 0, + '412' => 0, + '415' => 0, + '420' => 0, + '422' => 0, + '423' => 0, + '425' => 0, + '427' => 0, + '428' => 0, + '429' => 0, + '430' => 0, + '440' => 0, + '442' => 0, + '445' => 0, + '446' => 0, + '462' => 0, + '463' => 0, + '475' => 0, + '612' => 0, + '620' => 0, + '622' => 0, + '631' => 0, + '632' => 0, + '633' => 0, + '640' => 0, + '642' => 0, + '643' => 0, + '648' => 0, + 'IC2.TWIN' => 0, + 'IC2.KISS' => 0, + ); + + my @carriages = $self->carriages; + + for my $carriage (@carriages) { + if ( not $carriage->model ) { + next; + } + if ( $carriage->model == 401 + or ( $carriage->model >= 801 and $carriage->model <= 804 ) ) + { + $ml{'401'}++; + } + elsif ( $carriage->model == 402 + or ( $carriage->model >= 805 and $carriage->model <= 808 ) ) + { + $ml{'402'}++; + } + elsif ( $carriage->model == 403 + and substr( $carriage->uic_id, 9, 2 ) <= 37 ) + { + $ml{'403.S1'}++; + } + elsif ( $carriage->model == 403 + and substr( $carriage->uic_id, 9, 2 ) > 37 ) + { + $ml{'403.S2'}++; + } + elsif ( $carriage->model == 406 ) { + $ml{'406'}++; + } + elsif ( $carriage->model == 407 ) { + $ml{'407'}++; + } + elsif ( $carriage->model == 408 ) { + $ml{'408'}++; + } + elsif ( $carriage->model == 412 or $carriage->model == 812 ) { + $ml{'412'}++; + } + elsif ( $carriage->model == 411 + and substr( $carriage->uic_id, 9, 2 ) <= 32 ) + { + $ml{'411.S1'}++; + } + elsif ( $carriage->model == 411 + and substr( $carriage->uic_id, 9, 2 ) > 32 ) + { + $ml{'411.S2'}++; + } + elsif ( $carriage->model == 415 ) { + $ml{'415'}++; + } + elsif ( $carriage->model == 420 or $carriage->model == 421 ) { + $ml{'420'}++; + } + elsif ( $carriage->model == 422 or $carriage->model == 432 ) { + $ml{'422'}++; + } + elsif ( $carriage->model == 423 or $carriage->model == 433 ) { + $ml{'423'}++; + } + elsif ( $carriage->model == 425 or $carriage->model == 435 ) { + $ml{'425'}++; + } + elsif ( $carriage->model == 427 or $carriage->model == 827 ) { + $ml{'427'}++; + } + elsif ( $carriage->model == 428 or $carriage->model == 828 ) { + $ml{'428'}++; + } + elsif ( $carriage->model == 429 or $carriage->model == 829 ) { + $ml{'429'}++; + } + elsif ( $carriage->model == 430 or $carriage->model == 431 ) { + $ml{'430'}++; + } + elsif ($carriage->model == 440 + or $carriage->model == 441 + or $carriage->model == 841 ) + { + $ml{'440'}++; + } + elsif ($carriage->model == 442 + or $carriage->model == 443 ) + { + $ml{'442'}++; + } + elsif ($carriage->model == 462 + or $carriage->model == 862 ) + { + $ml{'462'}++; + } + elsif ($carriage->model == 463 + or $carriage->model == 863 ) + { + $ml{'463'}++; + } + elsif ( $carriage->model == 445 ) { + $ml{'445'}++; + } + elsif ( $carriage->model == 446 ) { + $ml{'446'}++; + } + elsif ( $carriage->model == 475 ) { + $ml{'475'}++; + } + elsif ( $carriage->model == 612 ) { + $ml{'612'}++; + } + elsif ( $carriage->model == 620 or $carriage->model == 621 ) { + $ml{'620'}++; + } + elsif ( $carriage->model == 622 ) { + $ml{'622'}++; + } + elsif ( $carriage->model == 631 ) { + $ml{'631'}++; + } + elsif ( $carriage->model == 632 ) { + $ml{'632'}++; + } + elsif ( $carriage->model == 633 ) { + $ml{'633'}++; + } + elsif ( $carriage->model == 640 ) { + $ml{'640'}++; + } + elsif ( $carriage->model == 642 ) { + $ml{'642'}++; + } + elsif ( $carriage->model == 643 or $carriage->model == 943 ) { + $ml{'643'}++; + } + elsif ( $carriage->model == 648 ) { + $ml{'648'}++; + } + elsif ( $self->train_type eq 'IC' and $carriage->model == 110 ) { + $ml{'IC2.KISS'}++; + } + elsif ( $self->train_type eq 'IC' and $carriage->is_dosto ) { + $ml{'IC2.TWIN'}++; + } + elsif ( substr( $carriage->uic_id, 4, 4 ) eq '4011' ) { + $ml{'011'}++; + } + } + + my @likelihood = reverse sort { $ml{$a} <=> $ml{$b} } keys %ml; + + # Less than two carriages are generally inconclusive. + # Exception: BR 631 (Link I) only has a single carriage + if ( + $ml{ $likelihood[0] } < 2 + and not($likelihood[0] eq '631' + and @carriages == 1 + and substr( $carriages[0]->uic_id, 0, 2 ) eq '95' ) + ) + { + $self->{subtype} = undef; + } + else { + $self->{subtype} = $likelihood[0]; + } + + if ( $self->{subtype} and $model_name{ $self->{subtype} } ) { + my @model = @{ $model_name{ $self->{subtype} } }; + $self->{model} = $model[0]; + $self->{series} = $model[-1]; } } -sub sort_wagons { +sub parse_description { my ($self) = @_; - @{ $self->{wagons} } - = sort { $a->{position}{start_percent} <=> $b->{position}{start_percent} } - @{ $self->{wagons} }; + $self->parse_powertype; + $self->parse_model; + + my $short; + my $ret = q{}; + + if ( $self->{model} ) { + $short = $self->{model}; + $ret .= $self->{model}; + } + + if ( $self->{powertype} and $power_desc{ $self->{powertype} } ) { + if ( not $ret and $power_desc{ $self->{powertype} } =~ m{^mit} ) { + $ret = "Zug"; + } + $ret .= ' ' . $power_desc{ $self->{powertype} }; + $short //= $ret; + $short =~ s{elektrischer }{E-}; + $short =~ s{[Ll]\Kokomotive}{ok}; + } + + if ( $self->{series} and $self->{series} ne $self->{model} ) { + $ret .= ' (' . $self->{series} . ')'; + } + + $self->{desc_short} = $short; + $self->{description} = $ret; } -sub sections { +sub sectors { my ($self) = @_; - return @{ $self->{sections} // [] }; + return @{ $self->{sectors} // [] }; } -sub wagons { +sub carriages { my ($self) = @_; - return @{ $self->{wagons} // [] }; + return @{ $self->{carriages} // [] }; } sub TO_JSON { diff --git a/lib/Travel/Status/DE/DBWagenreihung/Section.pm b/lib/Travel/Status/DE/DBWagenreihung/Section.pm deleted file mode 100644 index 5bbf1c9..0000000 --- a/lib/Travel/Status/DE/DBWagenreihung/Section.pm +++ /dev/null @@ -1,34 +0,0 @@ -package Travel::Status::DE::DBWagenreihung::Section; - -use strict; -use warnings; -use 5.020; -use utf8; - -use parent 'Class::Accessor'; - -our $VERSION = '0.14'; - -Travel::Status::DE::DBWagenreihung::Section->mk_ro_accessors( - qw(name start_percent end_percent length_percent start_meters end_meters length_meters) -); - -sub new { - my ( $obj, %opt ) = @_; - my $ref = \%opt; - - $ref->{length_meters} = $ref->{end_meters} - $ref->{start_meters}; - $ref->{length_percent} = $ref->{end_percent} - $ref->{start_percent}; - - return bless( $ref, $obj ); -} - -sub TO_JSON { - my ($self) = @_; - - my %copy = %{$self}; - - return {%copy}; -} - -1; diff --git a/lib/Travel/Status/DE/DBWagenreihung/Sector.pm b/lib/Travel/Status/DE/DBWagenreihung/Sector.pm new file mode 100644 index 0000000..a039845 --- /dev/null +++ b/lib/Travel/Status/DE/DBWagenreihung/Sector.pm @@ -0,0 +1,51 @@ +package Travel::Status::DE::DBWagenreihung::Sector; + +use strict; +use warnings; +use 5.020; +use utf8; + +use parent 'Class::Accessor'; + +our $VERSION = '0.14'; + +Travel::Status::DE::DBWagenreihung::Sector->mk_ro_accessors( + qw(name start_percent end_percent length_percent start_meters end_meters length_meters cube_meters cube_percent) +); + +sub new { + my ( $obj, %opt ) = @_; + + my %section = %{ $opt{json} }; + my %platform = %{ $opt{platform} }; + + my $platform_length = $platform{end} - $platform{start}; + + my $ref = { + name => $section{name}, + start_meters => $section{start}, + end_meters => $section{end}, + length_meters => $section{end} - $section{start}, + cube_meters => $section{cubePosition}, + start_percent => ( $section{start} - $platform{start} ) + * 100 / $platform_length, + end_percent => ( $section{end} - $platform{start} ) + * 100 / $platform_length, + cube_percent => ( $section{cubePosition} - $platform{start} ) + * 100 / $platform_length, + }; + + $ref->{length_percent} = $ref->{end_percent} - $ref->{start_percent}; + + return bless( $ref, $obj ); +} + +sub TO_JSON { + my ($self) = @_; + + my %copy = %{$self}; + + return {%copy}; +} + +1; diff --git a/lib/Travel/Status/DE/DBWagenreihung/Wagon.pm b/lib/Travel/Status/DE/DBWagenreihung/Wagon.pm deleted file mode 100644 index 549a5ff..0000000 --- a/lib/Travel/Status/DE/DBWagenreihung/Wagon.pm +++ /dev/null @@ -1,272 +0,0 @@ -package Travel::Status::DE::DBWagenreihung::Wagon; - -use strict; -use warnings; -use 5.020; -use utf8; - -use parent 'Class::Accessor'; -use Carp qw(cluck); - -our $VERSION = '0.14'; -Travel::Status::DE::DBWagenreihung::Wagon->mk_ro_accessors( - qw(attributes class_type group_index has_ac has_accessibility - has_bahn_comfort has_bike_storage has_bistro has_compartments - has_family_area has_phone_area has_quiet_area is_closed is_dosto - is_interregio is_locomotive is_powercar number model multipurpose section - train_no train_subtype type uic_id) -); - -our %type_attributes = ( - 'ICE 1' => [ - undef, ['has_quiet_area'], undef, ['has_quiet_area'], # 1 2 3 4 - ['has_family_area'], undef, ['has_bahn_comfort'], # 5 6 7 - undef, undef, undef, ['has_bahn_comfort'], # 8 9 (10) 11 - ['has_quiet_area'], undef, undef # 12 (13) 14 - ], - 'ICE 2' => [ - undef, ['has_quiet_area'], ['has_bahn_comfort'], - ['has_family_area'], # 1 2 3 4 - undef, ['has_bahn_comfort'], - [ 'has_quiet_area', 'has_phone_area' ] # 5 6 7 - ], - 'ICE 3' => [ - ['has_quiet_area'], undef, undef, undef, # 1 2 3 (4) - ['has_family_area'], undef, ['has_bahn_comfort'], # 5 6 7 - [ 'has_quiet_area', 'has_phone_area', 'has_bahn_comfort' ], undef # 8 9 - ], - 'ICE 3 Velaro' => [ - ['has_quiet_area'], undef, undef, ['has_family_area'], # 1 2 3 4 - ['has_bahn_comfort'], ['has_bahn_comfort'], undef, undef, # 5 6 (7) 8 - [ 'has_quiet_area', 'has_phone_area' ] # 9 - ], - 'ICE 4' => [ - ['has_bike_storage'], undef, ['has_quiet_area'], undef, - undef, # 1 2 3 4 5 - undef, ['has_bahn_comfort'], undef, ['has_family_area'], # 6 7 (8) 9 - undef, ['has_bahn_comfort'], undef, undef, - ['has_quiet_area'] # 10 11 12 (13) 14 - ], - 'ICE T 411' => [ - ['has_quiet_area'], ['has_quiet_area'], undef, - ['has_family_area'], # 1 2 3 4 - undef, undef, ['has_bahn_comfort'], - [ 'has_quiet_area', 'has_bahn_comfort' ] # (5) 6 7 8 - ], - 'ICE T 415' => [ - ['has_quiet_area'], ['has_quiet_area'], ['has_bahn_comfort'], - undef, # 1 2 3 (4) - undef, undef, ['has_family_area'], - [ 'has_quiet_area', 'has_bahn_comfort' ] # (5) (6) 7 8 - ], - 'IC2 Twindexx' => [ - [ 'has_family_area', 'has_bike_storage' ], ['has_bike_storage'], # 1 2 - ['has_bike_storage'], [ 'has_bike_storage', 'has_bahn_comfort' ], # 3 4 - [ 'has_bahn_comfort', 'has_quiet_area', 'has_phone_area' ] # 5 - ], -); - -sub new { - my ( $obj, %opt ) = @_; - my $ref = {}; - - $ref->{class_type} = 0; - $ref->{has_bistro} = 0; - $ref->{is_locomotive} = 0; - $ref->{is_powercar} = 0; - $ref->{is_closed} = 0; - $ref->{train_no} = $opt{train_no}; - $ref->{number} = $opt{wagenordnungsnummer}; - $ref->{model} = $opt{fahrzeugnummer}; - $ref->{uic_id} = $opt{fahrzeugnummer}; - $ref->{section} = $opt{fahrzeugsektor}; - $ref->{type} = $opt{fahrzeugtyp}; - - $ref->{model} =~ s{^.....(...)....$}{$1} or $ref->{model} = undef; - - my $self = bless( $ref, $obj ); - - $self->parse_type; - - if ( $opt{status} and $opt{status} eq 'GESCHLOSSEN' ) { - $ref->{is_closed} = 1; - } - - if ( $opt{kategorie} =~ m{SPEISEWAGEN} ) { - $ref->{has_bistro} = 1; - } - elsif ( $opt{kategorie} eq 'LOK' ) { - $ref->{is_locomotive} = 1; - } - elsif ( $opt{kategorie} eq 'TRIEBKOPF' ) { - $ref->{is_powercar} = 1; - } - - if ( $opt{fahrzeugtyp} =~ m{AB} ) { - $ref->{class_type} = 12; - } - elsif ( $opt{fahrzeugtyp} =~ m{A} ) { - $ref->{class_type} = 1; - } - elsif ( $opt{fahrzeugtyp} =~ m{B|WR} ) { - $ref->{class_type} = 2; - } - - my $pos = $opt{positionamhalt}; - - $ref->{position}{start_percent} = $pos->{startprozent}; - $ref->{position}{end_percent} = $pos->{endeprozent}; - $ref->{position}{start_meters} = $pos->{startmeter}; - $ref->{position}{end_meters} = $pos->{endemeter}; - - if ( $pos->{startprozent} eq '' - or $pos->{endeprozent} eq '' - or $pos->{startmeter} eq '' - or $pos->{endemeter} eq '' ) - { - $ref->{position}{valid} = 0; - } - else { - $ref->{position}{valid} = 1; - } - - return $self; -} - -sub attributes { - my ($self) = @_; - - return @{ $self->{attributes} // [] }; -} - -# See also: -# https://de.wikipedia.org/wiki/UIC-Bauart-Bezeichnungssystem_f%C3%BCr_Reisezugwagen#Kennbuchstaben -# https://www.deutsche-reisezugwagen.de/lexikon/erklarung-der-gattungszeichen/ -sub parse_type { - my ($self) = @_; - - my $type = $self->{type}; - my @desc; - - if ( $type =~ m{^D} ) { - $self->{is_dosto} = 1; - push( @desc, 'Doppelstock' ); - } - - if ( $type =~ m{b} ) { - $self->{has_accessibility} = 1; - push( @desc, 'Behindertengerechte Ausstattung' ); - } - - if ( $type =~ m{d} ) { - $self->{multipurpose} = 1; - push( @desc, 'Mehrzweck' ); - } - - if ( $type =~ m{f} ) { - push( @desc, 'Steuerabteil' ); - } - - if ( $type =~ m{i} ) { - $self->{is_interregio} = 1; - push( @desc, 'Interregio' ); - } - - if ( $type =~ m{mm} ) { - push( @desc, 'modernisiert' ); - } - - if ( $type =~ m{p} ) { - $self->{has_ac} = 1; - push( @desc, 'Großraum' ); - } - - if ( $type =~ m{s} ) { - push( @desc, 'Sonderabteil' ); - } - - if ( $type =~ m{v} ) { - $self->{has_ac} = 1; - $self->{has_compartments} = 1; - push( @desc, 'Abteil' ); - } - - if ( $type =~ m{w} ) { - $self->{has_ac} = 1; - $self->{has_compartments} = 1; - push( @desc, 'Abteil' ); - } - - $self->{attributes} = \@desc; -} - -sub set_traintype { - my ( $self, $group_index, $tt ) = @_; - - $self->{group_index} = $group_index; - - if ( not $tt ) { - return; - } - - $self->{train_subtype} = $tt; - - if ( not $self->{number} or not exists( $type_attributes{$tt} ) ) { - return; - } - - if ( $self->{number} !~ m{^\d+$} ) { - return; - } - - my $index = $self->{number} - 1; - - if ( $index >= 30 ) { - $index -= 30; - } - elsif ( $index >= 20 ) { - $index -= 20; - } - - if ( not $type_attributes{$tt}[$index] ) { - return; - } - - for my $attr ( @{ $type_attributes{$tt}[$index] } ) { - $self->{$attr} = 1; - } -} - -sub is_first_class { - my ($self) = @_; - - if ( $self->{type} =~ m{^D?A} ) { - return 1; - } - return 0; -} - -sub is_second_class { - my ($self) = @_; - - if ( $self->{type} =~ m{^D?A?B} ) { - return 1; - } - return 0; -} - -sub sections { - my ($self) = @_; - - return @{ $self->{sections} }; -} - -sub TO_JSON { - my ($self) = @_; - - my %copy = %{$self}; - - return {%copy}; -} - -1; |