|
Packit |
95306a |
package Date::Manip::Delta;
|
|
Packit |
95306a |
# Copyright (c) 1995-2017 Sullivan Beck. All rights reserved.
|
|
Packit |
95306a |
# This program is free software; you can redistribute it and/or modify it
|
|
Packit |
95306a |
# under the same terms as Perl itself.
|
|
Packit |
95306a |
|
|
Packit |
95306a |
########################################################################
|
|
Packit |
95306a |
# Any routine that starts with an underscore (_) is NOT intended for
|
|
Packit |
95306a |
# public use. They are for internal use in the the Date::Manip
|
|
Packit |
95306a |
# modules and are subject to change without warning or notice.
|
|
Packit |
95306a |
#
|
|
Packit |
95306a |
# ABSOLUTELY NO USER SUPPORT IS OFFERED FOR THESE ROUTINES!
|
|
Packit |
95306a |
########################################################################
|
|
Packit |
95306a |
|
|
Packit |
95306a |
use Date::Manip::Obj;
|
|
Packit |
95306a |
@ISA = ('Date::Manip::Obj');
|
|
Packit |
95306a |
|
|
Packit |
95306a |
require 5.010000;
|
|
Packit |
95306a |
use warnings;
|
|
Packit |
95306a |
use strict;
|
|
Packit |
95306a |
use utf8;
|
|
Packit |
95306a |
use IO::File;
|
|
Packit |
95306a |
#use re 'debug';
|
|
Packit |
95306a |
|
|
Packit |
95306a |
use Date::Manip::Base;
|
|
Packit |
95306a |
use Date::Manip::TZ;
|
|
Packit |
95306a |
|
|
Packit |
95306a |
our $VERSION;
|
|
Packit |
95306a |
$VERSION='6.60';
|
|
Packit |
95306a |
END { undef $VERSION; }
|
|
Packit |
95306a |
|
|
Packit |
95306a |
########################################################################
|
|
Packit |
95306a |
# BASE METHODS
|
|
Packit |
95306a |
########################################################################
|
|
Packit |
95306a |
|
|
Packit |
95306a |
sub is_delta {
|
|
Packit |
95306a |
return 1;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
sub config {
|
|
Packit |
95306a |
my($self,@args) = @_;
|
|
Packit |
95306a |
$self->SUPER::config(@args);
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# A new config can change the value of the format fields, so clear them.
|
|
Packit |
95306a |
$$self{'data'}{'f'} = {};
|
|
Packit |
95306a |
$$self{'data'}{'flen'} = {};
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# Call this every time a new delta is put in to make sure everything is
|
|
Packit |
95306a |
# correctly initialized.
|
|
Packit |
95306a |
#
|
|
Packit |
95306a |
sub _init {
|
|
Packit |
95306a |
my($self) = @_;
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my $def = [0,0,0,0,0,0,0];
|
|
Packit |
95306a |
my $dmt = $$self{'tz'};
|
|
Packit |
95306a |
my $dmb = $$dmt{'base'};
|
|
Packit |
95306a |
|
|
Packit |
95306a |
$$self{'err'} = '';
|
|
Packit |
95306a |
$$self{'data'} = {
|
|
Packit |
95306a |
'delta' => $def, # the delta (all negative fields signed)
|
|
Packit |
95306a |
'in' => '', # the string that was parsed (if any)
|
|
Packit |
95306a |
'length' => 0, # length of delta (in seconds)
|
|
Packit |
95306a |
|
|
Packit |
95306a |
'gotmode' => 0, # 1 if mode set explicitly
|
|
Packit |
95306a |
'business' => 0, # 1 for a business delta
|
|
Packit |
95306a |
|
|
Packit |
95306a |
'f' => {}, # format fields
|
|
Packit |
95306a |
'flen' => {}, # field lengths
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
sub _init_args {
|
|
Packit |
95306a |
my($self) = @_;
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my @args = @{ $$self{'args'} };
|
|
Packit |
95306a |
$self->parse(@args);
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
sub value {
|
|
Packit |
95306a |
my($self) = @_;
|
|
Packit |
95306a |
my $dmt = $$self{'tz'};
|
|
Packit |
95306a |
my $dmb = $$dmt{'base'};
|
|
Packit |
95306a |
|
|
Packit |
95306a |
return '' if ($$self{'err'});
|
|
Packit |
95306a |
if (wantarray) {
|
|
Packit |
95306a |
return @{ $$self{'data'}{'delta'} };
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
my @delta = @{ $$self{'data'}{'delta'} };
|
|
Packit |
95306a |
my $err;
|
|
Packit |
95306a |
($err,@delta) = $dmb->_delta_fields( { 'nonorm' => 1,
|
|
Packit |
95306a |
'source' => 'delta',
|
|
Packit |
95306a |
'sign' => 0 },
|
|
Packit |
95306a |
[@delta]);
|
|
Packit |
95306a |
return '' if ($err);
|
|
Packit |
95306a |
return join(':',@delta);
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
sub input {
|
|
Packit |
95306a |
my($self) = @_;
|
|
Packit |
95306a |
return $$self{'data'}{'in'};
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
########################################################################
|
|
Packit |
95306a |
# DELTA METHODS
|
|
Packit |
95306a |
########################################################################
|
|
Packit |
95306a |
|
|
Packit |
95306a |
BEGIN {
|
|
Packit |
95306a |
my %ops = map { $_,1 } qw( delta business normal standard );
|
|
Packit |
95306a |
my %f = qw( y 0 M 1 w 2 d 3 h 4 m 5 s 6 );
|
|
Packit |
95306a |
|
|
Packit |
95306a |
sub set {
|
|
Packit |
95306a |
my($self,$field,$val,$no_normalize) = @_;
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my $dmt = $$self{'tz'};
|
|
Packit |
95306a |
my $dmb = $$dmt{'base'};
|
|
Packit |
95306a |
my $zone = $$self{'data'}{'tz'};
|
|
Packit |
95306a |
my $gotmode = $$self{'data'}{'gotmode'};
|
|
Packit |
95306a |
my $business = 0;
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my (@delta,$err);
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if (exists $ops{lc($field)}) {
|
|
Packit |
95306a |
$field = lc($field);
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if ($field eq 'business') {
|
|
Packit |
95306a |
$business = 1;
|
|
Packit |
95306a |
$gotmode = 1;
|
|
Packit |
95306a |
} elsif ($field eq 'normal' || $field eq 'standard') {
|
|
Packit |
95306a |
$business = 0;
|
|
Packit |
95306a |
$gotmode = 1;
|
|
Packit |
95306a |
} elsif ($field eq 'delta') {
|
|
Packit |
95306a |
$business = $$self{'data'}{'business'};
|
|
Packit |
95306a |
$gotmode = $$self{'data'}{'gotmode'};
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
($err,@delta) = $dmb->_delta_fields( { 'nonorm' => $no_normalize,
|
|
Packit |
95306a |
'business' => $business,
|
|
Packit |
95306a |
'source' => 'delta',
|
|
Packit |
95306a |
'sign' => -1 },
|
|
Packit |
95306a |
$val);
|
|
Packit |
95306a |
|
|
Packit |
95306a |
} elsif (exists $f{$field}) {
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if ($$self{'err'}) {
|
|
Packit |
95306a |
$$self{'err'} = "[set] Invalid delta";
|
|
Packit |
95306a |
return 1;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
@delta = @{ $$self{'data'}{'delta'} };
|
|
Packit |
95306a |
$business = $$self{'data'}{'business'};
|
|
Packit |
95306a |
$delta[$f{$field}] = $val;
|
|
Packit |
95306a |
|
|
Packit |
95306a |
($err,@delta) = $dmb->_delta_fields( { 'nonorm' => $no_normalize,
|
|
Packit |
95306a |
'business' => $business,
|
|
Packit |
95306a |
'source' => 'delta',
|
|
Packit |
95306a |
'sign' => -1 },
|
|
Packit |
95306a |
[@delta]);
|
|
Packit |
95306a |
|
|
Packit |
95306a |
} elsif (lc($field) eq 'mode') {
|
|
Packit |
95306a |
|
|
Packit |
95306a |
@delta = @{ $$self{'data'}{'delta'} };
|
|
Packit |
95306a |
$val = lc($val);
|
|
Packit |
95306a |
if ($val eq 'business' || $val eq 'normal' || $val eq 'standard') {
|
|
Packit |
95306a |
$gotmode = 1;
|
|
Packit |
95306a |
$business = ($val eq 'business' ? 1 : 0);
|
|
Packit |
95306a |
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
$$self{'err'} = "[set] Invalid mode: $val";
|
|
Packit |
95306a |
return 1;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
|
|
Packit |
95306a |
$$self{'err'} = "[set] Invalid field: $field";
|
|
Packit |
95306a |
return 1;
|
|
Packit |
95306a |
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if ($err) {
|
|
Packit |
95306a |
$$self{'err'} = "[set] Invalid field value: $field";
|
|
Packit |
95306a |
return 1;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
$self->_init();
|
|
Packit |
95306a |
$$self{'data'}{'delta'} = [ @delta ];
|
|
Packit |
95306a |
$$self{'data'}{'business'} = $business;
|
|
Packit |
95306a |
$$self{'data'}{'gotmode'} = $gotmode;
|
|
Packit |
95306a |
$$self{'data'}{'length'} = 'unknown';
|
|
Packit |
95306a |
|
|
Packit |
95306a |
return 0;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
sub _rx {
|
|
Packit |
95306a |
my($self,$rx) = @_;
|
|
Packit |
95306a |
my $dmt = $$self{'tz'};
|
|
Packit |
95306a |
my $dmb = $$dmt{'base'};
|
|
Packit |
95306a |
|
|
Packit |
95306a |
return $$dmb{'data'}{'rx'}{'delta'}{$rx}
|
|
Packit |
95306a |
if (exists $$dmb{'data'}{'rx'}{'delta'}{$rx});
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if ($rx eq 'expanded') {
|
|
Packit |
95306a |
my $sign = '[-+]?\s*';
|
|
Packit |
95306a |
my $sep = '(?:,\s*|\s+|$)';
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my $nth = $$dmb{'data'}{'rx'}{'nth'}[0];
|
|
Packit |
95306a |
my $yf = $$dmb{data}{rx}{fields}[1];
|
|
Packit |
95306a |
my $mf = $$dmb{data}{rx}{fields}[2];
|
|
Packit |
95306a |
my $wf = $$dmb{data}{rx}{fields}[3];
|
|
Packit |
95306a |
my $df = $$dmb{data}{rx}{fields}[4];
|
|
Packit |
95306a |
my $hf = $$dmb{data}{rx}{fields}[5];
|
|
Packit |
95306a |
my $mnf = $$dmb{data}{rx}{fields}[6];
|
|
Packit |
95306a |
my $sf = $$dmb{data}{rx}{fields}[7];
|
|
Packit |
95306a |
my $num = '(?:\d+(?:\.\d*)?|\.\d+)';
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my $y = "(?:(?:(?<y>$sign$num)|(?<y>$nth))\\s*(?:$yf)$sep)";
|
|
Packit |
95306a |
my $m = "(?:(?:(?<m>$sign$num)|(?<m>$nth))\\s*(?:$mf)$sep)";
|
|
Packit |
95306a |
my $w = "(?:(?:(?<w>$sign$num)|(?<w>$nth))\\s*(?:$wf)$sep)";
|
|
Packit |
95306a |
my $d = "(?:(?:(?<d>$sign$num)|(?<d>$nth))\\s*(?:$df)$sep)";
|
|
Packit |
95306a |
my $h = "(?:(?:(?<h>$sign$num)|(?<h>$nth))\\s*(?:$hf)$sep)";
|
|
Packit |
95306a |
my $mn = "(?:(?:(?<mn>$sign$num)|(?<mn>$nth))\\s*(?:$mnf)$sep)";
|
|
Packit |
95306a |
my $s = "(?:(?:(?<s>$sign$num)|(?<s>$nth))\\s*(?:$sf)?)";
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my $exprx = qr/^\s*$y?$m?$w?$d?$h?$mn?$s?\s*$/i;
|
|
Packit |
95306a |
$$dmb{'data'}{'rx'}{'delta'}{$rx} = $exprx;
|
|
Packit |
95306a |
|
|
Packit |
95306a |
} elsif ($rx eq 'mode') {
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my $mode = qr/\b($$dmb{'data'}{'rx'}{'mode'}[0])\b/i;
|
|
Packit |
95306a |
$$dmb{'data'}{'rx'}{'delta'}{$rx} = $mode;
|
|
Packit |
95306a |
|
|
Packit |
95306a |
} elsif ($rx eq 'when') {
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my $when = qr/\b($$dmb{'data'}{'rx'}{'when'}[0])\b/i;
|
|
Packit |
95306a |
$$dmb{'data'}{'rx'}{'delta'}{$rx} = $when;
|
|
Packit |
95306a |
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
return $$dmb{'data'}{'rx'}{'delta'}{$rx};
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
sub parse {
|
|
Packit |
95306a |
my($self,$instring,@args) = @_;
|
|
Packit |
95306a |
my($business,$no_normalize,$gotmode,$err,@delta);
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if (@args == 2) {
|
|
Packit |
95306a |
($business,$no_normalize) = (lc($args[0]),lc($args[1]));
|
|
Packit |
95306a |
if ($business eq 'standard') {
|
|
Packit |
95306a |
$business = 0;
|
|
Packit |
95306a |
} elsif ($business eq 'business') {
|
|
Packit |
95306a |
$business = 1;
|
|
Packit |
95306a |
} elsif ($business) {
|
|
Packit |
95306a |
$business = 1;
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
$business = 0;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
if ($no_normalize) {
|
|
Packit |
95306a |
$no_normalize = 1;
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
$no_normalize = 0;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
$gotmode = 1;
|
|
Packit |
95306a |
|
|
Packit |
95306a |
} elsif (@args == 1) {
|
|
Packit |
95306a |
my $arg = lc($args[0]);
|
|
Packit |
95306a |
if ($arg eq 'standard') {
|
|
Packit |
95306a |
$business = 0;
|
|
Packit |
95306a |
$no_normalize = 0;
|
|
Packit |
95306a |
$gotmode = 1;
|
|
Packit |
95306a |
} elsif ($arg eq 'business') {
|
|
Packit |
95306a |
$business = 1;
|
|
Packit |
95306a |
$no_normalize = 0;
|
|
Packit |
95306a |
$gotmode = 1;
|
|
Packit |
95306a |
} elsif ($arg eq 'nonormalize') {
|
|
Packit |
95306a |
$business = 0;
|
|
Packit |
95306a |
$no_normalize = 1;
|
|
Packit |
95306a |
$gotmode = 0;
|
|
Packit |
95306a |
} elsif ($arg) {
|
|
Packit |
95306a |
$business = 1;
|
|
Packit |
95306a |
$no_normalize = 0;
|
|
Packit |
95306a |
$gotmode = 1;
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
$business = 0;
|
|
Packit |
95306a |
$no_normalize = 0;
|
|
Packit |
95306a |
$gotmode = 0;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
} elsif (@args == 0) {
|
|
Packit |
95306a |
$business = 0;
|
|
Packit |
95306a |
$no_normalize = 0;
|
|
Packit |
95306a |
$gotmode = 0;
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
$$self{'err'} = "[parse] Unknown arguments";
|
|
Packit |
95306a |
return 1;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my $dmt = $$self{'tz'};
|
|
Packit |
95306a |
my $dmb = $$dmt{'base'};
|
|
Packit |
95306a |
$self->_init();
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if (! $instring) {
|
|
Packit |
95306a |
$$self{'err'} = '[parse] Empty delta string';
|
|
Packit |
95306a |
return 1;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
#
|
|
Packit |
95306a |
# Parse the string
|
|
Packit |
95306a |
#
|
|
Packit |
95306a |
|
|
Packit |
95306a |
$$self{'err'} = '';
|
|
Packit |
95306a |
$instring =~ s/^\s*//;
|
|
Packit |
95306a |
$instring =~ s/\s*$//;
|
|
Packit |
95306a |
|
|
Packit |
95306a |
PARSE: {
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# First, we'll try the standard format (without a mode string)
|
|
Packit |
95306a |
|
|
Packit |
95306a |
($err,@delta) = $dmb->_split_delta($instring);
|
|
Packit |
95306a |
last PARSE if (! $err);
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# Next, we'll need to get a list of all the encodings and look
|
|
Packit |
95306a |
# for (and remove) the mode string from each. We'll also recheck
|
|
Packit |
95306a |
# the standard format for each.
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my @strings = $dmb->_encoding($instring);
|
|
Packit |
95306a |
my $moderx = $self->_rx('mode');
|
|
Packit |
95306a |
my %mode = ();
|
|
Packit |
95306a |
|
|
Packit |
95306a |
foreach my $string (@strings) {
|
|
Packit |
95306a |
if ($string =~ s/\s*$moderx\s*//i) {
|
|
Packit |
95306a |
my $b = $1;
|
|
Packit |
95306a |
if ($$dmb{'data'}{'wordmatch'}{'mode'}{lc($b)} == 1) {
|
|
Packit |
95306a |
$b = 0;
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
$b = 1;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
($err,@delta) = $dmb->_split_delta($string);
|
|
Packit |
95306a |
if (! $err) {
|
|
Packit |
95306a |
$business = $b;
|
|
Packit |
95306a |
$gotmode = 1;
|
|
Packit |
95306a |
last PARSE;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
$mode{$string} = $b;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# Now we'll check each string for an expanded form delta.
|
|
Packit |
95306a |
|
|
Packit |
95306a |
foreach my $string (@strings) {
|
|
Packit |
95306a |
my($b,$g);
|
|
Packit |
95306a |
if (exists $mode{$string}) {
|
|
Packit |
95306a |
$b = $mode{$string};
|
|
Packit |
95306a |
$g = 1;
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
$b = $business;
|
|
Packit |
95306a |
$g = 0;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my $past = 0;
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my $whenrx = $self->_rx('when');
|
|
Packit |
95306a |
if ($string &&
|
|
Packit |
95306a |
$string =~ s/$whenrx//i) {
|
|
Packit |
95306a |
my $when = $1;
|
|
Packit |
95306a |
if ($$dmb{'data'}{'wordmatch'}{'when'}{lc($when)} == 1) {
|
|
Packit |
95306a |
$past = 1;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my $rx = $self->_rx('expanded');
|
|
Packit |
95306a |
if ($string &&
|
|
Packit |
95306a |
$string =~ $rx) {
|
|
Packit |
95306a |
$business = $b;
|
|
Packit |
95306a |
$gotmode = $g;
|
|
Packit |
95306a |
@delta = @+{qw(y m w d h mn s)};
|
|
Packit |
95306a |
foreach my $f (@delta) {
|
|
Packit |
95306a |
if (! defined $f) {
|
|
Packit |
95306a |
$f = 0;
|
|
Packit |
95306a |
} elsif (exists $$dmb{'data'}{'wordmatch'}{'nth'}{lc($f)}) {
|
|
Packit |
95306a |
$f = $$dmb{'data'}{'wordmatch'}{'nth'}{lc($f)};
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
$f =~ s/\s//g;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# if $past, reverse the signs
|
|
Packit |
95306a |
if ($past) {
|
|
Packit |
95306a |
foreach my $v (@delta) {
|
|
Packit |
95306a |
$v *= -1;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
last PARSE;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if (! @delta) {
|
|
Packit |
95306a |
$$self{'err'} = "[parse] Invalid delta string";
|
|
Packit |
95306a |
return 1;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
($err,@delta) = $dmb->_delta_fields( { 'nonorm' => $no_normalize,
|
|
Packit |
95306a |
'business' => $business,
|
|
Packit |
95306a |
'source' => 'string',
|
|
Packit |
95306a |
'sign' => -1 },
|
|
Packit |
95306a |
[@delta]);
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if ($err) {
|
|
Packit |
95306a |
$$self{'err'} = "[parse] Invalid delta string";
|
|
Packit |
95306a |
return 1;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
$$self{'data'}{'in'} = $instring;
|
|
Packit |
95306a |
$$self{'data'}{'delta'} = [@delta];
|
|
Packit |
95306a |
$$self{'data'}{'business'} = $business;
|
|
Packit |
95306a |
$$self{'data'}{'gotmode'} = $gotmode;
|
|
Packit |
95306a |
$$self{'data'}{'length'} = 'unknown';
|
|
Packit |
95306a |
return 0;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
sub printf {
|
|
Packit |
95306a |
my($self,@in) = @_;
|
|
Packit |
95306a |
if ($$self{'err'}) {
|
|
Packit |
95306a |
warn "WARNING: [printf] Object must contain a valid delta\n";
|
|
Packit |
95306a |
return undef;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my($y,$M,$w,$d,$h,$m,$s) = @{ $$self{'data'}{'delta'} };
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my @out;
|
|
Packit |
95306a |
foreach my $in (@in) {
|
|
Packit |
95306a |
my $out = '';
|
|
Packit |
95306a |
while ($in) {
|
|
Packit |
95306a |
if ($in =~ s/^([^%]+)//) {
|
|
Packit |
95306a |
$out .= $1;
|
|
Packit |
95306a |
|
|
Packit |
95306a |
} elsif ($in =~ s/^%%//) {
|
|
Packit |
95306a |
$out .= "%";
|
|
Packit |
95306a |
|
|
Packit |
95306a |
} elsif ($in =~ s/^%
|
|
Packit |
95306a |
(\+)? # sign
|
|
Packit |
95306a |
([<>0])? # pad
|
|
Packit |
95306a |
(\d+)? # width
|
|
Packit |
95306a |
([yMwdhms]) # field
|
|
Packit |
95306a |
v # type
|
|
Packit |
95306a |
//ox) {
|
|
Packit |
95306a |
my($sign,$pad,$width,$field) = ($1,$2,$3,$4);
|
|
Packit |
95306a |
$out .= $self->_printf_field($sign,$pad,$width,0,$field);
|
|
Packit |
95306a |
|
|
Packit |
95306a |
} elsif ($in =~ s/^(%
|
|
Packit |
95306a |
(\+)? # sign
|
|
Packit |
95306a |
([<>0])? # pad
|
|
Packit |
95306a |
(\d+)? # width
|
|
Packit |
95306a |
(?:\.(\d+))? # precision
|
|
Packit |
95306a |
([yMwdhms]) # field
|
|
Packit |
95306a |
([yMwdhms]) # field0
|
|
Packit |
95306a |
([yMwdhms]) # field1
|
|
Packit |
95306a |
)//ox) {
|
|
Packit |
95306a |
my($match,$sign,$pad,$width,$precision,$field,$field0,$field1) =
|
|
Packit |
95306a |
($1,$2,$3,$4,$5,$6,$7,$8);
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# Get the list of fields we're expressing
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my @field = qw(y M w d h m s);
|
|
Packit |
95306a |
while (@field && $field[0] ne $field0) {
|
|
Packit |
95306a |
shift(@field);
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
while (@field && $field[$#field] ne $field1) {
|
|
Packit |
95306a |
pop(@field);
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if (! @field) {
|
|
Packit |
95306a |
$out .= $match;
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
$out .=
|
|
Packit |
95306a |
$self->_printf_field($sign,$pad,$width,$precision,$field,@field);
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
} elsif ($in =~ s/^%
|
|
Packit |
95306a |
(\+)? # sign
|
|
Packit |
95306a |
([<>])? # pad
|
|
Packit |
95306a |
(\d+)? # width
|
|
Packit |
95306a |
Dt
|
|
Packit |
95306a |
//ox) {
|
|
Packit |
95306a |
my($sign,$pad,$width) = ($1,$2,$3);
|
|
Packit |
95306a |
$out .= $self->_printf_delta($sign,$pad,$width,'y','s');
|
|
Packit |
95306a |
|
|
Packit |
95306a |
} elsif ($in =~ s/^(%
|
|
Packit |
95306a |
(\+)? # sign
|
|
Packit |
95306a |
([<>])? # pad
|
|
Packit |
95306a |
(\d+)? # width
|
|
Packit |
95306a |
D
|
|
Packit |
95306a |
([yMwdhms]) # field0
|
|
Packit |
95306a |
([yMwdhms]) # field1
|
|
Packit |
95306a |
)//ox) {
|
|
Packit |
95306a |
my($match,$sign,$pad,$width,$field0,$field1) = ($1,$2,$3,$4,$5,$6);
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# Get the list of fields we're expressing
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my @field = qw(y M w d h m s);
|
|
Packit |
95306a |
while (@field && $field[0] ne $field0) {
|
|
Packit |
95306a |
shift(@field);
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
while (@field && $field[$#field] ne $field1) {
|
|
Packit |
95306a |
pop(@field);
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if (! @field) {
|
|
Packit |
95306a |
$out .= $match;
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
$out .= $self->_printf_delta($sign,$pad,$width,$field[0],
|
|
Packit |
95306a |
$field[$#field]);
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
$in =~ s/^(%[^%]*)//;
|
|
Packit |
95306a |
$out .= $1;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
push(@out,$out);
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if (wantarray) {
|
|
Packit |
95306a |
return @out;
|
|
Packit |
95306a |
} elsif (@out == 1) {
|
|
Packit |
95306a |
return $out[0];
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
return ''
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
sub _printf_delta {
|
|
Packit |
95306a |
my($self,$sign,$pad,$width,$field0,$field1) = @_;
|
|
Packit |
95306a |
my $dmt = $$self{'tz'};
|
|
Packit |
95306a |
my $dmb = $$dmt{'base'};
|
|
Packit |
95306a |
my @delta = @{ $$self{'data'}{'delta'} };
|
|
Packit |
95306a |
my $delta;
|
|
Packit |
95306a |
my %tmp = qw(y 0 M 1 w 2 d 3 h 4 m 5 s 6);
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# Add a sign to each field
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my $s = "+";
|
|
Packit |
95306a |
foreach my $f (@delta) {
|
|
Packit |
95306a |
if ($f < 0) {
|
|
Packit |
95306a |
$s = "-";
|
|
Packit |
95306a |
} elsif ($f > 0) {
|
|
Packit |
95306a |
$s = "+";
|
|
Packit |
95306a |
$f *= 1;
|
|
Packit |
95306a |
$f = "+$f";
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
$f = "$s$f";
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# Split the delta into field sets containing only those fields to
|
|
Packit |
95306a |
# print.
|
|
Packit |
95306a |
#
|
|
Packit |
95306a |
# @set = ( [SETa] [SETb] ....)
|
|
Packit |
95306a |
# where [SETx] is a listref of fields from one set of fields
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my @set;
|
|
Packit |
95306a |
my $business = $$self{'data'}{'business'};
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my $f0 = $tmp{$field0};
|
|
Packit |
95306a |
my $f1 = $tmp{$field1};
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if ($field0 eq $field1) {
|
|
Packit |
95306a |
@set = ( [ $delta[$f0] ] );
|
|
Packit |
95306a |
|
|
Packit |
95306a |
} elsif ($business) {
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if ($f0 <= 1) {
|
|
Packit |
95306a |
# if (field0 = y or M)
|
|
Packit |
95306a |
# add [y,M]
|
|
Packit |
95306a |
# field0 = w OR done if field1 = M
|
|
Packit |
95306a |
push(@set, [ @delta[0..1] ]);
|
|
Packit |
95306a |
$f0 = ($f1 == 1 ? 7 : 2);
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if ($f0 == 2) {
|
|
Packit |
95306a |
# if (field0 = w)
|
|
Packit |
95306a |
# add [w]
|
|
Packit |
95306a |
# field0 = d OR done if field1 = w
|
|
Packit |
95306a |
push(@set, [ $delta[2] ]);
|
|
Packit |
95306a |
$f0 = ($f1 == 2 ? 7 : 3);
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if ($f0 <= 6) {
|
|
Packit |
95306a |
push(@set, [ @delta[$f0..$f1] ]);
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if ($f0 <= 1) {
|
|
Packit |
95306a |
# if (field0 = y or M)
|
|
Packit |
95306a |
# add [y,M]
|
|
Packit |
95306a |
# field0 = w OR done if field1 = M
|
|
Packit |
95306a |
push(@set, [ @delta[0..1] ]);
|
|
Packit |
95306a |
$f0 = ($f1 == 1 ? 7 : 2);
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if ($f0 <= 6) {
|
|
Packit |
95306a |
push(@set, [ @delta[$f0..$f1] ]);
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# If we're not forcing signs, remove signs from all fields
|
|
Packit |
95306a |
# except the first in each set.
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my @ret;
|
|
Packit |
95306a |
|
|
Packit |
95306a |
foreach my $set (@set) {
|
|
Packit |
95306a |
my @f = @$set;
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if (defined($sign) && $sign eq "+") {
|
|
Packit |
95306a |
push(@ret,@f);
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
push(@ret,shift(@f));
|
|
Packit |
95306a |
foreach my $f (@f) {
|
|
Packit |
95306a |
$f =~ s/[-+]//;
|
|
Packit |
95306a |
push(@ret,$f);
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# Width/pad
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my $ret = join(':',@ret);
|
|
Packit |
95306a |
if ($width && length($ret) < $width) {
|
|
Packit |
95306a |
if (defined $pad && $pad eq ">") {
|
|
Packit |
95306a |
$ret .= ' 'x($width-length($ret));
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
$ret = ' 'x($width-length($ret)) . $ret;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
return $ret;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
sub _printf_field {
|
|
Packit |
95306a |
my($self,$sign,$pad,$width,$precision,$field,@field) = @_;
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my $val = $self->_printf_field_val($field,@field);
|
|
Packit |
95306a |
$pad = "<" if (! defined($pad));
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# Strip off the sign.
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my $s = '';
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if ($val < 0) {
|
|
Packit |
95306a |
$s = "-";
|
|
Packit |
95306a |
$val *= -1;
|
|
Packit |
95306a |
} elsif ($sign) {
|
|
Packit |
95306a |
$s = "+";
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# Handle the precision.
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if (defined($precision)) {
|
|
Packit |
95306a |
$val = sprintf("%.${precision}f",$val);
|
|
Packit |
95306a |
|
|
Packit |
95306a |
} elsif (defined($width)) {
|
|
Packit |
95306a |
my $i = $s . int($val) . '.';
|
|
Packit |
95306a |
if (length($i) < $width) {
|
|
Packit |
95306a |
$precision = $width-length($i);
|
|
Packit |
95306a |
$val = sprintf("%.${precision}f",$val);
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# Handle padding.
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if ($width) {
|
|
Packit |
95306a |
if ($pad eq ">") {
|
|
Packit |
95306a |
$val = "$s$val";
|
|
Packit |
95306a |
my $pad = ($width > length($val) ? $width - length($val) : 0);
|
|
Packit |
95306a |
$val .= ' 'x$pad;
|
|
Packit |
95306a |
|
|
Packit |
95306a |
} elsif ($pad eq "<") {
|
|
Packit |
95306a |
$val = "$s$val";
|
|
Packit |
95306a |
my $pad = ($width > length($val) ? $width - length($val) : 0);
|
|
Packit |
95306a |
$val = ' 'x$pad . $val;
|
|
Packit |
95306a |
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
my $pad = ($width > length($val)-length($s) ?
|
|
Packit |
95306a |
$width - length($val) - length($s): 0);
|
|
Packit |
95306a |
$val = $s . '0'x$pad . $val;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
$val = "$s$val";
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
return $val;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# $$self{'data'}{'f'}{X}{Y} is the value of field X expressed in terms of Y.
|
|
Packit |
95306a |
#
|
|
Packit |
95306a |
sub _printf_field_val {
|
|
Packit |
95306a |
my($self,$field,@field) = @_;
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if (! exists $$self{'data'}{'f'}{'y'} &&
|
|
Packit |
95306a |
! exists $$self{'data'}{'f'}{'y'}{'y'}) {
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my($yv,$Mv,$wv,$dv,$hv,$mv,$sv) = map { $_*1 } @{ $$self{'data'}{'delta'} };
|
|
Packit |
95306a |
$$self{'data'}{'f'}{'y'}{'y'} = $yv;
|
|
Packit |
95306a |
$$self{'data'}{'f'}{'M'}{'M'} = $Mv;
|
|
Packit |
95306a |
$$self{'data'}{'f'}{'w'}{'w'} = $wv;
|
|
Packit |
95306a |
$$self{'data'}{'f'}{'d'}{'d'} = $dv;
|
|
Packit |
95306a |
$$self{'data'}{'f'}{'h'}{'h'} = $hv;
|
|
Packit |
95306a |
$$self{'data'}{'f'}{'m'}{'m'} = $mv;
|
|
Packit |
95306a |
$$self{'data'}{'f'}{'s'}{'s'} = $sv;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# A single field
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if (! @field) {
|
|
Packit |
95306a |
return $$self{'data'}{'f'}{$field}{$field};
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# Find the length of 1 unit of each field in terms of seconds.
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if (! exists $$self{'data'}{'flen'}{'s'}) {
|
|
Packit |
95306a |
my $business = $$self{'data'}{'business'};
|
|
Packit |
95306a |
my $dmb = $self->base();
|
|
Packit |
95306a |
$$self{'data'}{'flen'} = { 's' => 1,
|
|
Packit |
95306a |
'm' => 60,
|
|
Packit |
95306a |
'h' => 3600,
|
|
Packit |
95306a |
'd' => $$dmb{'data'}{'len'}{$business}{'dl'},
|
|
Packit |
95306a |
'w' => $$dmb{'data'}{'len'}{$business}{'wl'},
|
|
Packit |
95306a |
'M' => $$dmb{'data'}{'len'}{$business}{'ml'},
|
|
Packit |
95306a |
'y' => $$dmb{'data'}{'len'}{$business}{'yl'},
|
|
Packit |
95306a |
};
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# Calculate the value for each field.
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my $val = 0;
|
|
Packit |
95306a |
foreach my $f (@field) {
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# We want the value of $f expressed in terms of $field
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if (! exists $$self{'data'}{'f'}{$f}{$field}) {
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# Get the value of $f expressed in seconds
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if (! exists $$self{'data'}{'f'}{$f}{'s'}) {
|
|
Packit |
95306a |
$$self{'data'}{'f'}{$f}{'s'} =
|
|
Packit |
95306a |
$$self{'data'}{'f'}{$f}{$f} * $$self{'data'}{'flen'}{$f};
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# Get the value of $f expressed in terms of $field
|
|
Packit |
95306a |
|
|
Packit |
95306a |
$$self{'data'}{'f'}{$f}{$field} =
|
|
Packit |
95306a |
$$self{'data'}{'f'}{$f}{'s'} / $$self{'data'}{'flen'}{$field};
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
$val += $$self{'data'}{'f'}{$f}{$field};
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
return $val;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
sub type {
|
|
Packit |
95306a |
my($self,$op) = @_;
|
|
Packit |
95306a |
$op = lc($op);
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if ($op eq 'business') {
|
|
Packit |
95306a |
return $$self{'data'}{'business'};
|
|
Packit |
95306a |
} elsif ($op eq 'standard') {
|
|
Packit |
95306a |
return 1-$$self{'data'}{'business'};
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my($exact,$semi,$approx) = (0,0,0);
|
|
Packit |
95306a |
my($y,$m,$w,$d,$h,$mn,$s) = @{ $$self{'data'}{'delta'} };
|
|
Packit |
95306a |
if ($y || $m) {
|
|
Packit |
95306a |
$approx = 1;
|
|
Packit |
95306a |
} elsif ($w || (! $$self{'data'}{'business'} && $d)) {
|
|
Packit |
95306a |
$semi = 1;
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
$exact = 1;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if ($op eq 'exact') {
|
|
Packit |
95306a |
return $exact;
|
|
Packit |
95306a |
} elsif ($op eq 'semi') {
|
|
Packit |
95306a |
return $semi;
|
|
Packit |
95306a |
} elsif ($op eq 'approx') {
|
|
Packit |
95306a |
return $approx;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
return undef;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
sub calc {
|
|
Packit |
95306a |
my($self,$obj,$subtract,$no_normalize) = @_;
|
|
Packit |
95306a |
if ($$self{'err'}) {
|
|
Packit |
95306a |
$$self{'err'} = "[calc] First object invalid (delta)";
|
|
Packit |
95306a |
return undef;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if (ref($obj) eq 'Date::Manip::Date') {
|
|
Packit |
95306a |
if ($$obj{'err'}) {
|
|
Packit |
95306a |
$$self{'err'} = "[calc] Second object invalid (date)";
|
|
Packit |
95306a |
return undef;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
return $obj->calc($self,$subtract);
|
|
Packit |
95306a |
|
|
Packit |
95306a |
} elsif (ref($obj) eq 'Date::Manip::Delta') {
|
|
Packit |
95306a |
if ($$obj{'err'}) {
|
|
Packit |
95306a |
$$self{'err'} = "[calc] Second object invalid (delta)";
|
|
Packit |
95306a |
return undef;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
return $self->_calc_delta_delta($obj,$subtract,$no_normalize);
|
|
Packit |
95306a |
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
$$self{'err'} = "[calc] Second object must be a Date/Delta object";
|
|
Packit |
95306a |
return undef;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
sub _calc_delta_delta {
|
|
Packit |
95306a |
my($self,$delta,@args) = @_;
|
|
Packit |
95306a |
my $dmt = $$self{'tz'};
|
|
Packit |
95306a |
my $dmb = $$dmt{'base'};
|
|
Packit |
95306a |
my $ret = $self->new_delta;
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if ($self->err()) {
|
|
Packit |
95306a |
$$ret{'err'} = "[calc] First delta object invalid";
|
|
Packit |
95306a |
return $ret;
|
|
Packit |
95306a |
} elsif ($delta->err()) {
|
|
Packit |
95306a |
$$ret{'err'} = "[calc] Second delta object invalid";
|
|
Packit |
95306a |
return $ret;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my($subtract,$no_normalize);
|
|
Packit |
95306a |
if (@args == 2) {
|
|
Packit |
95306a |
($subtract,$no_normalize) = @args;
|
|
Packit |
95306a |
} elsif ($args[0] eq 'nonormalize') {
|
|
Packit |
95306a |
$subtract = 0;
|
|
Packit |
95306a |
$no_normalize = 1;
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
$subtract = 0;
|
|
Packit |
95306a |
$no_normalize = 0;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my $business = 0;
|
|
Packit |
95306a |
if ($$self{'data'}{'business'} != $$delta{'data'}{'business'}) {
|
|
Packit |
95306a |
$$ret{'err'} = "[calc] Delta/delta calculation objects must be of " .
|
|
Packit |
95306a |
'the same type';
|
|
Packit |
95306a |
return $ret;
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
$business = $$self{'data'}{'business'};
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my ($err,@delta);
|
|
Packit |
95306a |
for (my $i=0; $i<7; $i++) {
|
|
Packit |
95306a |
if ($subtract) {
|
|
Packit |
95306a |
$delta[$i] = $$self{'data'}{'delta'}[$i] - $$delta{'data'}{'delta'}[$i];
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
$delta[$i] = $$self{'data'}{'delta'}[$i] + $$delta{'data'}{'delta'}[$i];
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
($err,@delta) = $dmb->_delta_fields( { 'nonorm' => 0,
|
|
Packit |
95306a |
'source' => 'delta',
|
|
Packit |
95306a |
'sign' => -1 },
|
|
Packit |
95306a |
[@delta]) if (! $no_normalize);
|
|
Packit |
95306a |
|
|
Packit |
95306a |
$$ret{'data'}{'delta'} = [@delta];
|
|
Packit |
95306a |
$$ret{'data'}{'business'} = $business;
|
|
Packit |
95306a |
$$self{'data'}{'length'} = 'unknown';
|
|
Packit |
95306a |
|
|
Packit |
95306a |
return $ret;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
sub convert {
|
|
Packit |
95306a |
my($self,$to) = @_;
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# What mode are we currently in
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my $from;
|
|
Packit |
95306a |
my($y,$m,$w,$d,$h,$mn,$s) = @{ $$self{'data'}{'delta'} };
|
|
Packit |
95306a |
if ($y || $m) {
|
|
Packit |
95306a |
$from = 'approx';
|
|
Packit |
95306a |
} elsif ($w || (! $$self{'data'}{'business'} && $d)) {
|
|
Packit |
95306a |
$from = 'semi';
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
$from = 'exact';
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my $business = $$self{'data'}{'business'};
|
|
Packit |
95306a |
|
|
Packit |
95306a |
#
|
|
Packit |
95306a |
# Do the conversion
|
|
Packit |
95306a |
#
|
|
Packit |
95306a |
|
|
Packit |
95306a |
{
|
|
Packit |
95306a |
no integer;
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my $dmb = $self->base();
|
|
Packit |
95306a |
my $yl = $$dmb{'data'}{'len'}{$business}{'yl'};
|
|
Packit |
95306a |
my $ml = $$dmb{'data'}{'len'}{$business}{'ml'};
|
|
Packit |
95306a |
my $wl = $$dmb{'data'}{'len'}{$business}{'wl'};
|
|
Packit |
95306a |
my $dl = $$dmb{'data'}{'len'}{$business}{'dl'};
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# Convert it to seconds
|
|
Packit |
95306a |
|
|
Packit |
95306a |
$s += $y*$yl + $m*$ml + $w*$wl + $d*$dl + $h*3600 + $mn*60;
|
|
Packit |
95306a |
($y,$m,$w,$d,$h,$mn) = (0,0,0,0,0,0);
|
|
Packit |
95306a |
|
|
Packit |
95306a |
# Convert it to $to
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if ($to eq 'approx') {
|
|
Packit |
95306a |
# Figure out how many months there are
|
|
Packit |
95306a |
$m = int($s/$ml);
|
|
Packit |
95306a |
$s -= $m*$ml;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if ($to eq 'approx' || $to eq 'semi') {
|
|
Packit |
95306a |
if ($business) {
|
|
Packit |
95306a |
$w = int($s/$wl);
|
|
Packit |
95306a |
$s -= $w*$wl;
|
|
Packit |
95306a |
} else {
|
|
Packit |
95306a |
$d = int($s/$dl);
|
|
Packit |
95306a |
$s -= $d*$dl;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
$s = int($s);
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
$self->set('delta',[$y,$m,$w,$d,$h,$mn,$s]);
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
sub cmp {
|
|
Packit |
95306a |
my($self,$delta) = @_;
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if ($$self{'err'}) {
|
|
Packit |
95306a |
warn "WARNING: [cmp] Arguments must be valid deltas: delta1\n";
|
|
Packit |
95306a |
return undef;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if (! ref($delta) eq 'Date::Manip::Delta') {
|
|
Packit |
95306a |
warn "WARNING: [cmp] Argument must be a Date::Manip::Delta object\n";
|
|
Packit |
95306a |
return undef;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
if ($$delta{'err'}) {
|
|
Packit |
95306a |
warn "WARNING: [cmp] Arguments must be valid deltas: delta2\n";
|
|
Packit |
95306a |
return undef;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if ($$self{'data'}{'business'} != $$delta{'data'}{'business'}) {
|
|
Packit |
95306a |
warn "WARNING: [cmp] Deltas must both be business or standard\n";
|
|
Packit |
95306a |
return undef;
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
my $business = $$self{'data'}{'business'};
|
|
Packit |
95306a |
my $dmb = $self->base();
|
|
Packit |
95306a |
my $yl = $$dmb{'data'}{'len'}{$business}{'yl'};
|
|
Packit |
95306a |
my $ml = $$dmb{'data'}{'len'}{$business}{'ml'};
|
|
Packit |
95306a |
my $wl = $$dmb{'data'}{'len'}{$business}{'wl'};
|
|
Packit |
95306a |
my $dl = $$dmb{'data'}{'len'}{$business}{'dl'};
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if ($$self{'data'}{'length'} eq 'unknown') {
|
|
Packit |
95306a |
my($y,$m,$w,$d,$h,$mn,$s) = @{ $$self{'data'}{'delta'} };
|
|
Packit |
95306a |
|
|
Packit |
95306a |
no integer;
|
|
Packit |
95306a |
$$self{'data'}{'length'} = int($y*$yl + $m*$ml + $w*$wl +
|
|
Packit |
95306a |
$d*$dl + $h*3600 + $mn*60 + $s);
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
if ($$delta{'data'}{'length'} eq 'unknown') {
|
|
Packit |
95306a |
my($y,$m,$w,$d,$h,$mn,$s) = @{ $$delta{'data'}{'delta'} };
|
|
Packit |
95306a |
|
|
Packit |
95306a |
no integer;
|
|
Packit |
95306a |
$$delta{'data'}{'length'} = int($y*$yl + $m*$ml + $w*$wl +
|
|
Packit |
95306a |
$d*$dl + $h*3600 + $mn*60 + $s);
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
return ($$self{'data'}{'length'} <=> $$delta{'data'}{'length'});
|
|
Packit |
95306a |
}
|
|
Packit |
95306a |
|
|
Packit |
95306a |
1;
|
|
Packit |
95306a |
# Local Variables:
|
|
Packit |
95306a |
# mode: cperl
|
|
Packit |
95306a |
# indent-tabs-mode: nil
|
|
Packit |
95306a |
# cperl-indent-level: 3
|
|
Packit |
95306a |
# cperl-continued-statement-offset: 2
|
|
Packit |
95306a |
# cperl-continued-brace-offset: 0
|
|
Packit |
95306a |
# cperl-brace-offset: 0
|
|
Packit |
95306a |
# cperl-brace-imaginary-offset: 0
|
|
Packit |
95306a |
# cperl-label-offset: 0
|
|
Packit |
95306a |
# End:
|