#!/usr/bin/perl
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program.  If not, see <https://www.gnu.org/licenses/>.

use strict;
use warnings;

use Test::More tests => 102;
use Test::Dpkg qw(:paths);

use File::Basename;

use Dpkg::File;

BEGIN {
    use_ok('Dpkg::Changelog');
    use_ok('Dpkg::Changelog::Debian');
    use_ok('Dpkg::Vendor', qw(get_current_vendor));
};

my $datadir = test_get_data_path();

my $vendor = get_current_vendor();

sub setup_changelog
{
    my $name = shift;
    my $file = "$datadir/$name";

    my $changes = Dpkg::Changelog::Debian->new(verbose => 0);
    $changes->load($file);

    my $content = file_slurp($file);
    cmp_ok($content, 'eq', "$changes", "string output of Dpkg::Changelog on $file");

    my $errors = $changes->get_parse_errors();
    my $basename = basename($file);
    is($errors, '', "Parse example changelog $file without errors");

    my @data = @$changes;
    ok(@data, "data from $file is not empty");

    SKIP: {
        skip('avoid spurious warning with only one entry', 2)
            if @data == 1;

        my $str;
        my $oldest_version = $data[-1]->{Version};
        $str = $changes->format_range('dpkg', { since => $oldest_version });

        $str = $changes->format_range('rfc822');

        ok(1, 'TODO check rfc822 output');

        $str = $changes->format_range('rfc822', { since => $oldest_version });

        ok(1, 'TODO check rfc822 output with ranges');
    }

    return ($changes, \@data);
}

#########################

my $changes;
my $data;


## File: countme

($changes, $data) = setup_changelog('countme');

# Test range options.
cmp_ok(@{$data}, '==', 7, 'no options -> count');
my $all_versions = join('/', map { $_->get_version() } @{$data});

sub check_options {
    my (%opts) = @_;

    my @cnt = $changes->get_range($opts{range});
    cmp_ok(@cnt, '==', $opts{count}, "$opts{name} -> count");
    if ($opts{count} == @{$opts{data}}) {
        is_deeply(\@cnt, $opts{data}, "$opts{name} -> returns all");
    } else {
        is_deeply([ map { $_->get_version() } @cnt ],
                  $opts{versions}, "$opts{name} -> versions");
    }
}

my %ref = (
    changes => $changes,
    data => $data,
);

check_options(%ref, range => { count => 3 },
              count => 3,
              versions => [ '2:2.0-1', '1:2.0~rc2-3', '1:2.0~rc2-2' ],
              name => 'positive count');
check_options(%ref, range => { count => 3, reverse => 1 },
              count => 3,
              versions => [ '1:2.0~rc2-2', '1:2.0~rc2-3', '2:2.0-1' ],
              name => 'positive reverse count');
check_options(%ref, range => { count => -3 },
              count => 3,
              versions => [
                    '1:2.0~rc2-1sarge2',
                    '1:2.0~rc2-1sarge1',
                    '1.5-1',
              ],
              name => 'negative count');
check_options(%ref, range => { count => 1 },
              count => 1,
              versions => [ '2:2.0-1' ],
              name => 'count 1');
check_options(%ref, range => { count => 1, default_all => 1 },
              count => 1,
              versions => [ '2:2.0-1' ],
              name => 'count 1 (d_a 1)');
check_options(%ref, range => { count => -1 },
              count => 1,
              versions => [ '1.5-1' ],
              name => 'count -1');

check_options(%ref, range => { count => 3, offset => 2 },
              count => 3,
              versions => [
                    '1:2.0~rc2-2',
                    '1:2.0~rc2-1sarge3',
                    '1:2.0~rc2-1sarge2',
              ],
              name => 'positive count + positive offset');
check_options(%ref, range => { count => -3, offset => 4 },
              count => 3,
              versions => [
                    '1:2.0~rc2-3',
                    '1:2.0~rc2-2',
                    '1:2.0~rc2-1sarge3',
              ],
              name => 'negative count + positive offset');

check_options(%ref, range => { count => 4, offset => 5 },
              count => 2,
              versions => [ '1:2.0~rc2-1sarge1', '1.5-1' ],
              name => 'positive count + positive offset (>max)');
check_options(%ref, range =>  { count => -4, offset => 2 },
              count => 2,
              versions => [ '2:2.0-1', '1:2.0~rc2-3' ],
              name => 'negative count + positive offset (<0)');

