# desktop/dbus -- lintian check script, vaguely based on apache2 -*- perl -*- # # Copyright (C) 2012 Arno Toell # Copyright (C) 2014 Collabora Ltd. # Copyright (C) 2021 Felix Lechner # # 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 https://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::Desktop::Dbus; use v5.20; use warnings; use utf8; use Const::Fast; use List::UtilsBy qw(uniq_by); const my $EMPTY => q{}; use Moo; use namespace::clean; with 'Lintian::Check'; sub installable { my ($self) = @_; my $index = $self->processable->installed; my @files; for my $prefix (qw(etc/dbus-1 usr/share/dbus-1)) { for my $suffix (qw(session system)) { my $folder = $index->resolve_path("${prefix}/${suffix}.d"); next unless defined $folder; push(@files, $folder->children); } } my @unique = uniq_by { $_->name } @files; $self->check_policy($_) for @unique; if (my $folder= $index->resolve_path('usr/share/dbus-1/services')) { $self->check_service($_, session => 1) for $folder->children; } if (my $folder= $index->resolve_path('usr/share/dbus-1/system-services')) { $self->check_service($_) for $folder->children; } return; } my $PROPERTIES = 'org.freedesktop.DBus.Properties'; sub check_policy { my ($self, $item) = @_; $self->pointed_hint('dbus-policy-in-etc', $item->pointer) if $item->name =~ m{^etc/}; my $xml = $item->decoded_utf8; return unless length $xml; # Parsing XML via regexes is evil, but good enough here... # note that we are parsing the entire file as one big string, # so that we catch or whatever. my @rules; # a small rubbish state machine: we want to match a containing # any or rule that is about sending my $policy = $EMPTY; while ($xml =~ m{(]*>)|()|(<(?:allow|deny)[^>]*>)}sg) { if (defined $1) { $policy = $1; } elsif (defined $2) { $policy = $EMPTY; } else { push(@rules, $policy.$3); } } my $position = 1; for my $rule (@rules) { # normalize whitespace a bit so we can report it sensibly: # typically it will now look like # $rule =~ s{\s+}{ }g; if ($rule =~ m{send_} && $rule !~ m{send_destination=}) { # It is about sending but does not specify a send-destination. # This could be bad. if ($rule =~ m{[^>]*user=['"]root['"].*pointed_hint('dbus-policy-without-send-destination', $item->pointer($position), $rule); if ( $rule =~ m{send_interface=} && $rule !~ m{send_interface=['"]\Q${PROPERTIES}\E['"]}) { # That's undesirable, because it opens up communication # with arbitrary services and can undo DoS mitigation # efforts; but at least it's specific to an interface # other than o.fd.DBus.Properties, so all that should # happen is that the service sends back an error message. # # Properties doesn't count as an effective limitation, # because it's a sort of meta-interface. } elsif ($rule =~ m{pointed_hint('dbus-policy-excessively-broad', $item->pointer($position), $rule); } } } $self->pointed_hint('dbus-policy-at-console', $item->pointer($position), $rule) if $rule =~ m{at_console=['"]true}; } continue { ++$position; } return; } sub check_service { my ($self, $item, %kwargs) = @_; my $text = $item->decoded_utf8; return unless length $text; while ($text =~ m{^Name=(.*)$}gm) { my $name = $1; next if $item->basename eq "${name}.service"; if ($kwargs{session}) { $self->pointed_hint('dbus-session-service-wrong-name', $item->pointer,"better: ${name}.service"); } else { $self->pointed_hint('dbus-system-service-wrong-name', $item->pointer, "better: ${name}.service"); } } return; } 1; # Local Variables: # indent-tabs-mode: nil # cperl-indent-level: 4 # End: # vim: syntax=perl sw=4 sts=4 sr et