aboutsummaryrefslogtreecommitdiffstats
path: root/emi
blob: e7fabe5370c13a74a5aa80fb122343669c7bde92 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
#!/usr/bin/perl
#
# Copyright (C) 2005,2006 Mandriva
# 
# Author: Florent Villard <warly@mandriva.com>
#
# 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, 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, write to the Free Software
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
#
# upload packages in queue when all the mandatory architectures are done
#
# TODO
# 
# - take the packages in done/ and upload them with youri in queue/
# - check that the mandatory architectures are present
#
# PREFIX : sprintf "$year%02d%02d%02d%02d%02d.$user.$host.${$}_", $mon, $mday, $hour, $min, $sec;

use strict;
use Iurt::Config qw(config_usage get_date config_init dump_cache init_cache get_repsys_conf);
use Iurt::Process qw(check_pid);
use Iurt::Mail qw(sendmail);
use Iurt::File qw(check_upload_tree);
use Iurt::Util qw(plog_init plog);
use Data::Dumper;
my %run;
my $program_name = 'emi';
$run{program_name} = $program_name;

open(my $LOG, ">&STDERR");
plog_init($program_name, $LOG);
$run{LOG} = $LOG;
my $HOME = $ENV{HOME};

my $configfile = "$HOME/.upload.conf";

my %config_usage = ( 
    admin => { desc => 'mail address of the bot administrator', default => 'warly@mandriva.com' },
    arch => {  
	desc => "List of arch",
	default => [ 'i586', 'x86_64', 'ppc' , 'sparc', 'sparc64' ],
         },
   default_mail_domain => { desc => "Default mail domain to append", default => 'mandriva.org' },
    http_queue => { desc => 'Address where log can be consulted', default => 'http://svn.mandriva.com/queue/'},
    mandatory_arch => {
	desc => 'List of mandatory architecture to be able to upload',
	default => [ 'i586', 'x86_64' ],
      },
   tmp => { desc => "Temporary directory",  default => "$HOME/tmp/"},
   root => { desc => 'Architecture root dir', default => "/mnt/BIG/dis/" },
   cache_home => { desc => 'Where to store the cache files', default => "$HOME/.bugs" },
   upload_user => { desc => 'User who is uploading packages', default => 'mandrake' },
   repsys_conf => { desc => 'Path of the repsys.conf file which includes login mail corresponding', default => '/etc/repsys.conf' },
   queue => { desc => 'root directory of the various upload queues', default => '/home/mandrake/uploads/'},
   ssh_option => { desc => "SSH options", default => "-o ConectTimeout=20" },
);

my $config;
if (-f $configfile) {
    $config = do $configfile or die "FATAL $program_name: syntax error in $configfile";
} else {
    $config = {}
}

config_usage(\%config_usage, $config) if $run{config_usage};

config_init(\%config_usage, $config, \%run);

my $mail = get_repsys_conf($config->{repsys_conf});

$run{pidfile_home} = $config->{tmp};
$run{pidfile} = "upload";
my $pidfile = check_pid(\%run);

#my $cache = init_cache(\%run, $config, { arch => {} });
my $cache =  { arch => {} };

my ($fulldate, $daydate) = get_date();
$run{daydate} = $daydate;
print {$run{LOG}} "$program_name\n";

my $todo = "$config->{queue}/todo/";
my $done = "$config->{queue}/done/";
my $reject = "$config->{queue}/rejected/";

my %pkg_tree;
my %excluded;
check_upload_tree(\%run, $done, 
    sub {
	my ($todo, $f, $m, $s, $r) = @_;
	if ($r =~ /(\d{14}\.\w+\.\w+\.\d+)_(.*\.([^.]+)\.rpm)$/) {
	    my ($prefix, $rpm, $arch) = ($1, $2, $3);
	    print {$run{LOG}} "$program_name: found rpm $rpm ($prefix)\n";
	    $pkg_tree{$prefix}{path} = "/$f/$m/$s";
	    $pkg_tree{$prefix}{arch}{$arch} = 1;
	    $pkg_tree{$prefix}{target} = $f;
	    $pkg_tree{$prefix}{section} = "$m/$s";
	    push @{$pkg_tree{$prefix}{srpms}}, $rpm if ($arch eq 'src');
	    push @{$pkg_tree{$prefix}{rpms}} , $rpm;
	} elsif ($r =~ /(\d{14}\.\w+\.\w+\.\d+)_(.*)\.done$/) {
	    my ($prefix, $arch) = ($1, $2);
	    $cache->{arch}{$prefix}{$arch} = 1
	} elsif ($r =~ /(\d{14}\.\w+\.\w+\.\d+)_(.*)\.excluded$/) {
	    my ($prefix, $arch) = ($1, $2);
	    $cache->{arch}{$prefix}{$arch} = 1;
	    $excluded{$prefix}{$arch} = 1
	}

    },
);

