]> Sergey Matveev's repositories - public-inbox.git/commitdiff
nntp: support YYYYMMDD dates for commands
authorEric Wong <e@80x24.org>
Tue, 22 Sep 2015 10:09:43 +0000 (10:09 +0000)
committerEric Wong <e@80x24.org>
Tue, 22 Sep 2015 22:12:16 +0000 (22:12 +0000)
RFC 3977 supports YYYYMMDD dates while retaining backwards
compatibility for old YYMMDD dates.

lib/PublicInbox/NNTP.pm
t/nntp.t

index 91b10f20ac39894a561ecb313cc938a36b3e3c2b..bc8d6eda63b972b781ce2e05b897e4c38be294a8 100644 (file)
@@ -169,22 +169,26 @@ sub cmd_listgroup ($;$) {
 sub parse_time ($$;$) {
        my ($date, $time, $gmt) = @_;
        use Time::Local qw();
-       my ($YY, $MM, $DD) = unpack('A2A2A2', $date);
        my ($hh, $mm, $ss) = unpack('A2A2A2', $time);
        if (defined $gmt) {
                $gmt =~ /\A(?:UTC|GMT)\z/i or die "GM invalid: $gmt\n";
                $gmt = 1;
        }
        my @now = $gmt ? gmtime : localtime;
-       if ($YY > strftime('%y', @now)) {
-               my $cur_year = $now[5] + 1900;
-               $YY += int($cur_year / 1000) * 1000 - 100;
+       my ($YYYY, $MM, $DD);
+       if (length($date) == 8) { # RFC 3977 allows YYYYMMDD
+               ($YYYY, $MM, $DD) = unpack('A4A2A2', $date);
+       } else { # legacy clients send YYMMDD
+               ($YYYY, $MM, $DD) = unpack('A2A2A2', $date);
+               if ($YYYY > strftime('%y', @now)) {
+                       my $cur_year = $now[5] + 1900;
+                       $YYYY += int($cur_year / 1000) * 1000 - 100;
+               }
        }
-
        if ($gmt) {
-               Time::Local::timegm($ss, $mm, $hh, $DD, $MM - 1, $YY);
+               Time::Local::timegm($ss, $mm, $hh, $DD, $MM - 1, $YYYY);
        } else {
-               Time::Local::timelocal($ss, $mm, $hh, $DD, $MM - 1, $YY);
+               Time::Local::timelocal($ss, $mm, $hh, $DD, $MM - 1, $YYYY);
        }
 }
 
index 82918ff55456581964017b9bad2130c33f5021ff..c91788820d8fbeb5ffc08091da5a5ffc0c912322 100644 (file)
--- a/t/nntp.t
+++ b/t/nntp.t
@@ -61,4 +61,33 @@ use_ok 'PublicInbox::NNTP';
        ngpat_like('a.s.r', 'a.t,a.s.*');
 }
 
+{
+       use POSIX qw(strftime);
+       sub time_roundtrip {
+               my ($date, $time, $gmt) = @_;
+               my $m = join(' ', @_);
+               my $ts = PublicInbox::NNTP::parse_time(@_);
+               my @t = gmtime($ts);
+               my ($d, $t);
+               if (length($date) == 8) {
+                       ($d, $t) = split(' ', strftime('%Y%m%d %H%M%S', @t));
+               } else {
+                       ($d, $t) = split(' ', strftime('%g%m%d %H%M%S', @t));
+               }
+               is_deeply([$d, $t], [$date, $time], "roundtripped: $m");
+               $ts;
+       }
+       my $x1 = time_roundtrip(qw(20141109 060606 GMT));
+       my $x2 = time_roundtrip(qw(141109 060606 GMT));
+       my $x3 = time_roundtrip(qw(930724 060606 GMT));
+
+       SKIP: {
+               skip('YYMMDD test needs updating', 2) if (time > 0x7fffffff);
+               # our world probably ends in 2038, but if not we'll try to
+               # remember to update the test then
+               is($x1, $x2, 'YYYYMMDD and YYMMDD parse identically');
+               is(strftime('%Y', gmtime($x3)), '1993', '930724 was in 1993');
+       }
+}
+
 done_testing();