# Copyright © 2012 Niels Thykier <niels@thykier.net>
# Copyright © 2019-2020 Felix Lechner <felix.lechner@lease-up.com>
# Copyright © 2017-2018 Chris Lamb <lamby@debian.org>
#
# 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, you can find it on the World Wide
# Web at http://www.gnu.org/copyleft/gpl.html, or write to the Free
# Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston,
# MA 02110-1301, USA.

package Lintian::Check;

use v5.20;
use warnings;
use utf8;

use Carp;
use Const::Fast;
use Unicode::UTF8 qw(encode_utf8);

use Moo::Role;
use namespace::clean;

const my $EMPTY => q{};
const my $SLASH => q{/};
const my $DOT => q{.};
const my $UNDERSCORE => q{_};

=head1 NAME

Lintian::Check -- Common facilities for Lintian checks

=head1 SYNOPSIS

 use Moo;
 use namespace::clean;

 with('Lintian::Check');

=head1 DESCRIPTION

A class for collecting Lintian tags as they are issued

=head1 INSTANCE METHODS

=over 4

=item name

=item processable

Get processable underlying this check.

=item group

Get group that the processable is in.

=item profile

=cut

has name => (
    is => 'rw',
    coerce => sub { my ($string) = @_; return $string // $EMPTY;},
    default => $EMPTY
);

has processable => (is => 'rw', default => sub { {} });
has group => (is => 'rw', default => sub { {} });
has profile => (is => 'rw');

=item visit_files

=cut

sub visit_files {
    my ($self, $index) = @_;

    my $visit_hook = 'visit' . $UNDERSCORE . $index . $UNDERSCORE . 'files';

    return
      unless $self->can($visit_hook);

    my @items = @{$self->processable->$index->sorted_list};

    # do not look inside quilt directory
    @items = grep { $_->name !~ m{^\.pc/} } @items
      if $index eq 'patched';

    # exclude Lintian's test suite from source scans
    @items = grep { $_->name !~ m{^t/} } @items
      if $self->processable->name eq 'lintian' && $index eq 'patched';

    $self->$visit_hook($_) for @items;

    return;
}

=item run

Run the check.

=cut

sub run {
    my ($self) = @_;

    my $type = $self->processable->type;

    if ($type eq 'source') {

        $self->visit_files('orig');
        $self->visit_files('patched');
    }

    if ($type eq 'binary' || $type eq 'udeb') {

        $self->visit_files('control');
        $self->visit_files('installed');

        $self->installable
          if $self->can('installable');
    }

    $self->$type
      if $self->can($type);

    $self->always
      if $self->can('always');

    return;
}

=item find_tag

=cut

sub find_tag {
    my ($self, $tagname) = @_;

    croak encode_utf8('No tag name')
      unless length $tagname;

    # try local name space
    my $tag = $self->profile->get_tag($self->name . $SLASH . $tagname);

    warn encode_utf8("Using tag $tagname as name spaced in "
          . $self->name
          . ' while not so declared.')
      if defined $tag && !$tag->name_spaced;

    # try global name space
    $tag ||= $self->profile->get_tag($tagname);

    unless (defined $tag) {

        warn encode_utf8(
            "Unknown tag $tagname in check " . $self->name . $DOT);
        return undef;
    }

    unless ($tag->check eq $self->name) {

        warn encode_utf8('Check '
              . $self->name
              . " has no tag $tagname (but "
              . $tag->check
              . ' does).');
        return undef;
    }

    return $tag;
}

=item pointed_hint

=cut

sub pointed_hint {
    my ($self, $tagname, $pointer, @context) = @_;

    my $tag = $self->find_tag($tagname);
    return undef
      unless defined $tag;

    # pull name from tag; could be name-spaced
    return $self->processable->pointed_hint($tag->name, $pointer, @context);
}

=item hint

=cut

sub hint {
    my ($self, $tagname, @context) = @_;

    my $tag = $self->find_tag($tagname);
    return undef
      unless defined $tag;

    # pull name from tag; could be name-spaced
    return $self->processable->hint($tag->name, @context);
}

=back

=head1 AUTHOR

Originally written by Felix Lechner <felix.lechner@lease-up.com> for Lintian.

=head1 SEE ALSO

lintian(1)

=cut

1;

# Local Variables:
# indent-tabs-mode: nil
# cperl-indent-level: 4
# End:
# vim: syntax=perl sw=4 sts=4 sr et