check_upload_tree(\%run, $todo, 
    sub {
	my ($todo, $f, $m, $s, $r) = @_;
	if ($r =~ /(\d{14}\.\w+\.\w+\.\d+)_(.*\.([^.]+)\.rpm)$/) {
	    my ($prefix, $rpm, $arch) = ($1, $2, $3);
	    print {$run{LOG}} "$program_name: found todo rpm $rpm ($prefix)\n";
	    push @{$pkg_tree{$prefix}{todo}}, $rpm
	}
    },
);

foreach my $prefix (keys %pkg_tree) {
    my $ok = 1;
    my $target = $pkg_tree{$prefix}{target};
    my $path = $pkg_tree{$prefix}{path};
    my $section = $pkg_tree{$prefix}{section};
    print {$run{LOG}} "$program_name: processing $prefix in $path\n";
    my %missing;
    foreach my $mandatory_arch (@{$config->{mandatory_arch}}, 'src') {
	$excluded{$prefix}{$mandatory_arch} and next;
	print {$run{LOG}} "$program_name: checking if mandatory architecture $mandatory_arch is present: ";
	if (!$pkg_tree{$prefix}{arch}{$mandatory_arch} && !$pkg_tree{$prefix}{arch}{noarch}){
	    if (!$cache->{arch}{$prefix}{$mandatory_arch}) {
		$missing{$mandatory_arch} = 1;
		print {$run{LOG}} "no\n";
		$ok = 0
	    } else {
		print {$run{LOG}} "yes (in cache)\n"
	    }
	} else {
	    print {$run{LOG}} "yes\n"
	}
    }
    if ($ok) {
	# upload
	my @packages;
	my ($user) = $prefix =~ /\d{14}\.(\w+)\.\w+\.\d+$/;
	foreach my $rpm (@{$pkg_tree{$prefix}{rpms}}) {
	    push @packages, "$done/$path/${prefix}_$rpm"
	}
	$user ||= $config->{upload_user};
	my $command = "/usr/bin/sudo -H -u $config->{upload_user} /usr/bin/perl -I/usr/local/lib/perl/ /usr/local/bin/youri-submit --verbose --config /etc/youri/submit-queue.conf --define user=$user --define prefix=$prefix --define section=$section $target @packages &> $done/$path/$prefix.youri";
	print {$run{LOG}} "$program_name: running $command\n";
	if (!system($command)) {
	    print {$run{LOG}} "$program_name: upload succeeded.\n";
	    # now check if we need to keep the current srpm in todo
	    my $all_uploaded = 1;
	    foreach my $arch (@{$config->{arch}}) {
		if (!$cache->{arch}{$prefix}{$arch}) {
		    $all_uploaded = 0
		}
	    }
	    if ($all_uploaded) {
		print {$run{LOG}} "$program_name: cleaning upload tree for $prefix\n";
		# remove srpm
		# remove lock
	    }
	} else {
	    # should send a mail or something
	    print {$run{LOG}} "$program_name: upload failed ($!), rejecting files in $reject/$path/\n";
	    foreach my $rpm (@{$pkg_tree{$prefix}{rpms}}) {
		my $err = link "$done/$path/${prefix}_$rpm", "$reject/$path/${prefix}_$rpm";
		print {$run{LOG}} "ERROR $program_name: link of $rpm failed ($!)\n"
	    }
	    link "$done/$path/$prefix.youri", "$reject/$path/$prefix.youri";
	    my ($user) = $prefix =~ /\d{14}\.(\w+)\.\w+\.\d+/;
	    if ($user) {
		my $text = qq{The upload of the following packages failed:\n};
		my $rpms;
		foreach my $rpm (@{$pkg_tree{$prefix}{rpms}}) {
		    $rpm =~ /src\.rpm$/ or next;
		    $rpms .= "$rpm ";
		    $text .= "- $rpm\n"
		}
		my $to =  $mail->{$user} || "$user\@mandriva.com, $user\@mandriva.org";
		$text .= "\nUpload log available in $config->{http_queue}/rejected/$path/$prefix.youri\n";
		sendmail($to, $config->{admin} , "Upload failed for $rpms", $text, "Emi the upload bot <$config->{admin}>", 0)
	    }
	    # should delete the files
	}
	# delete the files which should have heen either put in queue or rejected
	unlink $_ foreach @packages;
	# keep the log file for debugging
	# unlink "$done/$path/$prefix.youri";
	# unlink the sources rpm, other arch will be able to grab it into the repository
	print "unlink $todo/$path/${prefix}_$_\n" foreach @{$pkg_tree{$prefix}{todo}};
	unlink "$todo/$path/${prefix}_$_" foreach @{$pkg_tree{$prefix}{todo}}
    } else {
	print {$run{LOG}} "WARNING $program_name: mandatory arch (", join(' ', keys %missing),") are missing for $prefix, waiting\n"
    }
}

#dump_cache(\%run);
unlink $pidfile;
exit;