aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorPascal Rigaux <pixel@mandriva.com>2007-06-29 13:30:24 +0000
committerPascal Rigaux <pixel@mandriva.com>2007-06-29 13:30:24 +0000
commit6cbb51a6bf5f4ccfc3388ce39a3bbd5b14a91217 (patch)
tree7e5723a41d828799e9759547bfd72c0133ba297f
downloadmga-youri-submit-6cbb51a6bf5f4ccfc3388ce39a3bbd5b14a91217.tar
mga-youri-submit-6cbb51a6bf5f4ccfc3388ce39a3bbd5b14a91217.tar.gz
mga-youri-submit-6cbb51a6bf5f4ccfc3388ce39a3bbd5b14a91217.tar.bz2
mga-youri-submit-6cbb51a6bf5f4ccfc3388ce39a3bbd5b14a91217.tar.xz
mga-youri-submit-6cbb51a6bf5f4ccfc3388ce39a3bbd5b14a91217.zip
prepare mergeupstream
-rw-r--r--ChangeLog17
-rw-r--r--Makefile.am8
-rw-r--r--README41
-rw-r--r--TODO8
-rw-r--r--bin/Makefile.am38
-rwxr-xr-xbin/youri-submit-proxy.in82
-rwxr-xr-xbin/youri-submit-restricted.in69
-rwxr-xr-xbin/youri-submit.in301
-rw-r--r--configure.ac19
-rw-r--r--etc/Makefile.am5
-rw-r--r--etc/submit.conf131
-rw-r--r--etc/youri-submit60
-rw-r--r--lib/Makefile.am34
-rw-r--r--lib/Youri/Submit/Action/Archive.pm73
-rw-r--r--lib/Youri/Submit/Action/Bugzilla.pm78
-rw-r--r--lib/Youri/Submit/Action/CVS.pm136
-rw-r--r--lib/Youri/Submit/Action/Clean.pm42
-rw-r--r--lib/Youri/Submit/Action/Install.pm74
-rw-r--r--lib/Youri/Submit/Action/Link.pm79
-rw-r--r--lib/Youri/Submit/Action/Mail.pm143
-rw-r--r--lib/Youri/Submit/Action/RSS.pm117
-rw-r--r--lib/Youri/Submit/Action/Reorder.pm37
-rw-r--r--lib/Youri/Submit/Action/Sign.pm60
-rw-r--r--lib/Youri/Submit/Step.pm125
-rw-r--r--lib/Youri/Submit/Test/History.pm65
-rw-r--r--lib/Youri/Submit/Test/Precedence.pm60
-rw-r--r--lib/Youri/Submit/Test/Recency.pm50
-rw-r--r--lib/Youri/Submit/Test/Rpmlint.pm97
-rw-r--r--lib/Youri/Submit/Test/Set.pm79
-rw-r--r--lib/Youri/Submit/Test/Tag.pm58
-rw-r--r--t/Makefile.am8
-rw-r--r--t/gpghome/pubring.gpgbin0 -> 565 bytes
-rw-r--r--t/gpghome/random_seedbin0 -> 600 bytes
-rw-r--r--t/gpghome/secring.gpgbin0 -> 628 bytes
-rw-r--r--t/gpghome/trustdb.gpgbin0 -> 1280 bytes
-rwxr-xr-xt/install.t51
-rwxr-xr-xt/perlcritic.t32
-rwxr-xr-xt/set.t124
-rwxr-xr-xt/sign.t57
-rwxr-xr-xt/tags.t55
40 files changed, 2513 insertions, 0 deletions
diff --git a/ChangeLog b/ChangeLog
new file mode 100644
index 0000000..a0bdf33
--- /dev/null
+++ b/ChangeLog
@@ -0,0 +1,17 @@
+2007-04-22 Guillaume Rousse <guillomovitch@zarb.org> 0.10
+ * internal changes:
+ - large code cleanup
+ - namespace changes:
+ . Youri::Upload::Check -> Youri::Submit::Test
+ . Youri::Upload::Action -> Youri::Submit::Action
+ * installation:
+ - autotools based installation system
+ - modules are now installed under private directory
+ * YAML-based configuration
+ * binaries:
+ - youri-upload is now youri-submit
+ - new wrappers youri-submit-proxy and youri-submit-restricted
+ - --list option for runtime configuration
+
+2006-04-23 Guillaume Rousse <guillomovitch@zarb.org> 0.9
+ * initial release
diff --git a/Makefile.am b/Makefile.am
new file mode 100644
index 0000000..6d69457
--- /dev/null
+++ b/Makefile.am
@@ -0,0 +1,8 @@
+SVNPATH = svn+ssh://youri.zarb.org/home/projects/youri/svn/soft/submit
+
+SUBDIRS = bin lib etc t
+
+svntag:
+ svn copy -m 'new release $(VERSION)'\
+ $(SVNPATH)/trunk \
+ $(SVNPATH)/tags/release-$(VERSION)
diff --git a/README b/README
new file mode 100644
index 0000000..3c84b66
--- /dev/null
+++ b/README
@@ -0,0 +1,41 @@
+youri-submit
+------------
+
+YOURI stands for "Youri Offers an Upload & Repository Infrastucture". It aims
+to build tools making management of a coherent set of packages easier.
+
+youri-submit is a generic package submission tool. It first runs a list of
+tests on each submitted package, and if no one fails, runs a list of actions on
+those packages.
+
+youri-submit-restricted and youri-submit-proxy are additional wrappers intended
+for collaborative work, allowing to run submission process with restricted
+permissions and under another identity.
+
+Dependencies
+------------
+The following perl modules are required:
+- Youri::Config
+- Youri::Repository
+- Youri::Package
+- Youri::Utils
+
+Depending on exact list of tests and actions, other modules may be needed also.
+
+Installation
+------------
+To install, just use:
+./configure
+make
+make install
+
+Copyright and License
+---------------------
+Copyright (C) 2002-2006, YOURI project
+
+This program is free software; you can redistribute it and/or modify it under
+the same terms as Perl itself.
+
+Authors
+-------
+Guillaume Rousse <guillomovitch@zarb.org>,
diff --git a/TODO b/TODO
new file mode 100644
index 0000000..da66ff6
--- /dev/null
+++ b/TODO
@@ -0,0 +1,8 @@
+1.0 Goals
+=========
+- svn action plugin
+- change plugin granularity to handle a set of packages
+- drop arbitrary distinction between tests and actions
+- make plugins able to modify their input
+- automatic bugzilla ticket closing on upload
+- more customizable (template based ?) mail notification
diff --git a/bin/Makefile.am b/bin/Makefile.am
new file mode 100644
index 0000000..41dcd32
--- /dev/null
+++ b/bin/Makefile.am
@@ -0,0 +1,38 @@
+bin_SCRIPTS = youri-submit youri-submit-restricted youri-submit-proxy
+man1_MANS = youri-submit.1 youri-submit-restricted.1 youri-submit-proxy.1
+
+CLEANFILES = $(bin_SCRIPTS) $(man1_MANS)
+EXTRA_DIST = youri-submit.in youri-submit-restricted.in youri-submit-proxy.in
+
+youri-submit: youri-submit.in
+ $(PERL) -pi \
+ -e 's|\@sysconfdir\@|$(sysconfdir)|;' \
+ -e 's|\@bindir\@|$(bindir)|;' \
+ -e 's|\@perllibdir\@|$(perllibdir)|;' \
+ < $< > $@
+
+youri-submit-restricted: youri-submit-restricted.in
+ $(PERL) -pi \
+ -e 's|\@sysconfdir\@|$(sysconfdir)|;' \
+ -e 's|\@bindir\@|$(bindir)|;' \
+ -e 's|\@perllibdir\@|$(perllibdir)|;' \
+ < $< > $@
+
+youri-submit-proxy: youri-submit-proxy.in
+ $(PERL) -pi \
+ -e 's|\@sysconfdir\@|$(sysconfdir)|;' \
+ -e 's|\@bindir\@|$(bindir)|;' \
+ -e 's|\@perllibdir\@|$(perllibdir)|;' \
+ < $< > $@
+
+youri-submit.1: youri-submit
+ $(POD2MAN) $< $@
+
+youri-submit-restricted.1: youri-submit-restricted
+ $(POD2MAN) $< $@
+
+youri-submit-proxy.1: youri-submit-proxy
+ $(POD2MAN) $< $@
+
+
+
diff --git a/bin/youri-submit-proxy.in b/bin/youri-submit-proxy.in
new file mode 100755
index 0000000..e563044
--- /dev/null
+++ b/bin/youri-submit-proxy.in
@@ -0,0 +1,82 @@
+#!/usr/bin/perl
+# $Id: youri-submit-proxy.in 1530 2007-03-08 20:42:13Z guillomovitch $
+
+=head1 NAME
+
+youri-submit-proxy - proxy wrapper over youri-submit-restricted
+
+=head1 VERSION
+
+Version 1.0
+
+=head1 SYNOPSIS
+
+youri-submit-proxy [options] <target> <files>
+
+=head1 DESCRIPTION
+
+youri-submit-proxy is a proxy wrapper over youri-submit-restricted, intended to
+be used in collaborative work to change uid before calling it through sudo.
+
+=head1 SEE ALSO
+
+youri-submit-restricted(1), youri-submit(1)
+
+=head1 COPYRIGHT AND LICENSE
+
+Copyright (C) 2002-2006, YOURI project
+
+This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
+
+=cut
+
+use strict;
+use warnings;
+use lib '@perllibdir@';
+
+use Fcntl ':mode';
+use File::Basename;
+
+my ($uid, $gid);
+if (-l $0) {
+ # this is a symlink, get uid and gid from it
+ ($uid, $gid) = (lstat($0))[4, 5];
+} else {
+ ($uid, $gid) = (stat($0))[4, 5];
+}
+my $user = getpwuid($uid) or die "unknown uid $uid\n";
+my $prog = '@bindir@/youri-submit-restricted';
+
+my %dirs;
+my @options;
+foreach my $arg (@ARGV) {
+ if (-f $arg) {
+ # push parent dir in list
+ my $parent = dirname($arg);
+ $dirs{$parent}++;
+ }
+ push(@options, $arg);
+}
+
+foreach my $dir (keys %dirs) {
+ # save original perms and gid
+ my ($orig_mode, $orig_gid) = (stat($dir))[2,5];
+ $dirs{$dir} = {
+ mode => $orig_mode,
+ gid => $orig_gid
+ };
+ # ensure correct perms and gid
+ chown -1, $gid, $dir;
+ chmod $orig_mode|S_IRGRP|S_IWGRP, $dir;
+}
+
+# call wrapped program
+my $status = system('sudo', '-H', '-u', $user, $prog, @options);
+
+foreach my $dir (keys %dirs) {
+ # restore original perms and gid
+ chown -1, $dirs{$dir}->{gid}, $dir;
+ chmod $dirs{$dir}->{mode}, $dir;
+}
+
+exit($status >> 8);
diff --git a/bin/youri-submit-restricted.in b/bin/youri-submit-restricted.in
new file mode 100755
index 0000000..360faf8
--- /dev/null
+++ b/bin/youri-submit-restricted.in
@@ -0,0 +1,69 @@
+#!/usr/bin/perl -T
+# $Id: youri-submit-restricted.in 1530 2007-03-08 20:42:13Z guillomovitch $
+
+=head1 NAME
+
+youri-submit-restricted - filtering wrapper over youri-submit
+
+=head1 VERSION
+
+Version 1.0
+
+=head1 SYNOPSIS
+
+youri-submit-restricted [options] <target> <files>
+
+=head1 DESCRIPTION
+
+youri-submit-restricted is just a filtering wrapper over youri-submit, intended
+to be used in collaborative work to sanitize environment and options before
+calling it.
+
+=head1 SEE ALSO
+
+youri-submit(1)
+
+=head1 COPYRIGHT AND LICENSE
+
+Copyright (C) 2002-2006, YOURI project
+
+This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
+
+=cut
+
+use strict;
+use warnings;
+use lib '@perllibdir@';
+
+my $prog = '@bindir@/youri-submit';
+my @prohibited_options = qw/--config --skip-check --skip-action/;
+my %prohibited_options = map { $_ => 1 } @prohibited_options;
+my @prohibited_envvars = qw/
+ ENV BASH_ENV IFS CDPATH
+ PERLLIB PERL5LIB PERL5OPT PERLIO
+ PERLIO_DEBUG PERL5DB PERL_ENCODING
+ PERL_HASH_SEED PERL_SIGNALS PERL_UNICODE
+/;
+
+my @options;
+while (my $arg = shift @ARGV) {
+ if ($prohibited_options{$arg}) {
+ # drop prohibited options
+ print STDERR "prohibited option $arg, skipping\n";
+ shift @ARGV;
+ } else {
+ # untaint everything else
+ $arg =~ /(.*)/;
+ push(@options, $1);
+ }
+}
+
+# secure ENV
+$ENV{PATH} = "/sbin:/usr/sbin:/bin:/usr/bin:/usr/X11R6/bin";
+delete $ENV{$_} foreach @prohibited_envvars;
+
+# call wrapped program
+my $status = system($prog, @options);
+
+# return wrapped program original exit status
+exit($status >> 8);
diff --git a/bin/youri-submit.in b/bin/youri-submit.in
new file mode 100755
index 0000000..70e6fb8
--- /dev/null
+++ b/bin/youri-submit.in
@@ -0,0 +1,301 @@
+#!/usr/bin/perl
+# $Id: youri-submit.in 1687 2007-06-28 22:44:07Z guillomovitch $
+
+=head1 NAME
+
+youri-submit - package submission tool
+
+=head1 VERSION
+
+Version 2.0
+
+=head1 SYNOPSIS
+
+youri-submit [options] <target> <files>
+
+youri-submit --list <category> [target]
+
+youri-submit --help [category] [item]
+
+Options:
+
+ --config <file> use file <file> as config file
+ --skip-step <step> skip step <step>
+ --define <key>=<value> pass additional values
+ --clean delete package after success
+ --verbose verbose run
+ --test test run
+ --list <category> list items from given category
+ --help [category] display contextual help
+
+=head1 DESCRIPTION
+
+B<youri-submit> allows to submit packages to a repository.
+
+All packages given on command lines are passed to a list of test plugins,
+depending on given upload target. If none of them fails, all packages are
+passed to a list of action plugins, depending also on given upload target.
+
+=head1 OPTIONS
+
+=over
+
+=item B<--config> I<file>
+
+Use given file as configuration, instead of normal one.
+
+=item B<--skip-step> I<id>
+
+Skip step with given identity.
+
+=item B<--define> <key>=<value>
+
+Define additional parameters, to be used by plugins.
+
+=item B<--clean>
+
+Delete submited packages upon successfull submission.
+
+=item B<--verbose>
+
+Produce more verbose output (can be used more than once)
+
+=item B<--test>
+
+Don't perform any modification.
+
+=item B<--list> I<category>
+
+List available items from given category and exits. Category must be either
+B<targets>, B<actions> or B<tests>. A target is needed for the two last ones.
+
+=item B<--help> I<category>
+
+Display help for given category and exits. Category must be either
+B<repository>, B<action> or B<test>. An item is needed for the two last ones.
+If no category given, display standard help.
+
+=back
+
+=head1 CONFIGURATION
+
+Configuration is read from the first file found among:
+
+=over
+
+=item * the one specified by B<--config> option on command-line
+
+=item * $HOME/.youri/submit.conf
+
+=item * @sysconfdir@/youri/submit.conf
+
+=back
+
+The configuration file should be a YAML-format files, with the following
+mandatory top-level directives:
+
+=over
+
+=item B<repository>
+
+The definition of repository plugin to be used.
+
+=item B<targets>
+
+The list of available submission targets, each one being composed from the
+following keys:
+
+=over
+
+=item B<steps>
+
+The list of steps to use for this target.
+
+=back
+
+=item B<steps>
+
+The definitions of steps, indexed by their identity.
+
+=back
+
+=head1 SEE ALSO
+
+Youri::Config, for additional details about configuration file format.
+
+Each used plugin man page, for available options.
+
+=head1 COPYRIGHT AND LICENSE
+
+Copyright (C) 2002-2006, YOURI project
+
+This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
+
+=cut
+
+use strict;
+use warnings;
+use lib '@perllibdir@';
+
+use Youri::Config;
+use Youri::Utils 0.002;
+use Pod::Usage;
+
+my $config = Youri::Config->new(
+ args => {
+ 'skip-step' => '=s@',
+ 'define' => '=s%',
+ 'timestamp' => '!',
+ 'clean' => '!',
+ 'test' => '|t!',
+ 'list' => '|l!'
+ },
+ directories => [ "$ENV{HOME}/.youri", '@sysconfdir@/youri' ],
+ file => 'submit.conf',
+);
+
+if ($config->get_arg('list')) {
+ my $category = $ARGV[0];
+ pod2usage(-verbose => 0, -message => "No category specified, aborting\n")
+ unless $category;
+ if ($category eq 'targets') {
+ print join(' ', keys %{$config->get_param('targets')});
+ } elsif ($category eq 'steps') {
+ my $target = $ARGV[1];
+ pod2usage(-verbose => 0, -message => "No target specified, aborting\n")
+ unless $target;
+ my $steps = $config->get_param('targets')->{$target}->{steps};
+ print join(' ', @{$steps}) if $steps;
+ } else {
+ pod2usage(-verbose => 0, -message => "Invalid category $category, aborting\n")
+ }
+ print "\n";
+ exit 0;
+}
+
+if ($config->get_arg('help')) {
+ my $category = $ARGV[0];
+ my ($item, $section);
+ if ($category eq 'repository') {
+ $section = $config->get_param('repository');
+ pod2usage(
+ -verbose => 0,
+ -message => "No repository defined, aborting\n"
+ ) unless $section;
+ } elsif ($category eq 'step') {
+ $item = $ARGV[1];
+ pod2usage(
+ -verbose => 0,
+ -message => "No item specified, aborting\n"
+ ) unless $item;
+ $section = $config->get_param('steps')->{$item};
+ pod2usage(
+ -verbose => 0,
+ -message => "No such step $item defined, aborting\n"
+ ) unless $section;
+ } else {
+ pod2usage(-verbose => 0, -message => "Invalid category $category, aborting\n")
+ }
+ my $file = $section->{class} . '.pm';
+ $file =~ s/::/\//g;
+ pod2usage(
+ -verbose => 99,
+ -sections => 'NAME|DESCRIPTION',
+ -input => $file,
+ -pathlist => \@INC
+ );
+}
+
+
+pod2usage(-verbose => 0, -message => "No target specified, aborting\n")
+ unless @ARGV > 0;
+pod2usage(-verbose => 0, -message => "No packages specified, aborting\n")
+ unless @ARGV > 1;
+
+# convenient global flags
+my $test = $config->get_arg('test');
+my $verbose = $config->get_arg('verbose');
+my $timestamp = $config->get_arg('timestamp');
+
+# test target
+my $target = shift @ARGV;
+my $target_conf = $config->get_param('targets')->{$target};
+
+# create repository
+my $repository;
+my $repository_conf = $config->get_param('repository');
+die "No repository declared\n" unless $repository_conf;
+log_message("Creating repository", $timestamp) if $verbose;
+eval {
+ $repository = create_instance(
+ 'Youri::Repository',
+ $repository_conf,
+ {
+ test => $test,
+ verbose => $verbose > 0 ? $verbose - 1 : 0,
+ targets => [ keys %{$config->get_param('targets')} ],
+ }
+ );
+};
+die "Failed to create repository: $@\n" if $@;
+
+# create packages
+my @packages;
+foreach my $file (@ARGV) {
+ push(
+ @packages,
+ create_instance(
+ 'Youri::Package',
+ {
+ class => $repository->get_package_class(),
+ },
+ {
+ file => $file
+ }
+ )
+ );
+}
+
+# check all packages pass all tests
+my %errors;
+my $skip_step = $config->get_arg('skip-step');
+my %skip_step = $skip_step ? map { $_ => 1 } @{$skip_step} : ();
+eval {
+ foreach my $id (@{$target_conf->{steps}}) {
+ next if $skip_step{$id};
+ my $step_conf = $config->get_param('steps')->{$id};
+
+ die "Invalid configuration, step $id is not defined\n"
+ unless $step_conf;
+
+ log_message("Creating step $id", $timestamp) if $verbose;
+ my $step;
+ $test = create_instance(
+ 'Youri::Submit::Step',
+ $step_conf,
+ {
+ id => $id,
+ test => $test,
+ verbose => $verbose > 0 ? $verbose - 1 : 0,
+ }
+ );
+
+ log_message("Running step $id", $timestamp) if $verbose;
+ $step->run(
+ \@packages
+ $repository,
+ $target,
+ $config->get_arg('define'),
+ );
+ }
+};
+if ($@) {
+ die "Error during submission: $@\n";
+}
+
+if ($config->get_arg('clean')) {
+ foreach my $package (@packages) {
+ log_message("cleaning file $package", $timestamp) if $verbose;
+ unlink $package->as_file();
+ }
+}
diff --git a/configure.ac b/configure.ac
new file mode 100644
index 0000000..0170347
--- /dev/null
+++ b/configure.ac
@@ -0,0 +1,19 @@
+AC_PREREQ(2.59)
+AC_INIT([youri-submit], [0.10])
+AM_INIT_AUTOMAKE([-Wall -Werror foreign])
+
+AC_PATH_PROG([PERL],[perl])
+if test -z $PERL; then
+ AC_MSG_ERROR([perl not found, aborting])
+fi
+
+AC_PATH_PROG([POD2MAN],[pod2man])
+if test -z $POD2MAN; then
+ AC_MSG_ERROR([pod2man not found, aborting])
+fi
+
+perllibdir=$datadir/youri/lib
+AC_SUBST(perllibdir)
+
+AC_CONFIG_FILES([Makefile bin/Makefile lib/Makefile etc/Makefile t/Makefile])
+AC_OUTPUT
diff --git a/etc/Makefile.am b/etc/Makefile.am
new file mode 100644
index 0000000..3ffc811
--- /dev/null
+++ b/etc/Makefile.am
@@ -0,0 +1,5 @@
+yourisysconfdir = $(sysconfdir)/youri
+completiondir = $(sysconfdir)/bash_completion.d
+
+dist_yourisysconf_DATA = submit.conf
+dist_completion_SCRIPTS = youri-submit
diff --git a/etc/submit.conf b/etc/submit.conf
new file mode 100644
index 0000000..a7fbf16
--- /dev/null
+++ b/etc/submit.conf
@@ -0,0 +1,131 @@
+# youri-submit sample configuration file
+# $Id: submit.conf 1671 2007-06-28 22:41:51Z guillomovitch $
+# vim:ft=yaml:et:sw=4
+
+# helper variables
+home: /home/user
+
+# repository definition
+repository:
+ class: Youri::Repository::PLF
+ options:
+ install_root: ${home}/ftp/mandriva
+ version_root: ${home}/cvs
+ archive_root: ${home}/backup/mandriva
+ noarch: i586
+
+# targets definitions
+targets:
+ cooker:
+ steps:
+ - check-tag
+ - check-recency
+ - check-history
+ - do-sign
+ - do-install
+ - do-link
+ - do-archive
+ - do-clean
+ - do-bugzilla
+ - do-cvs
+ - do-mail
+ - do-rss
+
+ 2006.0:
+ steps:
+ - check-type
+ - check-tag
+ - check-recency
+ - check-history
+ - check-precedence
+ - do-sign
+ - do-install
+ - do-link
+ - do-archive
+ - do-clean
+
+# steps definitions
+steps:
+ check-tag:
+ class: Youri::Submit::Check::Tag
+ options:
+ tags:
+ release: 'plf$'
+ packager: '<\w+@zarb\.org>$'
+ distribution: '^Mandriva Linux$'
+ vendor: '^Penguin Liberation Front$'
+
+ check-recency:
+ class: Youri::Submit::Check::Recency
+
+ check-history:
+ class: Youri::Submit::Check::History
+
+ check-precedence:
+ class: Youri::Submit::Check::Precedence
+ options:
+ target: cooker
+
+ check-type:
+ class: Youri::Submit::Check::Type
+ type: binary
+
+ do-sign:
+ class: Youri::Submit::Action::Sign
+ options:
+ name: plf@zarb.org
+ path: ${home}/.gnupg
+ passphrase: s3kr3t
+
+ do-install:
+ class: Youri::Submit::Action::Install
+
+ do-link:
+ class: Youri::Submit::Action::Link
+
+ do-archive:
+ class: Youri::Submit::Action::Archive
+
+ do-clean:
+ class: Youri::Submit::Action::Clean
+
+ do-mail:
+ class: Youri::Submit::Action::Mail
+ options:
+ mta: /usr/sbin/sendmail
+ to: plf-announce@zarb.org
+ reply_to: plf-discuss@zarb.org
+ from: plf@zarb.org
+ prefix: RPM
+ cc:
+ hot-base: david@dindinx.org bellamy@neverland.net
+ dcgui: mathen@ketelhot.de
+ dclib: mathen@ketelhot.de
+ Video-DVDRip: dvdrip-users@exit1.org
+ hackVideo-DVDRip: dvdrip-users@exit1.org
+ goosnes: tak@bard.sytes.net
+ avidemux: fixounet@free.fr
+ vobcopy: robos@muon.de
+ drip: drip-devel@lists.sourceforge.net
+ libdscaler: vektor@dumbterm.net
+ xawdecode: pingus77@ifrance.com
+
+ do-rss:
+ class: Youri::Submit::Action::RSS
+ options:
+ file: ${home}/www/changelog.rss
+ title: PLF packages updates
+ link: http://plf.zarb.org/
+ description: ChangeLog for PLF packages
+
+ do-cvs:
+ class: Youri::Submit::Action::CVS
+
+ do-bugzilla:
+ class: Youri::Submit::Action::Bugzilla
+ options:
+ host: localhost
+ base: plf_bugs
+ user: plf
+ pass: s3kr3t
+ contact: plf@zarb.org
diff --git a/etc/youri-submit b/etc/youri-submit
new file mode 100644
index 0000000..6feb2c7
--- /dev/null
+++ b/etc/youri-submit
@@ -0,0 +1,60 @@
+# youri-submit completion
+# $Id$
+
+_youri-submit()
+{
+
+ local cur prev config
+
+ COMPREPLY=()
+ cur=${COMP_WORDS[COMP_CWORD]}
+ prev=${COMP_WORDS[COMP_CWORD-1]}
+
+ case "$prev" in
+ --config)
+ _filedir
+ return 0
+ ;;
+ --list)
+ COMPREPLY=( $( compgen -W 'targets steps' -- $cur ) )
+ return 0
+ ;;
+ --help)
+ COMPREPLY=( $( compgen -W 'repository steps' -- $cur ) )
+ return 0
+ ;;
+ esac
+
+ if [[ "$cur" == -* ]]; then
+ COMPREPLY=( $( compgen -W '--define --clean -l --list -h --help -t \
+ --test -v --verbose' -- $cur ) )
+ # add dangereous option for main command
+ if [[ ${COMP_WORDS[0]} == youri-submit ]]; then
+ COMPREPLY=( $( compgen -W '${COMPREPLY[@]} --config --skip-step' \
+ -- $cur ) )
+ fi
+ else
+ _count_args
+ case $args in
+ 1)
+ _find_config
+ COMPREPLY=( $( compgen -W '$( youri-submit $config --list targets )' -- $cur ) )
+ ;;
+ *)
+ _filedir
+ ;;
+ esac
+ fi
+
+}
+complete -F _youri-submit youri-submit youri-submit-restricted youri-submit-proxy
+
+_find_config()
+{
+ for (( i=1; i < COMP_CWORD; i++ )); do
+ if [[ "${COMP_WORDS[i]}" == --config ]]; then
+ config="--config ${COMP_WORDS[i+1]}"
+ break
+ fi
+ done
+}
diff --git a/lib/Makefile.am b/lib/Makefile.am
new file mode 100644
index 0000000..40ae36a
--- /dev/null
+++ b/lib/Makefile.am
@@ -0,0 +1,34 @@
+nobase_perllib_DATA = Youri/Submit/Test/Rpmlint.pm \
+ Youri/Submit/Test/Recency.pm \
+ Youri/Submit/Test/Tag.pm \
+ Youri/Submit/Test/Precedence.pm \
+ Youri/Submit/Test/History.pm \
+ Youri/Submit/Test/Type.pm \
+ Youri/Submit/Action.pm \
+ Youri/Submit/Plugin.pm \
+ Youri/Submit/Test.pm \
+ Youri/Submit/Action/Install.pm \
+ Youri/Submit/Action/RSS.pm \
+ Youri/Submit/Action/Bugzilla.pm \
+ Youri/Submit/Action/CVS.pm \
+ Youri/Submit/Action/Link.pm \
+ Youri/Submit/Action/Sign.pm \
+ Youri/Submit/Action/Archive.pm \
+ Youri/Submit/Action/Clean.pm \
+ Youri/Submit/Action/Mail.pm
+
+EXTRA_DIST = $(nobase_perllib_DATA)
+
+install-data-hook:
+ $(INSTALL) -d -m 755 $(DESTDIR)$(mandir)/man3
+ for file in $(nobase_perllib_DATA); do \
+ page=`echo $$file | sed -e 's/\//::/g' -e 's/\.pm$$/.3pm/'`; \
+ $(POD2MAN) --section=3 $(srcdir)/$$file \
+ $(DESTDIR)$(mandir)/man3/$$page; \
+ done
+
+uninstall-hook:
+ for file in $(nobase_perllib_DATA); do \
+ page=`echo $$file | sed -e 's/\//::/g' -e 's/\.pm$$/.3pm/'`; \
+ rm -f $(DESTDIR)$(mandir)/man3/$$page; \
+ done
diff --git a/lib/Youri/Submit/Action/Archive.pm b/lib/Youri/Submit/Action/Archive.pm
new file mode 100644
index 0000000..75e69cb
--- /dev/null
+++ b/lib/Youri/Submit/Action/Archive.pm
@@ -0,0 +1,73 @@
+# $Id: Archive.pm 1689 2007-06-28 22:44:24Z guillomovitch $
+package Youri::Submit::Action::Archive;
+
+=head1 NAME
+
+Youri::Submit::Action::Archive - Old revisions archiving
+
+=head1 DESCRIPTION
+
+This action plugin ensures archiving of old package revisions.
+
+=cut
+
+use warnings;
+use strict;
+use Carp;
+use base qw/Youri::Submit::Step/;
+
+sub _init {
+ my $self = shift;
+ my %options = (
+ perms => 644,
+ @_
+ );
+
+ $self->{_perms} = $options{perms};
+}
+
+sub process_packages {
+ my ($self, $packages, $repository, $target, $context) = @_;
+ croak "Not a class method" unless ref $self;
+
+ foreach my $package (@$packages) {
+ foreach my $replaced_package (
+ $repository->get_replaced_packages($package, $target, $context)
+ ) {
+ my $file = $replaced_package->as_file();
+ my $dest = $repository->get_archive_dir($package, $target, $context);
+
+ print "archiving file $file to $dest\n" if $self->{_verbose};
+
+ unless ($self->{_test}) {
+ # create destination dir if needed
+ if (! -d $dest) {
+ eval {
+ mkpath($dest, 0, ($self->{_perms} + 111));
+ };
+ if ($@) {
+ croak "Unable to create directory $dest: $@";
+ }
+ }
+
+ # copy file to new location
+ copy($file, $dest)
+ or croak "Unable to copy file $file to $dest: $!";
+
+ # set permissions
+ chmod oct($self->{_perms}), $dest . '/' . basename($file)
+ or croak "Unable to set perms of file $file to $self->{_perms}: $!";
+ }
+ }
+ }
+}
+
+=head1 COPYRIGHT AND LICENSE
+
+Copyright (C) 2002-2006, YOURI project
+
+This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
+
+=cut
+
+1;
diff --git a/lib/Youri/Submit/Action/Bugzilla.pm b/lib/Youri/Submit/Action/Bugzilla.pm
new file mode 100644
index 0000000..a8ad5ce
--- /dev/null
+++ b/lib/Youri/Submit/Action/Bugzilla.pm
@@ -0,0 +1,78 @@
+# $Id: Bugzilla.pm 1689 2007-06-28 22:44:24Z guillomovitch $
+package Youri::Submit::Action::Bugzilla;
+
+=head1 NAME
+
+Youri::Submit::Action::Bugzilla - Bugzilla synchronisation
+
+=head1 DESCRIPTION
+
+This action ensures synchronisation with Bugzilla. It assumes given package set
+is canonical, with source package first.
+
+=cut
+
+use warnings;
+use strict;
+use Carp;
+use Youri::BTS::Bugzilla;
+use base qw/Youri::Submit::Step/;
+
+sub _init {
+ my $self = shift;
+ my %options = (
+ lib => undef,
+ project => undef,
+ contact => undef,
+ @_
+ );
+
+ $self->{_bugzilla} = Youri::BTS::Bugzilla->new(
+ $options{lib} ? (lib => $options{lib}) : (),
+ $options{project} ? (project => $options{project}) : (),
+ );
+ $self->{_contact} = $options{contact};
+}
+
+sub process_packages {
+ my ($self, $packages, $repository, $target, $context) = @_;
+ croak "Not a class method" unless ref $self;
+
+ my $package = $packages->[0];
+
+ my $name = $package->get_name();
+ my $version = $package->get_version();
+ my $summary = $package->get_summary();
+ my $packager = $package->get_packager();
+ $packager =~ s/.*<(.*)>/$1/;
+
+ if ($self->{_bugzilla}->has_package($name)) {
+ my %versions =
+ map { $_ => 1 }
+ $self->{_bugzilla}->get_versions($name);
+ unless ($versions{$version}) {
+ print "adding version $version to bugzilla\n" if $self->{_verbose};
+ $self->{_bugzilla}->add_version($name, $version)
+ unless $self->{_test};
+ }
+ } else {
+ print "adding package $name to bugzilla\n" if $self->{_verbose};
+ $self->{_bugzilla}->add_package(
+ $name,
+ $summary,
+ $version,
+ $packager,
+ $self->{_contact}
+ ) unless $self->{_test};
+ }
+}
+
+=head1 COPYRIGHT AND LICENSE
+
+Copyright (C) 2002-2006, YOURI project
+
+This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
+
+=cut
+
+1;
diff --git a/lib/Youri/Submit/Action/CVS.pm b/lib/Youri/Submit/Action/CVS.pm
new file mode 100644
index 0000000..48e3eae
--- /dev/null
+++ b/lib/Youri/Submit/Action/CVS.pm
@@ -0,0 +1,136 @@
+# $Id: CVS.pm 1689 2007-06-28 22:44:24Z guillomovitch $
+package Youri::Submit::Action::CVS;
+
+=head1 NAME
+
+Youri::Submit::Action::CVS - CVS versionning
+
+=head1 DESCRIPTION
+
+This action ensures CVS versionning of package sources. It assumes given
+package set is canonical, with source package first.
+
+=cut
+
+use warnings;
+use strict;
+use Carp;
+use Cwd;
+use File::Temp qw/tempdir/;
+use base qw/Youri::Submit::Step/;
+
+sub _init {
+ my $self = shift;
+ my %options = (
+ exclude => '\.(tar(\.(gz|bz2))?|zip)$',
+ perms => 644,
+ @_
+ );
+
+ $self->{_exclude} = $options{exclude};
+ $self->{_perms} = $options{perms};
+}
+
+sub process_packages {
+ my ($self, $packages, $repository, $target, $context) = @_;
+ croak "Not a class method" unless ref $self;
+
+ my $package = $packages->[0];
+
+ my $name = $package->get_name();
+ my $version = $package->get_version();
+ my $release = $package->get_release();
+
+ my $root = $repository->get_version_root();
+ my $path = $repository->get_version_path($package, $target, $define);
+
+ # remember original directory
+ my $original_dir = cwd();
+
+ # get a safe temporary directory
+ my $dir = tempdir( CLEANUP => 1 );
+ chdir $dir;
+
+ # first checkout base directory only
+ system("cvs -Q -d $root co -l $path");
+
+ # try to checkout package directory
+ my $dest = $path . '/' . $name;
+ system("cvs -Q -d $root co $dest");
+
+ # create directory if previous import failed
+ unless (-d $dest) {
+ print "adding directory $dest\n" if $self->{_verbose};
+ system("install -d -m " . ($self->{_perms} + 111) . " $dest");
+ system("cvs -Q -d $root add $dest");
+ }
+
+ chdir $dest;
+
+ # remove all files
+ unlink grep { -f } glob '*';
+
+ # extract all rpm files locally
+ $package->extract();
+
+ # remove excluded files
+ if ($self->{_exclude}) {
+ unlink grep { -f && /$self->{_exclude}/ } glob '*';
+ }
+
+ # uncompress all compressed files
+ system("bunzip2 *.bz2 2>/dev/null");
+ system("gunzip *.gz 2>/dev/null");
+
+ my (@to_remove, @to_add, @to_add_binary);
+ foreach my $line (`cvs -nq update`) {
+ if ($line =~ /^\? (\S+)/) {
+ if (-B $1) {
+ push(@to_add_binary, $1);
+ } else {
+ push(@to_add, $1);
+ }
+ }
+ if ($line =~ /^U (\S+)/) {
+ push(@to_remove, $1);
+ }
+ }
+ if (@to_remove) {
+ my $to_remove = join(' ', @to_remove);
+ print "removing file(s) $to_remove\n" if $self->{_verbose};
+ system("cvs -Q remove $to_remove");
+ }
+ if (@to_add) {
+ my $to_add = join(' ', @to_add);
+ print "adding text file(s) $to_add\n" if $self->{_verbose};
+ system("cvs -Q add $to_add");
+ }
+ if (@to_add_binary) {
+ my $to_add_binary = join(' ', @to_add_binary);
+ print "adding binary file(s) $to_add_binary\n" if $self->{_verbose};
+ system("cvs -Q add -kb $to_add_binary");
+ }
+
+ print "committing current directory\n" if $self->{_verbose};
+ system("cvs -Q commit -m $version-$release") unless $self->{_test};
+
+ # tag new release
+ my $tag = "r$version-$release";
+ $tag =~ s/\./_/g;
+ print "tagging current directory as $tag\n" if $self->{_verbose};
+ system("cvs -Q tag $tag") unless $self->{_test};
+
+ # get back to original directory
+ chdir $original_dir;
+
+}
+
+=head1 COPYRIGHT AND LICENSE
+
+Copyright (C) 2002-2006, YOURI project
+
+This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
+
+=cut
+
+1;
diff --git a/lib/Youri/Submit/Action/Clean.pm b/lib/Youri/Submit/Action/Clean.pm
new file mode 100644
index 0000000..110326a
--- /dev/null
+++ b/lib/Youri/Submit/Action/Clean.pm
@@ -0,0 +1,42 @@
+# $Id: Clean.pm 1689 2007-06-28 22:44:24Z guillomovitch $
+package Youri::Submit::Action::Clean;
+
+=head1 NAME
+
+Youri::Submit::Action::Clean - Old revisions cleanup
+
+=head1 DESCRIPTION
+
+This action plugin ensures cleanup of old package revisions.
+
+=cut
+
+use warnings;
+use strict;
+use Carp;
+use base qw/Youri::Submit::Step/;
+
+sub process_packages {
+ my ($self, $packages, $repository, $target, $context) = @_;
+ croak "Not a class method" unless ref $self;
+
+ foreach my $package (@$packages) {
+ foreach my $replaced_package (
+ $repository->get_replaced_packages($package, $target, $define)
+ ) {
+ my $file = $replaced_package->as_file();
+ print "deleting file $file\n" if $self->{_verbose};
+ unlink $file unless $self->{_test};
+ }
+ }
+}
+
+=head1 COPYRIGHT AND LICENSE
+
+Copyright (C) 2002-2006, YOURI project
+
+This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
+
+=cut
+
+1;
diff --git a/lib/Youri/Submit/Action/Install.pm b/lib/Youri/Submit/Action/Install.pm
new file mode 100644
index 0000000..c9a419f
--- /dev/null
+++ b/lib/Youri/Submit/Action/Install.pm
@@ -0,0 +1,74 @@
+# $Id: Install.pm 1689 2007-06-28 22:44:24Z guillomovitch $
+package Youri::Submit::Action::Install;
+
+=head1 NAME
+
+Youri::Submit::Action::Install - Package installation
+
+=head1 DESCRIPTION
+
+This action plugin ensures installation of new package revisions.
+
+=cut
+
+use warnings;
+use strict;
+use Carp;
+use File::Copy;
+use File::Path;
+use File::Basename;
+use base qw/Youri::Submit::Step/;
+
+sub _init {
+ my $self = shift;
+ my %options = (
+ perms => 644,
+ @_
+ );
+
+ $self->{_perms} = $options{perms};
+
+ return $self;
+}
+
+sub process_packages {
+ my ($self, $packages, $repository, $target, $context) = @_;
+ croak "Not a class method" unless ref $self;
+
+ foreach my $package (@$packages) {
+ my $file = $package->as_file();
+ my $dest = $repository->get_install_dir($package, $target, $context);
+
+ print "installing file $file to $dest\n" if $self->{_verbose};
+
+ unless ($self->{_test}) {
+ # create destination dir if needed
+ if (! -d $dest) {
+ eval {
+ mkpath($dest, 0, ($self->{_perms} + 111));
+ };
+ if ($@) {
+ croak "Unable to create directory $dest: $@";
+ }
+ }
+
+ # copy file to new location
+ copy($file, $dest)
+ or croak "Unable to copy file $file to $dest: $!";
+
+ # set permissions
+ chmod oct($self->{_perms}), $dest . '/' . basename($file)
+ or croak "Unable to set perms of file $file to $self->{_perms}: $!";
+ }
+ }
+}
+
+=head1 COPYRIGHT AND LICENSE
+
+Copyright (C) 2002-2006, YOURI project
+
+This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
+
+=cut
+
+1;
diff --git a/lib/Youri/Submit/Action/Link.pm b/lib/Youri/Submit/Action/Link.pm
new file mode 100644
index 0000000..940f7d7
--- /dev/null
+++ b/lib/Youri/Submit/Action/Link.pm
@@ -0,0 +1,79 @@
+# $Id: Link.pm 1689 2007-06-28 22:44:24Z guillomovitch $
+package Youri::Submit::Action::Link;
+
+=head1 NAME
+
+Youri::Submit::Action::Link - Noarch packages linking
+
+=head1 DESCRIPTION
+
+This action plugin ensures linking of noarch packages between arch-specific
+directories.
+
+=cut
+
+use warnings;
+use strict;
+use Carp;
+use Cwd;
+use File::Spec;
+use base qw/Youri::Submit::Step/;
+
+sub _init {
+ my $self = shift;
+ my %options = (
+ symbolic => 0, # use symbolic linking
+ @_
+ );
+
+ $self->{_symbolic} = $options{symbolic};
+}
+
+sub process_packages {
+ my ($self, $packages, $repository, $target, $context) = @_;
+ croak "Not a class method" unless ref $self;
+
+ foreach my $package (@$packages) {
+ # only needed for noarch packages
+ next unless $package->get_arch() eq 'noarch';
+
+ my $default_dir = $repository->get_install_dir($package, $target, $define);
+ my $file = $package->get_file_name();
+
+ foreach my $arch ($repository->get_extra_arches()) {
+ # compute installation target, forcing arch
+ my $other_dir = $repository->get_install_dir(
+ $package,
+ $target,
+ $context,
+ { arch => $arch }
+ );
+
+ if (! $self->{_test} && -d $other_dir) {
+ my $current_dir = cwd();
+ chdir $other_dir
+ or croak "Can't change directory to $other_dir: $!";
+ my $default_file = File::Spec->abs2rel($default_dir) . '/' . $file;
+ if ($self->{_symbolic}) {
+ symlink $default_file, $file
+ or croak "Can't symlink $default_file to $file: $!";
+ } else {
+ link $default_file, $file
+ or croak "Can't link $default_file to $file: $!";
+ }
+ chdir $current_dir
+ or croak "Can't change directory to $current_dir: $!";
+ }
+ }
+ }
+}
+
+=head1 COPYRIGHT AND LICENSE
+
+Copyright (C) 2002-2006, YOURI project
+
+This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
+
+=cut
+
+1;
diff --git a/lib/Youri/Submit/Action/Mail.pm b/lib/Youri/Submit/Action/Mail.pm
new file mode 100644
index 0000000..4e39ef8
--- /dev/null
+++ b/lib/Youri/Submit/Action/Mail.pm
@@ -0,0 +1,143 @@
+# $Id: Mail.pm 1689 2007-06-28 22:44:24Z guillomovitch $
+package Youri::Submit::Action::Mail;
+
+=head1 NAME
+
+Youri::Submit::Action::Mail - Mail notification
+
+=head1 DESCRIPTION
+
+This action ensures mail notification of new package revisions. It assumes
+given package set is canonical, with source package first.
+
+=cut
+
+use warnings;
+use strict;
+use MIME::Entity;
+use Encode qw/from_to/;
+use Carp;
+use Youri::Package;
+use base qw/Youri::Submit::Step/;
+
+sub _init {
+ my $self = shift;
+ my %options = (
+ mta => '/usr/sbin/sendmail',
+ to => '',
+ from => '',
+ cc => '',
+ prefix => '',
+ encoding => 'quoted-printable',
+ charset => 'iso-8859-1',
+ @_
+ );
+
+ croak "undefined mail MTA" unless $options{mta};
+ croak "invalid mail MTA $options{mta}" unless -x $options{mta};
+ croak "undefined to" unless $options{to};
+ if ($options{cc}) {
+ croak "cc should be an hashref" unless ref $options{cc} eq 'HASH';
+ }
+ croak "invalid charset $options{charset}"
+ unless Encode::resolve_alias($options{charset});
+
+ $self->{_mta} = $options{mta};
+ $self->{_to} = $options{to};
+ $self->{_from} = $options{from};
+ $self->{_cc} = $options{cc};
+ $self->{_prefix} = $options{prefix};
+ $self->{_encoding} = $options{encoding};
+ $self->{_charset} = $options{charset};
+}
+
+sub process_packages {
+ my ($self, $packages, $repository, $target, $context) = @_;
+ croak "Not a class method" unless ref $self;
+
+ my $package = $packages->[0];
+
+ my $from = $package->get_packager();
+
+ # force from adress if defined
+ $from =~ s/<.*>/<$self->{_from}>/ if $self->{_from};
+
+ my $subject = $self->get_subject($package, $repository, $target, $define);
+ my $content = $self->get_content($package, $repository, $target, $define);
+
+ # ensure proper codeset conversion
+ # for informations coming from package
+ my $charset = $repository->get_package_charset();
+ from_to($content, $charset, $self->{_charset});
+ from_to($subject, $charset, $self->{_charset});
+
+ my $mail = MIME::Entity->build(
+ Type => 'text/plain',
+ Charset => $self->{_charset},
+ Encoding => $self->{_encoding},
+ From => $from,
+ To => $self->{_to},
+ Subject => $subject,
+ Data => $content,
+ );
+
+ if ($self->{_cc}) {
+ my $cc = $self->{_cc}->{$package->get_name()};
+ $mail->head()->add('cc', $cc) if $cc;
+ }
+
+ if ($self->{_test}) {
+ $mail->print(\*STDOUT);
+ } else {
+ my $command = "$self->{_mta} -t -oi -oem";
+ open(my $output, '|-', $command) or croak "Can't run $command: $!";
+ $mail->print($output);
+ close $output;
+ }
+
+}
+
+sub get_subject {
+ my ($self, $package, $repository, $target, $define) = @_;
+ croak "Not a class method" unless ref $self;
+
+ return
+ ($self->{_prefix} ? '[' . $self->{_prefix} . '] ' : '' ) .
+ $package->as_formated_string('%{name}-%{version}-%{release}');
+}
+
+sub get_content {
+ my ($self, $package, $repository, $target, $define) = @_;
+ croak "Not a class method" unless ref $self;
+
+ my $information = $package->as_formated_string(<<EOF);
+Name : %-27{NAME} Relocations: %|PREFIXES?{[%{PREFIXES} ]}:{(not relocatable)}|
+Version : %-27{VERSION} Vendor: %{VENDOR}
+Release : %-27{RELEASE} Build Date: %{BUILDTIME:date}
+Install Date: %|INSTALLTIME?{%-27{INSTALLTIME:date}}:{(not installed) }| Build Host: %{BUILDHOST}
+Group : %-27{GROUP} Source RPM: %{SOURCERPM}
+Size : %-27{SIZE}%|LICENSE?{ License: %{LICENSE}}|
+Signature : %|DSAHEADER?{%{DSAHEADER:pgpsig}}:{%|RSAHEADER?{%{RSAHEADER:pgpsig}}:{%|SIGGPG?{%{SIGGPG:pgpsig}}:{%|SIGPGP?{%{SIGPGP:pgpsig}}:{(none)}|}|}|}|
+%|PACKAGER?{Packager : %{PACKAGER}\n}|%|URL?{URL : %{URL}\n}|Summary : %{SUMMARY}
+Description :\n%{DESCRIPTION}
+EOF
+
+ my $last_change = $package->get_last_change();
+
+ return $last_change ?
+ $information . "\n" .
+ $last_change->get_author() . ":\n" .
+ $last_change->get_raw_text() :
+ $information;
+}
+
+
+=head1 COPYRIGHT AND LICENSE
+
+Copyright (C) 2002-2006, YOURI project
+
+This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
+
+=cut
+
+1;
diff --git a/lib/Youri/Submit/Action/RSS.pm b/lib/Youri/Submit/Action/RSS.pm
new file mode 100644
index 0000000..11fb5b5
--- /dev/null
+++ b/lib/Youri/Submit/Action/RSS.pm
@@ -0,0 +1,117 @@
+# $Id: RSS.pm 1689 2007-06-28 22:44:24Z guillomovitch $
+package Youri::Submit::Action::RSS;
+
+=head1 NAME
+
+Youri::Submit::Action::RSS - RSS notification
+
+=head1 DESCRIPTION
+
+This action ensures RSS notification of new package revisions. It assumes given package set is canonical, with source package first.
+
+=cut
+
+use warnings;
+use strict;
+use XML::RSS;
+use Encode qw/from_to/;
+use Carp;
+use base qw/Youri::Submit::Step/;
+
+sub _init {
+ my $self = shift;
+ my %options = (
+ file => '',
+ title => '',
+ link => '',
+ description => '',
+ charset => 'iso-8859-1',
+ max_items => 10,
+ @_
+ );
+
+ croak "undefined rss file" unless $options{file};
+ croak "invalid charset $options{charset}"
+ unless Encode::resolve_alias($options{charset});
+
+ $self->{_file} = $options{file};
+ $self->{_title} = $options{title};
+ $self->{_link} = $options{link};
+ $self->{_description} = $options{description};
+ $self->{_charset} = $options{charset};
+ $self->{_max_items} = $options{max_items};
+}
+
+sub process_packages {
+ my ($self, $packages, $repository, $target, $context) = @_;
+ croak "Not a class method" unless ref $self;
+
+ my $package = $packages->[0];
+
+ my $subject = $package->as_formated_string('%{name}-%{version}-%{release}');
+ my $content = $package->as_formated_string(<<EOF);
+Name : %-27{NAME} Relocations: %|PREFIXES?{[%{PREFIXES} ]}:{(not relocatable)}|
+Version : %-27{VERSION} Vendor: %{VENDOR}
+Release : %-27{RELEASE} Build Date: %{BUILDTIME:date}
+Install Date: %|INSTALLTIME?{%-27{INSTALLTIME:date}}:{(not installed) }| Build Host: %{BUILDHOST}
+Group : %-27{GROUP} Source RPM: %{SOURCERPM}
+Size : %-27{SIZE}%|LICENSE?{ License: %{LICENSE}}|
+Signature : %|DSAHEADER?{%{DSAHEADER:pgpsig}}:{%|RSAHEADER?{%{RSAHEADER:pgpsig}}:{%|SIGGPG?{%{SIGGPG:pgpsig}}:{%|SIGPGP?{%{SIGPGP:pgpsig}}:{(none)}|}|}|}|
+%|PACKAGER?{Packager : %{PACKAGER}\n}|%|URL?{URL : %{URL}\n}|Summary : %{SUMMARY}
+Description :\n%{DESCRIPTION}
+EOF
+
+ $content =~ s/$/<br\/>/mg;
+
+ # ensure proper codeset conversion
+ # for informations coming from package
+ my $charset = $repository->get_package_charset();
+ from_to($content, $charset, $self->{_charset});
+ from_to($subject, $charset, $self->{_charset});
+
+ my $rss = XML::RSS->new(
+ encoding => $self->{_charset},
+ encode_output => 1
+ );
+
+ my $file = $self->{_file};
+ if (-e $file) {
+ eval {
+ $rss->parsefile($file);
+ };
+ if ($@) {
+ croak "Unable to parse RSS file $file: $@";
+ }
+ splice(@{$rss->{items}}, $self->{_max_items})
+ if @{$rss->{items}} >= $self->{_max_items};
+ } else {
+ $rss->channel(
+ title => $self->{_title},
+ link => $self->{_link},
+ description => $self->{_description},
+ language => 'en'
+ );
+ }
+
+ $rss->add_item(
+ title => $subject,
+ description => $content,
+ mode => 'insert'
+ );
+
+ if ($self->{_test}) {
+ print $rss->as_string();
+ } else {
+ $rss->save($file);
+ }
+}
+
+=head1 COPYRIGHT AND LICENSE
+
+Copyright (C) 2002-2006, YOURI project
+
+This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
+
+=cut
+
+1;
diff --git a/lib/Youri/Submit/Action/Reorder.pm b/lib/Youri/Submit/Action/Reorder.pm
new file mode 100644
index 0000000..a6ed3da
--- /dev/null
+++ b/lib/Youri/Submit/Action/Reorder.pm
@@ -0,0 +1,37 @@
+# $Id: Reorder.pm 1688 2007-06-28 22:44:16Z guillomovitch $
+package Youri::Submit::Action::Reorder;
+
+=head1 NAME
+
+Youri::Submit::Action::Reorder - Reorder packages with source first
+
+=head1 DESCRIPTION
+
+This action ensures that the source package is the first in the list.
+
+=cut
+
+use warnings;
+use strict;
+use Carp;
+use base qw/Youri::Submit::Step/;
+
+sub process_packages {
+ my ($self, $packages, $repository, $target, $context) = @_;
+ croak "Not a class method" unless ref $self;
+
+ $packages = [
+ grep { $_->is_source() } @$packages,
+ grep { !$_->is_source() } @$packages,
+ ];
+}
+
+=head1 COPYRIGHT AND LICENSE
+
+Copyright (C) 2002-2006, YOURI project
+
+This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
+
+=cut
+
+1;
diff --git a/lib/Youri/Submit/Action/Sign.pm b/lib/Youri/Submit/Action/Sign.pm
new file mode 100644
index 0000000..ce8f4a9
--- /dev/null
+++ b/lib/Youri/Submit/Action/Sign.pm
@@ -0,0 +1,60 @@
+# $Id: Sign.pm 1689 2007-06-28 22:44:24Z guillomovitch $
+package Youri::Submit::Action::Sign;
+
+=head1 NAME
+
+Youri::Submit::Action::Sign - GPG signature
+
+=head1 DESCRIPTION
+
+This action plugin ensures GPG signature of packages.
+
+=cut
+
+use warnings;
+use strict;
+use Carp;
+use base qw/Youri::Submit::Step/;
+
+sub _init {
+ my $self = shift;
+ my %options = (
+ name => '',
+ path => $ENV{HOME} . '/.gnupg',
+ passphrase => '',
+ @_
+ );
+
+ croak "undefined name" unless $options{name};
+ croak "undefined path" unless $options{path};
+ croak "invalid path $options{path}" unless -d $options{path};
+
+ $self->{_name} = $options{name};
+ $self->{_path} = $options{path};
+ $self->{_passphrase} = $options{passphrase};
+}
+
+sub process_packages {
+ my ($self, $packages, $repository, $target, $context) = @_;
+ croak "Not a class method" unless ref $self;
+
+ return if $self->{_test};
+
+ foreach my $package (@$packages) {
+ $package->sign(
+ $self->{_name},
+ $self->{_path},
+ $self->{_passphrase}
+ );
+ }
+}
+
+=head1 COPYRIGHT AND LICENSE
+
+Copyright (C) 2002-2006, YOURI project
+
+This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
+
+=cut
+
+1;
diff --git a/lib/Youri/Submit/Step.pm b/lib/Youri/Submit/Step.pm
new file mode 100644
index 0000000..0085277
--- /dev/null
+++ b/lib/Youri/Submit/Step.pm
@@ -0,0 +1,125 @@
+# $Id: Step.pm 1687 2007-06-28 22:44:07Z guillomovitch $
+package Youri::Submit::Step;
+
+=head1 NAME
+
+Youri::Submit::Step - Abstract submission step
+
+=head1 DESCRIPTION
+
+This abstract class defines youri-submit plugin interface.
+
+=cut
+
+use warnings;
+use strict;
+use Carp;
+
+=head1 CLASS METHODS
+
+=head2 new(%args)
+
+Creates and returns a new Youri::Submit::Plugin object.
+
+Generic parameters:
+
+=over
+
+=item skip $skip
+
+List of packages for which to skip this plugin
+
+=over
+
+Warning: do not call directly, call subclass constructor instead.
+
+=cut
+
+sub new {
+ my $class = shift;
+ croak "Abstract class" if $class eq __PACKAGE__;
+
+ my %options = (
+ id => '', # object id
+ test => 0, # test mode
+ verbose => 0, # verbose mode
+ skip => undef,
+ @_
+ );
+
+ croak "skip should be an arrayref"
+ if $options{skip} and ref $options{skip} ne 'ARRAY';
+
+
+ my $self = bless {
+ _id => $options{id},
+ _test => $options{test},
+ _verbose => $options{verbose},
+ _skip => { map { $_ => 1 } @{$options{skip}} }
+ }, $class;
+
+ $self->_init(%options);
+
+ return $self;
+}
+
+sub _init {
+ # do nothing
+}
+
+=head1 INSTANCE METHODS
+
+=head2 get_id()
+
+Returns plugin identity.
+
+=cut
+
+sub get_id {
+ my ($self) = @_;
+ croak "Not a class method" unless ref $self;
+
+ return $self->{_id};
+}
+
+=head2 run($packages, $repository, $target, $context)
+
+Execute action on given L<Youri::Package> object.
+
+=cut
+
+sub run {
+ my ($self, $packages, $repository, $target, $context) = @_;
+ croak "Not a class method" unless ref $self;
+ croak "packages is not an array reference" unless ref $packages eq 'ARRAY';
+
+ # filter exceptions
+ $packages = [
+ grep { !$self->{_skip}->{$_->get_canonical_name()} }
+ @$packages
+ ];
+
+ $self->process_packages($repository, $packages, $target, $context)
+ if @$packages;
+}
+
+
+=head1 SUBCLASSING
+
+The following methods have to be implemented:
+
+=over
+
+=item run
+
+=back
+
+=head1 COPYRIGHT AND LICENSE
+
+Copyright (C) 2002-2006, YOURI project
+
+This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
+
+=cut
+
+1;
diff --git a/lib/Youri/Submit/Test/History.pm b/lib/Youri/Submit/Test/History.pm
new file mode 100644
index 0000000..b4b7969
--- /dev/null
+++ b/lib/Youri/Submit/Test/History.pm
@@ -0,0 +1,65 @@
+# $Id: History.pm 1687 2007-06-28 22:44:07Z guillomovitch $
+package Youri::Submit::Test::History;
+
+=head1 NAME
+
+Youri::Submit::Test::History - Non-linear history test
+
+=head1 DESCRIPTION
+
+This test rejects packages whose history does not include last available
+revision one. It assumes given set of package is canonical, and only check
+first one.
+
+=cut
+
+use warnings;
+use strict;
+use Carp;
+use Youri::Package;
+use List::MoreUtils qw/none/;
+use base qw/Youri::Submit::Step/;
+
+sub process_packages {
+ my ($self, $packages, $repository, $target, $context) = @_;
+ croak "Not a class method" unless ref $self;
+
+ my $errors;
+ my $package = $packages->[0];
+ my $current_package =
+ $repository->get_last_older_revision($package, $target, $context);
+
+ # new package
+ return unless $current_package;
+
+ # skip the test if last revision has been produced from another
+ # source package, as it occurs during package split/merges
+ return
+ if $current_package->get_canonical_name()
+ ne $package->get_canonical_name();
+
+ my $last_change_author =
+ $current_package->get_last_change()->get_author();
+
+ if (
+ none { $last_change_author eq $_ }
+ map { $_->get_author() }
+ $package->get_changes()
+ ) {
+ $errors .=
+ "last changelog entry $last_change_author from current package "
+ . "$current_package missing from changelog";
+ }
+
+ croak $errors if $errors;
+}
+
+=head1 COPYRIGHT AND LICENSE
+
+Copyright (C) 2002-2006, YOURI project
+
+This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
+
+=cut
+
+1;
diff --git a/lib/Youri/Submit/Test/Precedence.pm b/lib/Youri/Submit/Test/Precedence.pm
new file mode 100644
index 0000000..4cd35a6
--- /dev/null
+++ b/lib/Youri/Submit/Test/Precedence.pm
@@ -0,0 +1,60 @@
+# $Id: Precedence.pm 1687 2007-06-28 22:44:07Z guillomovitch $
+package Youri::Submit::Test::Precedence;
+
+=head1 NAME
+
+Youri::Submit::Test::Precedence - Release test against another test
+
+=head1 DESCRIPTION
+
+This test rejects packages whose an older revision already exists for another
+upload target. It assumes given set of package is canonical, and only check
+first one.
+
+
+=cut
+
+use warnings;
+use strict;
+use Carp;
+use base qw/Youri::Submit::Step/;
+
+sub _init {
+ my $self = shift;
+ my %options = (
+ _target => undef, # mandatory targets
+ @_
+ );
+
+ croak "undefined target" unless $options{target};
+
+ $self->{_target} = $options{target};
+}
+
+sub process_package {
+ my ($self, $packages, $repository, $target, $context) = @_;
+ croak "Not a class method" unless ref $self;
+
+ my $errors;
+ my $package = $packages->[0];
+
+ my @older_revisions =
+ $repository->get_older_revisions($package, $self->{_target}, $context);
+ $errors .=
+ "older revisions still exists for $self->{_target}: "
+ . join(', ', @older_revisions)
+ if @older_revisions;
+ }
+
+ croak $errors if $errors;
+}
+
+=head1 COPYRIGHT AND LICENSE
+
+Copyright (C) 2002-2006, YOURI project
+
+This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
+
+=cut
+
+1;
diff --git a/lib/Youri/Submit/Test/Recency.pm b/lib/Youri/Submit/Test/Recency.pm
new file mode 100644
index 0000000..db6ba59
--- /dev/null
+++ b/lib/Youri/Submit/Test/Recency.pm
@@ -0,0 +1,50 @@
+# $Id: Recency.pm 1687 2007-06-28 22:44:07Z guillomovitch $
+package Youri::Submit::Test::Recency;
+
+=head1 NAME
+
+Youri::Submit::Test::Recency - Release test against current target
+
+=head1 DESCRIPTION
+
+This test rejects packages whose a current or newer revision already exists for
+current upload target. It assumes given set of package is canonical, and only
+check first one.
+
+=cut
+
+use warnings;
+use strict;
+use Carp;
+use base qw/Youri::Submit::Step/;
+
+sub process_packages {
+ my ($self, $packages, $repository, $target, $context) = @_;
+ croak "Not a class method" unless ref $self;
+
+ my $errors;
+ my $package = $packages->[0];
+
+ my $file = $repository->get_install_file($package, $target, $context);
+ $errors .= "current revision already exists for $target\n"
+ if -f $file;
+
+ my @newer_revisions =
+ $repository->get_newer_revisions($package, $target, $define);
+ $errors .=
+ "newer revisions already exists for $target: "
+ . join(', ', @newer_revisions) . "\n"
+ if @newer_revisions;
+
+ croak $errors if $errors;
+}
+
+=head1 COPYRIGHT AND LICENSE
+
+Copyright (C) 2002-2006, YOURI project
+
+This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
+
+=cut
+
+1;
diff --git a/lib/Youri/Submit/Test/Rpmlint.pm b/lib/Youri/Submit/Test/Rpmlint.pm
new file mode 100644
index 0000000..170d4c3
--- /dev/null
+++ b/lib/Youri/Submit/Test/Rpmlint.pm
@@ -0,0 +1,97 @@
+# $Id: Rpmlint.pm 1687 2007-06-28 22:44:07Z guillomovitch $
+package Youri::Submit::Test::Rpmlint;
+
+=head1 NAME
+
+Youri::Submit::Test::Rpmlint - Rpmlint-based test
+
+=head1 DESCRIPTION
+
+This test plugin wraps rpmlint, and reject packages triggering results
+declared as fatal.
+
+=cut
+
+use warnings;
+use strict;
+use Carp;
+use base qw/Youri::Submit::Step/;
+
+=head2 new(%args)
+
+Creates and returns a new Youri::Submit::Test::Rpmlint object.
+
+Specific parameters:
+
+=over
+
+=item results $results
+
+List of rpmlint result id considered as fatal.
+
+=item path $path
+
+Path to the rpmlint executable (default: /usr/bin/rpmlint)
+
+=item config $config
+
+Specific rpmlint configuration.
+
+=back
+
+=cut
+
+
+sub _init {
+ my $self = shift;
+ my %options = (
+ results => undef,
+ path => '/usr/bin/rpmlint',
+ config => '',
+ @_
+ );
+
+ croak "no results to test" unless $options{results};
+ croak "results should be an arrayref" unless ref $options{results} eq 'ARRAY';
+
+ $self->{_results} = map { $_ => 1 } @{$options{results}};
+ $self->{_path} = $options{path};
+ $self->{_config} = $options{config};
+}
+
+sub process_packages {
+ my ($self, $packages, $repository, $target, $context) = @_;
+ croak "Not a class method" unless ref $self;
+
+ my $errors;
+
+ foreach my $package (@$packages) {
+ my $command =
+ $self->{_path} . ' ' .
+ ($self->{_config} ? "-f $self->{_config} " : '' ) .
+ $package->as_file();
+
+ open(my $input, '-|', $command) or croak "Can't run $command: $!";
+ my $pattern = qr/^[EW]: \S+ (\S+) (.*)$/;
+ while (my $line = <$input>) {
+ next unless $line =~ $pattern;
+ my $id = $1;
+ my $value = $2;
+ $errors .= "$id $value"
+ if $self->{_results}->{$id};
+ }
+ close $input;
+ }
+
+ croak $errors if $errors;
+}
+=head1 COPYRIGHT AND LICENSE
+
+Copyright (C) 2002-2006, YOURI project
+
+This program is free software; you can redistribute it and/or modify it under
+the same terms as Perl itself.
+
+=cut
+
+1;
diff --git a/lib/Youri/Submit/Test/Set.pm b/lib/Youri/Submit/Test/Set.pm
new file mode 100644
index 0000000..2ed4994
--- /dev/null
+++ b/lib/Youri/Submit/Test/Set.pm
@@ -0,0 +1,79 @@
+# $Id: Set.pm 1687 2007-06-28 22:44:07Z guillomovitch $
+package Youri::Submit::Test::Set;
+
+=head1 NAME
+
+Youri::Submit::Test::Set - Packages set test
+
+=head1 DESCRIPTION
+
+This test rejects non-canonical packages set, meaning packages generated from
+different sources packages, checking expected number or source and binaries
+packages.
+
+=cut
+
+use warnings;
+use strict;
+use Carp;
+use List::MoreUtils qw/all/;
+use base qw/Youri::Submit::Step/;
+
+sub _init {
+ my $self = shift;
+ my %options = (
+ min_sources => 1,
+ max_sources => 1,
+ min_binaries => 1,
+ max_binaries => undef,
+ @_
+ );
+
+ $self->{_min_sources} = $options{min_sources};
+ $self->{_max_sources} = $options{max_sources};
+ $self->{_min_binaries} = $options{min_binaries};
+ $self->{_max_binaries} = $options{max_binaries};
+}
+
+sub process_packages {
+ my ($self, $packages, $repository, $target, $context) = @_;
+ croak "Not a class method" unless ref $self;
+
+ my $errors;
+
+ if (@$packages > 1) {
+ my $name = $packages->[0]->get_canonical_name();
+ $errors .= "not a canonical package set\n"
+ unless
+ all { $_->get_canonical_name() eq $name }
+ @$packages;
+ }
+
+ my $sources = grep { $_->is_source() } @$packages;
+
+ $errors .= "number of source packages > $self->{_max_sources}\n"
+ if defined $self->{_max_sources} && $sources > $self->{_max_sources};
+
+ $errors .= "number of source packages < $self->{_min_sources}\n"
+ if defined $self->{_min_sources} && $sources < $self->{_min_sources};
+
+ my $binaries = grep { $_->is_binary() } @$packages;
+
+ $errors .= "number of binary packages > $self->{_max_binaries}\n"
+ if defined $self->{_max_binaries} && $binaries > $self->{_max_binaries};
+
+ $errors .= "number of binary packages < $self->{_min_binaries}\n"
+ if defined $self->{_min_binaries} && $binaries < $self->{_min_binaries};
+
+ croak $errors if $errors;
+}
+
+=head1 COPYRIGHT AND LICENSE
+
+Copyright (C) 2002-2006, YOURI project
+
+This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
+
+=cut
+
+1;
diff --git a/lib/Youri/Submit/Test/Tag.pm b/lib/Youri/Submit/Test/Tag.pm
new file mode 100644
index 0000000..59342e4
--- /dev/null
+++ b/lib/Youri/Submit/Test/Tag.pm
@@ -0,0 +1,58 @@
+# $Id: Tag.pm 1687 2007-06-28 22:44:07Z guillomovitch $
+package Youri::Submit::Test::Tag;
+
+=head1 NAME
+
+Youri::Submit::Test::Tag - Incorrect tag values test
+
+=head1 DESCRIPTION
+
+This test plugin rejects packages with incorrect tag values, based on regular
+expressions.
+
+=cut
+
+use warnings;
+use strict;
+use Carp;
+use base qw/Youri::Submit::Step/;
+
+sub _init {
+ my $self = shift;
+ my %options = (
+ tags => undef, # expected tag values
+ @_
+ );
+
+ croak "no tags to test" unless $options{tags};
+ croak "tag should be an hashref" unless ref $options{tags} eq 'HASH';
+
+ $self->{_tags} = $options{tags};
+}
+
+sub process_packages {
+ my ($self, $packages, $repository, $target, $context) = @_;
+ croak "Not a class method" unless ref $self;
+
+ my $errors;
+
+ foreach my $package (@$packages) {
+ foreach my $tag (keys %{$self->{_tags}}) {
+ my $value = $package->get_tag($tag);
+ $errors .= "invalid value $value for tag $tag"
+ if $value !~ /$self->{_tags}->{$tag}/;
+ }
+ }
+
+ croak $errors if $errors;
+}
+
+=head1 COPYRIGHT AND LICENSE
+
+Copyright (C) 2002-2006, YOURI project
+
+This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
+
+=cut
+
+1;
diff --git a/t/Makefile.am b/t/Makefile.am
new file mode 100644
index 0000000..0fd009c
--- /dev/null
+++ b/t/Makefile.am
@@ -0,0 +1,8 @@
+TESTS = perlcritic.t \
+ install.t \
+ sign.t \
+ tags.t
+
+TESTS_ENVIRONMENT = perl -I $(top_srcdir)/lib
+
+EXTRA_DIST = $(TESTS) gpghome
diff --git a/t/gpghome/pubring.gpg b/t/gpghome/pubring.gpg
new file mode 100644
index 0000000..fcfc0e3
--- /dev/null
+++ b/t/gpghome/pubring.gpg
Binary files differ
diff --git a/t/gpghome/random_seed b/t/gpghome/random_seed
new file mode 100644
index 0000000..1b0201e
--- /dev/null
+++ b/t/gpghome/random_seed
Binary files differ
diff --git a/t/gpghome/secring.gpg b/t/gpghome/secring.gpg
new file mode 100644
index 0000000..9017e4d
--- /dev/null
+++ b/t/gpghome/secring.gpg
Binary files differ
diff --git a/t/gpghome/trustdb.gpg b/t/gpghome/trustdb.gpg
new file mode 100644
index 0000000..30712a3
--- /dev/null
+++ b/t/gpghome/trustdb.gpg
Binary files differ
diff --git a/t/install.t b/t/install.t
new file mode 100755
index 0000000..69113ff
--- /dev/null
+++ b/t/install.t
@@ -0,0 +1,51 @@
+#!/usr/bin/perl
+# $Id: install.t 1426 2006-12-17 21:00:25Z guillomovitch $
+
+use Test::More tests => 3;
+use Test::Exception;
+use Youri::Repository::Test;
+use Youri::Package::RPM::Generator;
+use Youri::Package::RPM::URPM;
+use Youri::Submit::Action::Install;
+
+my $writable_repository = Youri::Repository::Test->new(perms => '755');
+my $unwritable_repository = Youri::Repository::Test->new(perms => '000');
+
+my $action = Youri::Submit::Action::Install->new(
+ skip => [ 'cheater' ]
+);
+
+dies_ok {
+ $action->run(
+ Youri::Package::RPM::URPM->new(
+ file => Youri::Package::RPM::Generator->new()->get_source(),
+ ),
+ $unwritable_repository,
+ undef,
+ undef
+ )
+} 'installing in a non-writable directory';
+
+lives_ok {
+ $action->run(
+ Youri::Package::RPM::URPM->new(
+ file => Youri::Package::RPM::Generator->new(tags => {
+ name => 'cheater',
+ })->get_source(),
+ ),
+ $unwritable_repository,
+ undef,
+ undef
+ )
+} 'installing in a non-writable directory with an exception';
+
+lives_ok {
+ $action->run(
+ Youri::Package::RPM::URPM->new(
+ file => Youri::Package::RPM::Generator->new()->get_source(),
+ ),
+ $writable_repository,
+ undef,
+ undef
+ )
+} 'installing in a writable directory';
diff --git a/t/perlcritic.t b/t/perlcritic.t
new file mode 100755
index 0000000..e929713
--- /dev/null
+++ b/t/perlcritic.t
@@ -0,0 +1,32 @@
+#!/usr/bin/perl
+# $Id: perlcritic.t 1530 2007-03-08 20:42:13Z guillomovitch $
+
+use strict;
+use warnings;
+use Test::More;
+use File::Basename;
+use File::Spec;
+
+if (!$ENV{TEST_AUTHOR}) {
+ plan(
+ skip_all => 'Author test. Set $ENV{TEST_AUTHOR} to a true value to run.'
+ );
+}
+
+eval {
+ require Test::Perl::Critic;
+};
+
+if ($@) {
+ plan(
+ skip_all => 'Test::Perl::Critic not installed, skipping'
+ );
+}
+
+Test::Perl::Critic->import();
+my $libdir = File::Spec->catdir(
+ dirname($0),
+ File::Spec->updir(),
+ 'lib'
+);
+all_critic_ok($libdir);
diff --git a/t/set.t b/t/set.t
new file mode 100755
index 0000000..ae44e3a
--- /dev/null
+++ b/t/set.t
@@ -0,0 +1,124 @@
+#!/usr/bin/perl
+# $Id: /local/youri/soft/submit/trunk/t/tags.t 3062 2007-06-28T16:21:35.624871Z guillaume $
+
+use Test::More tests => 7;
+use Test::Exception;
+use Youri::Package::RPM::Test;
+use Youri::Submit::Test::Set;
+
+my $test = Youri::Submit::Test::Set->new(
+ skip => [ 'bar' ]
+);
+
+lives_ok {
+ $test->run(
+ undef,
+ undef,
+ undef,
+ [
+ Youri::Package::RPM::Test->new(tags => {
+ arch => 'noarch'
+ }),
+ Youri::Package::RPM::Test->new(tags => {
+ arch => 'src',
+ })
+ ]
+ )
+} 'one source and one binary package from the same source succeed';
+
+throws_ok {
+ $test->run(
+ undef,
+ undef,
+ undef,
+ [
+ Youri::Package::RPM::Test->new(tags => {
+ arch => 'src'
+ }),
+ Youri::Package::RPM::Test->new(tags => {
+ arch => 'noarch',
+ sourcerpm => 'other-1-1.src.rpm',
+ })
+ ]
+ )
+} qr/not a canonical package set/,
+'one source and one binary packages from different sources fails';
+
+throws_ok {
+ $test->run(
+ undef,
+ undef,
+ undef,
+ [
+ Youri::Package::RPM::Test->new(tags => {
+ arch => 'noarch'
+ })
+ ]
+ )
+} qr/number of source packages < 1/,
+'just one source binary fails';
+
+throws_ok {
+ $test->run(
+ undef,
+ undef,
+ undef,
+ [
+ Youri::Package::RPM::Test->new(tags => {
+ arch => 'src'
+ })
+ ]
+ )
+} qr/number of binary packages < 1/,
+'just one source binary fails';
+
+throws_ok {
+ $test->run(
+ undef,
+ undef,
+ undef,
+ [
+ Youri::Package::RPM::Test->new(tags => {
+ arch => 'src'
+ }),
+ Youri::Package::RPM::Test->new(tags => {
+ arch => 'src'
+ }),
+ ]
+ )
+} qr/number of source packages > 1/,
+'two source packages fails';
+
+lives_ok {
+ $test->run(
+ undef,
+ undef,
+ undef,
+ [
+ Youri::Package::RPM::Test->new(tags => {
+ arch => 'src'
+ }),
+ Youri::Package::RPM::Test->new(tags => {
+ arch => 'noarch'
+ }),
+ Youri::Package::RPM::Test->new(tags => {
+ arch => 'noarch'
+ })
+ ]
+ )
+}
+'one source and two binary packages succedd';
+
+lives_ok {
+ $test->run(
+ undef,
+ undef,
+ undef,
+ [
+ Youri::Package::RPM::Test->new(tags => {
+ name => 'bar',
+ release => '1mdk'
+ })
+ ]
+ )
+} 'non-compliant package with exception succeed';
diff --git a/t/sign.t b/t/sign.t
new file mode 100755
index 0000000..db6f619
--- /dev/null
+++ b/t/sign.t
@@ -0,0 +1,57 @@
+#!/usr/bin/perl
+# $Id: sign.t 1407 2006-12-03 12:02:02Z guillomovitch $
+
+use Test::More tests => 3;
+use Test::Exception;
+use File::Basename;
+use Youri::Package::RPM::Generator;
+use Youri::Package::RPM::URPM;
+use Youri::Submit::Action::Sign;
+
+my $action1 = Youri::Submit::Action::Sign->new(
+ name => 'Youri',
+ path => dirname($0) . '/gpghome',
+ passphrase => 'Youri sux',
+ skip => [ 'cheater' ]
+);
+
+dies_ok {
+ $action1->run(
+ Youri::Package::RPM::URPM->new(
+ file => Youri::Package::RPM::Generator->new()->get_source(),
+ ),
+ undef,
+ undef,
+ undef
+ )
+} 'signing with wrong key';
+
+lives_ok {
+ $action1->run(
+ Youri::Package::RPM::URPM->new(
+ file => Youri::Package::RPM::Generator->new(tags => {
+ name => 'cheater',
+ })->get_source(),
+ ),
+ undef,
+ undef,
+ undef
+ )
+} 'signing with wrong key using an exception';
+
+my $action2 = Youri::Submit::Action::Sign->new(
+ name => 'Youri',
+ path => dirname($0) . '/gpghome',
+ passphrase => 'Youri rulez',
+);
+
+lives_ok {
+ $action2->run(
+ Youri::Package::RPM::URPM->new(
+ file => Youri::Package::RPM::Generator->new()->get_source(),
+ ),
+ undef,
+ undef,
+ undef
+ )
+} 'signing with correct key';
diff --git a/t/tags.t b/t/tags.t
new file mode 100755
index 0000000..4946799
--- /dev/null
+++ b/t/tags.t
@@ -0,0 +1,55 @@
+#!/usr/bin/perl
+# $Id: tags.t 1687 2007-06-28 22:44:07Z guillomovitch $
+
+use Test::More tests => 3;
+use Test::Exception;
+use Youri::Package::RPM::Test;
+use Youri::Submit::Test::Tag;
+
+my $test = Youri::Submit::Test::Tag->new(
+ tags => { release => 'plf$' },
+ skip => [ 'bar' ]
+);
+
+lives_ok {
+ $test->run(
+ undef,
+ undef,
+ undef,
+ [
+ Youri::Package::RPM::Test->new(tags => {
+ name => 'foo',
+ release => '1plf'
+ })
+ ]
+ )
+} 'compliant package succeed';
+
+throws_ok {
+ $test->run(
+ undef,
+ undef,
+ undef,
+ [
+ Youri::Package::RPM::Test->new(tags => {
+ name => 'foo',
+ release => '1mdk'
+ })
+ ]
+ )
+} qr/invalid value 1mdk for tag release/,
+'non-compliant package fails';
+
+lives_ok {
+ $test->run(
+ undef,
+ undef,
+ undef,
+ [
+ Youri::Package::RPM::Test->new(tags => {
+ name => 'bar',
+ release => '1mdk'
+ })
+ ],
+ )
+} 'non-compliant package with exception succeed';