allow checking into traewelling
This commit is contained in:
parent
590d3de518
commit
07e0f89706
4 changed files with 103 additions and 65 deletions
|
@ -259,6 +259,26 @@ sub run {
|
||||||
)->wait;
|
)->wait;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for my $candidate ( $self->app->traewelling->get_pushable_accounts ) {
|
||||||
|
$self->app->log->debug(
|
||||||
|
"Pushing to Traewelling for UID $candidate->{uid}");
|
||||||
|
my $trip_id = $candidate->{journey_data}{trip_id};
|
||||||
|
if ( not $trip_id ) {
|
||||||
|
$self->app->log->debug("... trip_id is missing");
|
||||||
|
|
||||||
|
# TODO log traewelling error
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if ( $candidate->{data}{latest_push_ts}
|
||||||
|
and $candidate->{data}{latest_push_ts} == $candidate->{checkin_ts} )
|
||||||
|
{
|
||||||
|
$self->app->log->debug("... already handled");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
$self->app->traewelling_api->checkin( %{$candidate},
|
||||||
|
trip_id => $trip_id );
|
||||||
|
}
|
||||||
|
|
||||||
# Computing yearly stats may take a while, but we've got all time in the
|
# Computing yearly stats may take a while, but we've got all time in the
|
||||||
# world here. This means users won't have to wait when loading their
|
# world here. This means users won't have to wait when loading their
|
||||||
# own by-year journey log.
|
# own by-year journey log.
|
||||||
|
|
|
@ -267,41 +267,31 @@ sub logout_p {
|
||||||
}
|
}
|
||||||
|
|
||||||
sub checkin {
|
sub checkin {
|
||||||
my ( $self, $uid ) = @_;
|
my ( $self, %opt ) = @_;
|
||||||
if ( my $token = $self->get_traewelling_push_token($uid) ) {
|
|
||||||
my $user = $self->get_user_status;
|
|
||||||
|
|
||||||
# TODO delete previous traewelling status if the train's destination has been changed
|
|
||||||
# TODO delete traewelling status when undoing a travelynx checkin
|
|
||||||
if ( $user->{checked_in} and $user->{extra_data}{trip_id} ) {
|
|
||||||
my $traewelling = $self->{model}->get($uid);
|
|
||||||
if ( $traewelling->{data}{trip_id} eq $user->{extra_data}{trip_id} )
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
my $header = {
|
my $header = {
|
||||||
'User-Agent' => 'travelynx/' . $self->{version},
|
'User-Agent' => $self->{header}{'User-Agent'},
|
||||||
'Authorization' => "Bearer $token",
|
'Authorization' => "Bearer $opt{token}",
|
||||||
};
|
};
|
||||||
|
|
||||||
my $request = {
|
my $request = {
|
||||||
tripID => $user->{extra_data}{trip_id},
|
tripID => $opt{trip_id},
|
||||||
start => q{} . $user->{dep_eva},
|
start => q{} . $opt{dep_eva},
|
||||||
destination => q{} . $user->{arr_eva},
|
destination => q{} . $opt{arr_eva},
|
||||||
};
|
};
|
||||||
my $trip_req = sprintf(
|
my $trip_req = sprintf(
|
||||||
"tripID=%s&lineName=%s%%20%s&start=%s",
|
"tripID=%s&lineName=%s%%20%s&start=%s",
|
||||||
$user->{extra_data}{trip_id}, $user->{train_type},
|
$opt{trip_id}, $opt{train_type}, $opt{train_line} // $opt{train_no},
|
||||||
$user->{train_line} // $user->{train_no}, $user->{dep_eva}
|
$opt{dep_eva}
|
||||||
);
|
);
|
||||||
|
|
||||||
$self->{user_agent}->request_timeout(20)
|
$self->{user_agent}->request_timeout(20)
|
||||||
->get_p(
|
->get_p(
|
||||||
"https://traewelling.de/api/v0/trains/trip?$trip_req" =>
|
"https://traewelling.de/api/v0/trains/trip?$trip_req" => $header )
|
||||||
$header )->then(
|
->then(
|
||||||
sub {
|
sub {
|
||||||
return $self->{user_agent}->request_timeout(20)
|
return $self->{user_agent}->request_timeout(20)
|
||||||
->post_p(
|
->post_p( "https://traewelling.de/api/v0/trains/checkin" =>
|
||||||
"https://traewelling.de/api/v0/trains/checkin" =>
|
|
||||||
$header => json => $request );
|
$header => json => $request );
|
||||||
}
|
}
|
||||||
)->then(
|
)->then(
|
||||||
|
@ -309,24 +299,40 @@ sub checkin {
|
||||||
my ($tx) = @_;
|
my ($tx) = @_;
|
||||||
if ( my $err = $tx->error ) {
|
if ( my $err = $tx->error ) {
|
||||||
my $err_msg = "HTTP $err->{code} $err->{message}";
|
my $err_msg = "HTTP $err->{code} $err->{message}";
|
||||||
$self->mark_trwl_checkin_error( $uid, $user, $err_msg );
|
$self->{log}->debug("... error: $err_msg");
|
||||||
|
$self->{model}->log(
|
||||||
|
uid => $opt{uid},
|
||||||
|
message =>
|
||||||
|
"Fehler bei $opt{train_type} $opt{train_no}: $err_msg",
|
||||||
|
is_error => 1
|
||||||
|
);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
else {
|
$self->{log}->debug("... success!");
|
||||||
# TODO check for traewelling error ("error" key in response)
|
$self->{model}->log(
|
||||||
# TODO store ID of resulting status (request /user/{name} and store status ID)
|
uid => $opt{uid},
|
||||||
$self->mark_trwl_checkin_success( $uid, $user );
|
message => "Eingecheckt in $opt{train_type} $opt{train_no}",
|
||||||
|
status_id => $tx->res->json->{statusId}
|
||||||
|
);
|
||||||
|
$self->{model}->set_latest_push_ts(
|
||||||
|
uid => $opt{uid},
|
||||||
|
ts => $opt{checkin_ts}
|
||||||
|
);
|
||||||
|
|
||||||
# mark success: checked into (trip_id, start, destination)
|
# TODO store status_id in in_transit object so that it can be shown
|
||||||
}
|
# on the user status page
|
||||||
}
|
}
|
||||||
)->catch(
|
)->catch(
|
||||||
sub {
|
sub {
|
||||||
my ($err) = @_;
|
my ($err) = @_;
|
||||||
$self->mark_trwl_checkin_error( $uid, $user, $err );
|
$self->{log}->debug("... error: $err");
|
||||||
|
$self->{model}->log(
|
||||||
|
uid => $opt{uid},
|
||||||
|
message => "Fehler bei $opt{train_type} $opt{train_no}: $err",
|
||||||
|
is_error => 1
|
||||||
|
);
|
||||||
}
|
}
|
||||||
)->wait;
|
)->wait;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
1;
|
1;
|
||||||
|
|
|
@ -146,16 +146,16 @@ sub set_latest_pull_status_id {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
sub set_latest_push_status_id {
|
sub set_latest_push_ts {
|
||||||
my ( $self, %opt ) = @_;
|
my ( $self, %opt ) = @_;
|
||||||
my $uid = $opt{uid};
|
my $uid = $opt{uid};
|
||||||
my $status_id = $opt{status_id};
|
my $ts = $opt{ts};
|
||||||
my $db = $opt{db} // $self->{pg}->db;
|
my $db = $opt{db} // $self->{pg}->db;
|
||||||
|
|
||||||
my $res_h
|
my $res_h
|
||||||
= $db->select( 'traewelling', 'data', { user_id => $uid } )->expand->hash;
|
= $db->select( 'traewelling', 'data', { user_id => $uid } )->expand->hash;
|
||||||
|
|
||||||
$res_h->{data}{latest_push_status_id} = $status_id;
|
$res_h->{data}{latest_push_ts} = $ts;
|
||||||
|
|
||||||
$db->update(
|
$db->update(
|
||||||
'traewelling',
|
'traewelling',
|
||||||
|
@ -181,12 +181,24 @@ sub set_sync {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
sub get_push_accounts {
|
sub get_pushable_accounts {
|
||||||
my ($self) = @_;
|
my ($self) = @_;
|
||||||
my $res = $self->{pg}->db->select(
|
my $now = $self->now->epoch;
|
||||||
'traewelling',
|
my $res = $self->{pg}->db->query(
|
||||||
[ 'user_id', 'token', 'data' ],
|
qq{select t.user_id as uid, t.token as token, t.data as data,
|
||||||
{ push_sync => 1 }
|
i.checkin_station_id as dep_eva, i.checkout_station_id as arr_eva,
|
||||||
|
i.data as journey_data, i.train_type as train_type,
|
||||||
|
i.train_line as train_line, i.train_no as train_no,
|
||||||
|
extract(epoch from i.checkin_time) as checkin_ts
|
||||||
|
from traewelling as t
|
||||||
|
join in_transit as i on t.user_id = i.user_id
|
||||||
|
where t.push_sync = True
|
||||||
|
and i.checkout_station_id is not null
|
||||||
|
and i.cancelled = False
|
||||||
|
and (extract(epoch from i.sched_departure) > ?
|
||||||
|
or extract(epoch from i.real_departure) > ?)
|
||||||
|
and extract(epoch from i.sched_departure) < ?
|
||||||
|
}, $now - 300, $now - 300, $now + 600
|
||||||
);
|
);
|
||||||
return $res->expand->hashes->each;
|
return $res->expand->hashes->each;
|
||||||
}
|
}
|
||||||
|
|
|
@ -135,7 +135,7 @@
|
||||||
<div class="input-field col s12">
|
<div class="input-field col s12">
|
||||||
<div>
|
<div>
|
||||||
<label>
|
<label>
|
||||||
%= radio_button sync_source => 'travelynx', disabled => undef
|
%= radio_button sync_source => 'travelynx'
|
||||||
<span>Checkin-Synchronisierung travelynx → Träwelling</span>
|
<span>Checkin-Synchronisierung travelynx → Träwelling</span>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
|
|
Loading…
Reference in a new issue