check_options(%ref, range => { count => 3, offset => -4 },
              count => 3,
              versions => [
                '1:2.0~rc2-1sarge3',
                '1:2.0~rc2-1sarge2',
                '1:2.0~rc2-1sarge1',
              ],
              name => 'positive count + negative offset');
check_options(%ref, range => { count => -3, offset => -3 },
              count => 3,
              versions => [
                  '1:2.0~rc2-3',
                  '1:2.0~rc2-2',
                  '1:2.0~rc2-1sarge3',
              ],
              name => 'negative count + negative offset');

check_options(%ref, range => { count => 5, offset => -2 },
              count => 2,
              versions => [ '1:2.0~rc2-1sarge1', '1.5-1' ],
              name => 'positive count + negative offset (>max)');
check_options(%ref, range => { count => -5, offset => -4 },
              count => 3,
              versions => [ '2:2.0-1', '1:2.0~rc2-3', '1:2.0~rc2-2' ],
              name => 'negative count + negative offset (<0)');

check_options(%ref, range => { count => 7 },
              count => 7,
              name => 'count 7 (max)');
check_options(%ref, range => { count => -7 },
              count => 7,
              name => 'count -7 (-max)');
check_options(%ref, range => { count => 10 },
              count => 7,
              name => 'count 10 (>max)');
check_options(%ref, range => { count => -10 },
              count => 7,
              name => 'count -10 (<-max)');

check_options(%ref, range => { from => '1:2.0~rc2-1sarge3' },
              count => 4,
              versions => [
                  '2:2.0-1',
                  '1:2.0~rc2-3',
                  '1:2.0~rc2-2',
                  '1:2.0~rc2-1sarge3',
              ],
              name => 'from => "1:2.0~rc2-1sarge3"');
check_options(%ref, range => { since => '1:2.0~rc2-1sarge3' },
              count => 3,
              versions => [
                    '2:2.0-1',
                    '1:2.0~rc2-3',
                    '1:2.0~rc2-2',
              ],
              name => 'since => "1:2.0~rc2-1sarge3"');
$SIG{__WARN__} = sub {};
check_options(%ref, range => { since => 0 },
              count => 7,
              name => 'since => 0 returns all');
delete $SIG{__WARN__};
check_options(%ref, range => { to => '1:2.0~rc2-1sarge2' },
              count => 3,
              versions => [
                    '1:2.0~rc2-1sarge2',
                    '1:2.0~rc2-1sarge1',
                    '1.5-1',
              ],
              name => 'to => "1:2.0~rc2-1sarge2"');
## no critic (ControlStructures::ProhibitUntilBlocks)
check_options(%ref, range => { until => '1:2.0~rc2-1sarge2' },
              count => 2,
              versions => [ '1:2.0~rc2-1sarge1', '1.5-1' ],
              name => 'until => "1:2.0~rc2-1sarge2"');
## use critic
#TODO: test combinations


## File: shadow

($changes, $data) = setup_changelog('shadow');


## File: fields

($changes, $data) = setup_changelog('fields');

    my $str = $changes->format_range('dpkg', { all => 1 });
    my $expected = 'Source: fields
