2020-11-28 20:03:51 +00:00
|
|
|
package Travelynx::Model::JourneyStatsCache;
|
2020-12-03 20:42:17 +00:00
|
|
|
|
2023-06-24 16:36:59 +00:00
|
|
|
# Copyright (C) 2020-2023 Birthe Friesel
|
2020-11-28 20:03:51 +00:00
|
|
|
#
|
2021-01-29 17:32:13 +00:00
|
|
|
# SPDX-License-Identifier: AGPL-3.0-or-later
|
2020-11-28 20:03:51 +00:00
|
|
|
|
|
|
|
use strict;
|
|
|
|
use warnings;
|
|
|
|
use 5.020;
|
|
|
|
use utf8;
|
|
|
|
|
|
|
|
import JSON;
|
|
|
|
|
|
|
|
sub new {
|
|
|
|
my ( $class, %opt ) = @_;
|
|
|
|
|
|
|
|
return bless( \%opt, $class );
|
|
|
|
}
|
|
|
|
|
|
|
|
sub add {
|
|
|
|
my ( $self, %opt ) = @_;
|
|
|
|
|
2020-12-03 20:42:17 +00:00
|
|
|
my $db = $opt{db} // $self->{pg}->db;
|
2020-11-28 20:03:51 +00:00
|
|
|
|
|
|
|
eval {
|
|
|
|
$db->insert(
|
|
|
|
'journey_stats',
|
|
|
|
{
|
|
|
|
user_id => $opt{uid},
|
|
|
|
year => $opt{year},
|
|
|
|
month => $opt{month},
|
2020-12-03 20:42:17 +00:00
|
|
|
data => JSON->new->encode( $opt{stats} ),
|
2020-11-28 20:03:51 +00:00
|
|
|
}
|
|
|
|
);
|
|
|
|
};
|
|
|
|
if ( my $err = $@ ) {
|
2020-12-03 20:42:17 +00:00
|
|
|
if ( $err =~ m{duplicate key value violates unique constraint} ) {
|
|
|
|
|
|
|
|
# If a user opens the same history page several times in
|
|
|
|
# short succession, there is a race condition where several
|
|
|
|
# Mojolicious workers execute this helper, notice that there is
|
|
|
|
# no up-to-date history, compute it, and insert it using the
|
|
|
|
# statement above. This will lead to a uniqueness violation
|
|
|
|
# in each successive insert. However, this is harmless, and
|
|
|
|
# thus ignored.
|
2020-11-28 20:03:51 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
# Otherwise we probably have a problem.
|
|
|
|
die($@);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sub get {
|
|
|
|
my ( $self, %opt ) = @_;
|
|
|
|
|
2020-12-03 20:42:17 +00:00
|
|
|
my $db = $opt{db} // $self->{pg}->db;
|
2020-11-28 20:03:51 +00:00
|
|
|
|
|
|
|
my $stats = $db->select(
|
|
|
|
'journey_stats',
|
|
|
|
['data'],
|
|
|
|
{
|
|
|
|
user_id => $opt{uid},
|
|
|
|
year => $opt{year},
|
|
|
|
month => $opt{month}
|
|
|
|
}
|
|
|
|
)->expand->hash;
|
|
|
|
|
|
|
|
return $stats->{data};
|
|
|
|
}
|
|
|
|
|
|
|
|
# Statistics are partitioned by real_departure, which must be provided
|
|
|
|
# when calling this function e.g. after journey deletion or editing.
|
|
|
|
# If a joureny's real_departure has been edited, this function must be
|
|
|
|
# called twice: once with the old and once with the new value.
|
|
|
|
sub invalidate {
|
|
|
|
my ( $self, %opt ) = @_;
|
|
|
|
|
|
|
|
my $ts = $opt{ts};
|
|
|
|
my $db = $opt{db} // $self->{pg}->db;
|
|
|
|
my $uid = $opt{uid};
|
|
|
|
|
|
|
|
$db->delete(
|
|
|
|
'journey_stats',
|
|
|
|
{
|
|
|
|
user_id => $uid,
|
|
|
|
year => $ts->year,
|
|
|
|
month => $ts->month,
|
|
|
|
}
|
|
|
|
);
|
|
|
|
$db->delete(
|
|
|
|
'journey_stats',
|
|
|
|
{
|
|
|
|
user_id => $uid,
|
|
|
|
year => $ts->year,
|
|
|
|
month => 0,
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-12-03 20:42:17 +00:00
|
|
|
sub get_yyyymm_having_stats {
|
|
|
|
my ( $self, %opt ) = @_;
|
|
|
|
my $uid = $opt{uid};
|
|
|
|
my $db = $opt{db} // $self->{pg}->db;
|
|
|
|
my $res = $db->select(
|
|
|
|
'journey_stats',
|
|
|
|
[ 'year', 'month' ],
|
|
|
|
{ user_id => $uid },
|
|
|
|
{ order_by => { -asc => [ 'year', 'month' ] } }
|
|
|
|
);
|
|
|
|
|
|
|
|
my @ret;
|
|
|
|
for my $row ( $res->hashes->each ) {
|
|
|
|
if ( $row->{month} != 0 ) {
|
|
|
|
push( @ret, [ $row->{year}, $row->{month} ] );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return @ret;
|
|
|
|
}
|
|
|
|
|
2020-11-28 20:03:51 +00:00
|
|
|
1;
|