Version: 2.0-0etch1
Distribution: stable
Urgency: high
Maintainer: Frank Lichtenheld <frank@lichtenheld.de>
Timestamp: 1200235759
Date: Sun, 13 Jan 2008 15:49:19 +0100
Closes: 1000000 1111111 2222222
Changes:
 fields (2.0-0etch1) stable; urgency=low
 .
   * Upload to stable (Closes: #1111111, #2222222)
   * Fix more stuff. (LP: #54321, #2424242)
 .
 fields (2.0-1) unstable  frozen; urgency=medium
 .
   [ Frank Lichtenheld ]
   * Upload to unstable (Closes: #1111111, #2222222)
   * Fix stuff. (LP: #12345, #424242)
 .
   [ Raphaël Hertzog ]
   * New upstream release.
     - implements a
     - implements b
   * Update S-V.
 .
 fields (2.0~b1-1) unstable; urgency=low,xc-userfield=foobar
 .
   * Beta
 .
 fields (1.0) experimental; urgency=high,xb-userfield2=foobar
 .
   * First upload (Closes: #1000000)
Xb-Userfield2: foobar
Xc-Userfield: foobar

';
    if ($vendor eq 'Ubuntu') {
        $expected =~ s/^(Closes:.*)/$1\nLaunchpad-Bugs-Fixed: 12345 54321 424242 2424242/m;
    }
    cmp_ok($str, 'eq', $expected, 'fields handling');

    $str = $changes->format_range('dpkg', { offset => 1, count => 2 });
    $expected = 'Source: fields
Version: 2.0-1
Distribution: unstable frozen
Urgency: medium
Maintainer: Frank Lichtenheld <djpig@debian.org>
Timestamp: 1200149359
Date: Sun, 12 Jan 2008 15:49:19 +0100
Closes: 1111111 2222222
Changes:
 fields (2.0-1) unstable  frozen; urgency=medium
 .
   [ Frank Lichtenheld ]
   * Upload to unstable (Closes: #1111111, #2222222)
   * Fix stuff. (LP: #12345, #424242)
 .
   [ Raphaël Hertzog ]
   * New upstream release.
     - implements a
     - implements b
   * Update S-V.
 .
 fields (2.0~b1-1) unstable; urgency=low,xc-userfield=foobar
 .
   * Beta
Xc-Userfield: foobar

';
    if ($vendor eq 'Ubuntu') {
        $expected =~ s/^(Closes:.*)/$1\nLaunchpad-Bugs-Fixed: 12345 424242/m;
    }
    cmp_ok($str, 'eq', $expected, 'fields handling 2');

    $str = $changes->format_range('rfc822', { offset => 2, count => 2 });
    $expected = 'Source: fields
Version: 2.0~b1-1
Distribution: unstable
Urgency: low
Maintainer: Frank Lichtenheld <frank@lichtenheld.de>
Timestamp: 1200062959
Date: Sun, 11 Jan 2008 15:49:19 +0100
Changes:
 fields (2.0~b1-1) unstable; urgency=low,xc-userfield=foobar
 .
   * Beta
Xc-Userfield: foobar

Source: fields
Version: 1.0
Distribution: experimental
Urgency: high
Maintainer: Frank Lichtenheld <djpig@debian.org>
Timestamp: 1199976559
Date: Sun, 10 Jan 2008 15:49:19 +0100
Closes: 1000000
Changes:
 fields (1.0) experimental; urgency=high,xb-userfield2=foobar
 .
   * First upload (Closes: #1000000)
Xb-Userfield2: foobar

';
    cmp_ok($str, 'eq', $expected, 'fields handling 3');

    # Test Dpkg::Changelog::Entry methods
    is($data->[1]->get_version(), '2.0-1', 'get_version');
    is($data->[1]->get_source(), 'fields', 'get_source');
    is(scalar $data->[1]->get_distributions(), 'unstable', 'get_distribution');
    is(join('|', $data->[1]->get_distributions()), 'unstable|frozen',
        'get_distributions');
    is($data->[3]->get_optional_fields(),
        "Urgency: high\nCloses: 1000000\nXb-Userfield2: foobar\n",
        'get_optional_fields');
    is($data->[1]->get_maintainer(), 'Frank Lichtenheld <djpig@debian.org>',
        'get_maintainer');
    is($data->[1]->get_timestamp(), 'Sun, 12 Jan 2008 15:49:19 +0100',
        'get_timestamp');
    my @items = $data->[1]->get_change_items();
    is($items[0], "  [ Frank Lichtenheld ]\n", 'change items 1');
    is($items[4], '  * New upstream release.
    - implements a
    - implements b
', 'change items 2');
    is($items[5], "  * Update S-V.\n", 'change items 3');


## File: date-format

($changes, $data) = setup_changelog('date-format');

is($data->[0]->get_timestamp(), '01 Jul 2100 23:59:59 -1200',
    'get date w/o DoW, and negative timezone offset');
is($data->[1]->get_timestamp(), 'Tue, 27 Feb 2050 12:00:00 +1245',
    'get date w/ DoW, and positive timezone offset');
is($data->[2]->get_timestamp(), 'Mon, 01 Jan 2000 00:00:00 +0000',
    'get date w/ DoW, and zero timezone offset');


## File: stop-modeline

($changes, $data) = setup_changelog('stop-modeline');

is($changes->get_unparsed_tail(), "vim: et\n",
    'get unparsed modeline at EOF');


## File: regressions

($changes, $data) = setup_changelog('regressions');

my $f = ($changes->format_range('dpkg'))[0];
is("$f->{Version}", '0', 'version 0 correctly parsed');


## Files

foreach my $test (([ "$datadir/misplaced-tz", 6 ],
                   [ "$datadir/unreleased", 5, 7 ])) {

    my $file = shift @$test;
    my $changes = Dpkg::Changelog::Debian->new(verbose => 0);
    $changes->load($file);
    my @errors = $changes->get_parse_errors();

    ok(@errors, 'errors occured');
    is_deeply([ map { $_->[1] } @errors ], $test, 'check line numbers');
}