1 | #!/usr/bin/perl -w
|
---|
2 | #
|
---|
3 | # Project Builder main application
|
---|
4 | #
|
---|
5 | # $Id$
|
---|
6 | #
|
---|
7 | # Copyright B. Cornec 2007-2016
|
---|
8 | # Eric Anderson's changes are (c) Copyright 2012 Hewlett Packard
|
---|
9 | # Provided under the GPL v2
|
---|
10 |
|
---|
11 | # Syntax: see at end
|
---|
12 |
|
---|
13 | use strict 'vars';
|
---|
14 |
|
---|
15 | # The modules mentioned here are required by pb when used both
|
---|
16 | # locally or inside a VE/VM/RM
|
---|
17 | # Additional required modules only used locally are called with a require
|
---|
18 | # in their respective section
|
---|
19 | use Getopt::Long qw(:config auto_abbrev no_ignore_case);
|
---|
20 | use Carp qw/confess cluck/;
|
---|
21 | use Data::Dumper;
|
---|
22 | use English;
|
---|
23 | use File::Basename;
|
---|
24 | use File::Copy;
|
---|
25 | use File::stat;
|
---|
26 | use File::Find;
|
---|
27 | use Time::localtime qw(localtime);
|
---|
28 | use POSIX qw(strftime);
|
---|
29 | use lib qw (lib);
|
---|
30 | use ProjectBuilder::Version;
|
---|
31 | use ProjectBuilder::Base;
|
---|
32 | use ProjectBuilder::Display;
|
---|
33 | use ProjectBuilder::Conf;
|
---|
34 | use ProjectBuilder::Distribution;
|
---|
35 | use ProjectBuilder::VCS;
|
---|
36 | use ProjectBuilder::CMS;
|
---|
37 | use ProjectBuilder::Env;
|
---|
38 | use ProjectBuilder::Filter;
|
---|
39 | use ProjectBuilder::Changelog;
|
---|
40 | use ProjectBuilder::VE;
|
---|
41 |
|
---|
42 | # Global variables
|
---|
43 | $Global::pb_stop_on_error = 0; # False by default
|
---|
44 | $Global::pb_show_sudo = 0; # False by default
|
---|
45 |
|
---|
46 | my %opts; # CLI Options
|
---|
47 | my $action; # action to realize
|
---|
48 | my $test = "FALSE"; # Not used
|
---|
49 | my $pbforce = 0; # Force VM stop
|
---|
50 | my $pbsnap = 0; # Do not use snapshot mode for VM/VE by default
|
---|
51 | my $pbkeep = 0; # keep temporary directory at the end
|
---|
52 | my $option = ""; # Not used
|
---|
53 | my @pkgs; # list of packages
|
---|
54 | my $pbtag; # Global Tag variable
|
---|
55 | my $pbver; # Global Version variable
|
---|
56 | my %pbscript; # Name of the scripts per V
|
---|
57 | my $pbrev; # Global REVISION variable
|
---|
58 | my $pbtarget = undef; # Target os-ver-arch you want to build for
|
---|
59 | my $pbport; # Port to use to connect to the VM/RM
|
---|
60 | my $newver; # New version to create
|
---|
61 | my $pbimage = undef; # ISO image for the VM to create or docker image for the VE
|
---|
62 | my $vetype = undef; # Type of Virtual environment to deal with
|
---|
63 |
|
---|
64 | my @date = pb_get_date();
|
---|
65 | my $pbdate = strftime("%Y-%m-%d", @date);
|
---|
66 |
|
---|
67 | =pod
|
---|
68 |
|
---|
69 | =head1 NAME
|
---|
70 |
|
---|
71 | pb, aka project-builder.org - builds packages for your projects
|
---|
72 |
|
---|
73 | =head1 DESCRIPTION
|
---|
74 |
|
---|
75 | pb helps you build various packages directly from your project sources.
|
---|
76 | Those sources could be handled by a CMS (Configuration Management System)
|
---|
77 | such as Subversion, CVS, Git, Mercurial... or being a simple reference to a compressed tar file.
|
---|
78 | It's based on a set of configuration files, a set of provided macros to help
|
---|
79 | you keeping build files as generic as possible. For example, a single .spec
|
---|
80 | file should be required to generate for all rpm based distributions, even
|
---|
81 | if you could also have multiple .spec files if required.
|
---|
82 |
|
---|
83 | =head1 SYNOPSIS
|
---|
84 |
|
---|
85 | pb [-vhSq][-r pbroot][-p project][[-s script -a account -P port][-T VEtype][-t [os-ver-arch]][-m os-ver-arch[,...]]][-g][-i image] <action> [<pkg1> ...]
|
---|
86 |
|
---|
87 | pb [--verbose][--help][--man][--quiet][--snapshot][--revision pbroot][--project project][[--script script --account account --port port][-T VEtype][--target [os-ver-arch]][--machine os-ver-arch[,...]]][--nographic][--image image][--rebuild] <action> [<pkg1> ...]
|
---|
88 |
|
---|
89 | =head1 OPTIONS
|
---|
90 |
|
---|
91 | =over 4
|
---|
92 |
|
---|
93 | =item B<-v|--verbose>
|
---|
94 |
|
---|
95 | Increase verbosity
|
---|
96 |
|
---|
97 | =item B<-q|--quiet>
|
---|
98 |
|
---|
99 | Do not print any output.
|
---|
100 |
|
---|
101 | =item B<-h|--help>
|
---|
102 |
|
---|
103 | Print a brief help message and exits.
|
---|
104 |
|
---|
105 | =item B<-S|--snapshot>
|
---|
106 |
|
---|
107 | Use the snapshot mode of VMs or VEs
|
---|
108 |
|
---|
109 | =item B<--man>
|
---|
110 |
|
---|
111 | Prints the manual page and exits.
|
---|
112 |
|
---|
113 | =item B<-t|--target os-ver-arch>
|
---|
114 |
|
---|
115 | Name of the target system you want to build for.
|
---|
116 | All if none precised.
|
---|
117 |
|
---|
118 | =item B<-m|--machine os-ver-arch[,os-ver-arch,...]>
|
---|
119 |
|
---|
120 | Name of the Virtual Machines (VM), Virtual Environments (VE) or Remote Machines (RM)
|
---|
121 | you want to build on (comma separated).
|
---|
122 | All if none precised (or use the env variable PBV).
|
---|
123 |
|
---|
124 | =item B<-T|--vetype VEtype]>
|
---|
125 |
|
---|
126 | Type of Virtual Environments (VE)
|
---|
127 | Can be chroot or docker.
|
---|
128 |
|
---|
129 | =item B<-s|--script script>
|
---|
130 |
|
---|
131 | Name of the script you want to execute on the related VMs/VEs/RMs.
|
---|
132 |
|
---|
133 | =item B<-g|--nographic>
|
---|
134 |
|
---|
135 | Do not launch VMs in graphical mode.
|
---|
136 |
|
---|
137 | =item B<-i|--image image>
|
---|
138 |
|
---|
139 | It could be either:
|
---|
140 | - The name of the ISO image of the distribution you want to install on the related VMs
|
---|
141 | - The name of the docker image of the distribution you want to install on the related VEs
|
---|
142 |
|
---|
143 |
|
---|
144 | =item B<-a|--account account>
|
---|
145 |
|
---|
146 | Name of the account to use to connect on the related VMs/RMs.
|
---|
147 |
|
---|
148 | =item B<-P|--port port_number>
|
---|
149 |
|
---|
150 | Port number to use to connect on the related VMs/RMs.";
|
---|
151 |
|
---|
152 | =item B<-p|--project project_name>
|
---|
153 |
|
---|
154 | Name of the project you're working on (or use the env variable PBPROJ)
|
---|
155 |
|
---|
156 | =item B<-r|--revision revision>
|
---|
157 |
|
---|
158 | Path Name of the project revision under the CMS (or use the env variable PBROOT)
|
---|
159 |
|
---|
160 | =item B<-V|--version new_version>
|
---|
161 |
|
---|
162 | New version of the project to create based on the current one.
|
---|
163 |
|
---|
164 | =item B<-k|--keep>
|
---|
165 |
|
---|
166 | Keep the temporary dir where files have been created in or der to help debug
|
---|
167 |
|
---|
168 | =item B<--rebuild>
|
---|
169 |
|
---|
170 | Only valid with the checkssh action, it alllows to automatically relaunch the build of the failed packages
|
---|
171 |
|
---|
172 | =item B<--no-stop-on-error>
|
---|
173 |
|
---|
174 | Continue through errors with best effort.
|
---|
175 |
|
---|
176 | =back
|
---|
177 |
|
---|
178 | =head1 ARGUMENTS
|
---|
179 |
|
---|
180 | <action> can be:
|
---|
181 |
|
---|
182 | =over 4
|
---|
183 |
|
---|
184 | =item B<sbx2build>
|
---|
185 |
|
---|
186 | Create tar files for the project under your CMS.
|
---|
187 | Current state of the exported content is taken.
|
---|
188 | CMS supported are SVN, SVK, CVS, Git and Mercurial
|
---|
189 | parameters are packages to build
|
---|
190 | if not using default list
|
---|
191 |
|
---|
192 | =item B<cms2build>
|
---|
193 |
|
---|
194 | Create tar files for the project under your CMS.
|
---|
195 | Current state of the CMS is taken.
|
---|
196 | CMS supported are SVN, SVK, CVS, Git and Mercurial
|
---|
197 | parameters are packages to build
|
---|
198 | if not using default list
|
---|
199 |
|
---|
200 | =item B<build2prep>
|
---|
201 |
|
---|
202 | Prepare the environment for build by installing required dependencies. Done once on the build system.
|
---|
203 |
|
---|
204 | =item B<build2pkg>
|
---|
205 |
|
---|
206 | Create packages for your running distribution
|
---|
207 |
|
---|
208 | =item B<cms2pkg>
|
---|
209 |
|
---|
210 | cms2build + build2pkg
|
---|
211 |
|
---|
212 | =item B<sbx2pkg>
|
---|
213 |
|
---|
214 | sbx2build + build2pkg
|
---|
215 |
|
---|
216 | =item B<sbx2pkg2ins>
|
---|
217 |
|
---|
218 | sbx2pkg + final install of packages
|
---|
219 |
|
---|
220 | =item B<build2ssh>
|
---|
221 |
|
---|
222 | Send the tar files to a SSH host
|
---|
223 |
|
---|
224 | =item B<sbx2ssh>
|
---|
225 |
|
---|
226 | sbx2build + build2ssh
|
---|
227 |
|
---|
228 | =item B<cms2ssh>
|
---|
229 |
|
---|
230 | cms2build + build2ssh
|
---|
231 |
|
---|
232 | =item B<pkg2ssh>
|
---|
233 |
|
---|
234 | Send the packages built to a SSH host
|
---|
235 |
|
---|
236 | =item B<build2vm>
|
---|
237 |
|
---|
238 | Create packages in VMs, launching them if needed
|
---|
239 | and send those packages to a SSH host once built
|
---|
240 | VM type supported are QEMU and KVM
|
---|
241 |
|
---|
242 | =item B<build2ve>
|
---|
243 |
|
---|
244 | Create packages in VEs, creating it if needed
|
---|
245 | and send those packages to a SSH host once built
|
---|
246 |
|
---|
247 | =item B<build2rm>
|
---|
248 |
|
---|
249 | Create packages in RMs, which should pre-exist,
|
---|
250 | and send those packages to a SSH host once built
|
---|
251 | RM means Remote Machine, and could be a physical or Virtual one.
|
---|
252 | This is one buildfarm integration for pb.
|
---|
253 |
|
---|
254 | =item B<prepvm>
|
---|
255 |
|
---|
256 | Prepare the VMs to have all requirements to build the project
|
---|
257 |
|
---|
258 | =item B<prepve>
|
---|
259 |
|
---|
260 | Prepare the VEs to have all requirements to build the project
|
---|
261 |
|
---|
262 | =item B<preprm>
|
---|
263 |
|
---|
264 | Prepare the RMs to have all requirements to build the project
|
---|
265 |
|
---|
266 | =item B<sbx2vm>
|
---|
267 |
|
---|
268 | sbx2build + build2vm
|
---|
269 |
|
---|
270 | =item B<sbx2ve>
|
---|
271 |
|
---|
272 | sbx2build + build2ve
|
---|
273 |
|
---|
274 | =item B<sbx2docker>
|
---|
275 |
|
---|
276 | sbx2build + build2ve with a potential build of all necessary docker containers to perform it
|
---|
277 |
|
---|
278 | =item B<sbx2rm>
|
---|
279 |
|
---|
280 | sbx2build + build2rm
|
---|
281 |
|
---|
282 | =item B<cms2vm>
|
---|
283 |
|
---|
284 | cms2build + build2vm
|
---|
285 |
|
---|
286 | =item B<cms2ve>
|
---|
287 |
|
---|
288 | cms2build + build2ve
|
---|
289 |
|
---|
290 | =item B<cms2rm>
|
---|
291 |
|
---|
292 | cms2build + build2rm
|
---|
293 |
|
---|
294 | =item B<launchvm>
|
---|
295 |
|
---|
296 | Launch one virtual machine
|
---|
297 |
|
---|
298 | =item B<launchve>
|
---|
299 |
|
---|
300 | Launch one virtual environment
|
---|
301 |
|
---|
302 | =item B<script2vm>
|
---|
303 |
|
---|
304 | Launch one virtual machine if needed
|
---|
305 | and executes a script on it
|
---|
306 |
|
---|
307 | =item B<script2ve>
|
---|
308 |
|
---|
309 | Execute a script in a virtual environment
|
---|
310 |
|
---|
311 | =item B<script2rm>
|
---|
312 |
|
---|
313 | Execute a script on a remote machine
|
---|
314 |
|
---|
315 | =item B<newvm>
|
---|
316 |
|
---|
317 | Create a new virtual machine
|
---|
318 |
|
---|
319 | =item B<newve>
|
---|
320 |
|
---|
321 | Create a new virtual environment
|
---|
322 |
|
---|
323 | =item B<setupvm>
|
---|
324 |
|
---|
325 | Setup a virtual machine for pb usage
|
---|
326 |
|
---|
327 | =item B<setupve>
|
---|
328 |
|
---|
329 | Setup a virtual environment for pb usage
|
---|
330 |
|
---|
331 | =item B<setuprm>
|
---|
332 |
|
---|
333 | Setup a remote machine for pb usage
|
---|
334 |
|
---|
335 | =item B<sbx2setupvm>
|
---|
336 |
|
---|
337 | Setup a virtual machine for pb usage using the sandbox version of pb instead of the latest stable
|
---|
338 | Reserved to dev team.
|
---|
339 |
|
---|
340 | =item B<sbx2setupve>
|
---|
341 |
|
---|
342 | Setup a virtual environment for pb usage using the sandbox version of pb instead of the latest stable
|
---|
343 | Reserved to dev team.
|
---|
344 |
|
---|
345 | =item B<sbx2setuprm>
|
---|
346 |
|
---|
347 | Setup a remote machine for pb usage using the sandbox version of pb instead of the latest stable
|
---|
348 | Reserved to dev team.
|
---|
349 |
|
---|
350 | =item B<build2setupvm>
|
---|
351 |
|
---|
352 | Setup a virtual machine for pb usage using the build available
|
---|
353 | Reserved to dev team.
|
---|
354 |
|
---|
355 | =item B<build2setupve>
|
---|
356 |
|
---|
357 | Setup a virtual environment for pb usage using the build available
|
---|
358 | Reserved to dev team.
|
---|
359 |
|
---|
360 | =item B<build2setuprm>
|
---|
361 |
|
---|
362 | Setup a remote machine for pb usage using the build available
|
---|
363 | Reserved to dev team.
|
---|
364 |
|
---|
365 | =item B<snapvm>
|
---|
366 |
|
---|
367 | Snapshot a virtual machine for pb usage
|
---|
368 |
|
---|
369 | =item B<snapve>
|
---|
370 |
|
---|
371 | Snapshot a virtual environment for pb usage
|
---|
372 |
|
---|
373 | =item B<updatevm>
|
---|
374 |
|
---|
375 | Update the distribution in the virtual machine
|
---|
376 |
|
---|
377 | =item B<updateve>
|
---|
378 |
|
---|
379 | Update the distribution in the virtual environment
|
---|
380 |
|
---|
381 | =item B<updaterm>
|
---|
382 |
|
---|
383 | Update the distribution in the remote machine
|
---|
384 |
|
---|
385 | =item B<test2pkg>
|
---|
386 |
|
---|
387 | Test a package locally
|
---|
388 |
|
---|
389 | =item B<test2vm>
|
---|
390 |
|
---|
391 | Test a package in a virtual machine
|
---|
392 |
|
---|
393 | =item B<test2ve>
|
---|
394 |
|
---|
395 | Test a package in a virtual environment
|
---|
396 |
|
---|
397 | =item B<test2rm>
|
---|
398 |
|
---|
399 | Test a package in a remote machine
|
---|
400 |
|
---|
401 | =item B<checkssh>
|
---|
402 |
|
---|
403 | Check the delivery of the packages on the repository
|
---|
404 |
|
---|
405 | =item B<checkps>
|
---|
406 |
|
---|
407 | Check the process running the VM concerned
|
---|
408 |
|
---|
409 | =item B<newver>
|
---|
410 |
|
---|
411 | Create a new version of the project derived
|
---|
412 | from the current one
|
---|
413 |
|
---|
414 | =item B<newproj>
|
---|
415 |
|
---|
416 | Create a new project and a template set of
|
---|
417 | configuration files under pbconf
|
---|
418 |
|
---|
419 | =item B<announce>
|
---|
420 |
|
---|
421 | Announce the availability of the project through various means
|
---|
422 |
|
---|
423 | =item B<sbx2webssh>
|
---|
424 |
|
---|
425 | Create tar files for the website under your CMS.
|
---|
426 | Current state of the exported content is taken.
|
---|
427 | Deliver the content to the target server using ssh from the exported dir.
|
---|
428 |
|
---|
429 | =item B<cms2webssh>
|
---|
430 |
|
---|
431 | Create tar files for the website from your CMS.
|
---|
432 | Deliver the content to the target server using ssh from the DVCS.
|
---|
433 |
|
---|
434 | =item B<sbx2webpkg>
|
---|
435 |
|
---|
436 | Create tar files for the website under your CMS.
|
---|
437 | Current state of the exported content is taken.
|
---|
438 |
|
---|
439 | =item B<cms2webpkg>
|
---|
440 |
|
---|
441 | Create tar files for the website under your CMS.
|
---|
442 |
|
---|
443 | =item B<getconf>
|
---|
444 |
|
---|
445 | Print the full configuration parameters as found in the various configuration files. Help to debug conf issues.
|
---|
446 | Also accepts a parameter to display only this value, and a VM/VE/RM
|
---|
447 |
|
---|
448 | =item B<getvar>
|
---|
449 |
|
---|
450 | Print the full variables expanded based on the distrubution tuple. Help to debug conf issues.
|
---|
451 | Also accepts a parameter to display only the values for this package, and a VM/VE/RM
|
---|
452 |
|
---|
453 | =item B<clean>
|
---|
454 |
|
---|
455 | Purge the build and delivery directories related to the current project
|
---|
456 |
|
---|
457 | =item B<cleanssh>
|
---|
458 |
|
---|
459 | Purge the ssh server of its packages (only for testver and test packages)
|
---|
460 |
|
---|
461 | =back
|
---|
462 |
|
---|
463 | <pkgs> can be a list of packages, the keyword 'all' or nothing, in which case the default list of packages is taken (corresponding to the defpkgdir list of arguments in the configuration file).
|
---|
464 |
|
---|
465 | =head1 WEB SITES
|
---|
466 |
|
---|
467 | The main Web site of the project is available at L<http://www.project-builder.org/>. Bug reports should be filled using the trac instance of the project at L<http://trac.project-builder.org/>.
|
---|
468 |
|
---|
469 | =head1 USER MAILING LIST
|
---|
470 |
|
---|
471 | None exists for the moment.
|
---|
472 |
|
---|
473 | =head1 ENVIRONMENT VARIABLES
|
---|
474 |
|
---|
475 | =item B<PBACCOUNT>
|
---|
476 |
|
---|
477 | Login to use to connect to the VM/RM, undef by default
|
---|
478 |
|
---|
479 | =back
|
---|
480 |
|
---|
481 | =head1 CONFIGURATION FILES
|
---|
482 |
|
---|
483 | Each pb user may have a configuration in F<$HOME/.pbrc>. The values in this file may overwrite any other configuration file value.
|
---|
484 |
|
---|
485 | Here is an example of such a configuration file:
|
---|
486 |
|
---|
487 | #
|
---|
488 | # Define for each project the URL of its pbconf repository
|
---|
489 | # No default option allowed here as they need to be all different
|
---|
490 | #
|
---|
491 | # URL of the pbconf content
|
---|
492 | # This is the format of a classical URL with the extension of additional schema such as
|
---|
493 | # svn+ssh, cvs+ssh, ...
|
---|
494 | #
|
---|
495 | pbconfurl linuxcoe = cvs+ssh://:ext:bcornec@linuxcoe.cvs.sourceforge.net:/cvsroot/linuxcoe/pbconf
|
---|
496 |
|
---|
497 | # This is normaly defined in the project's configuration file
|
---|
498 | # Url of the project
|
---|
499 | #
|
---|
500 | pburl linuxcoe = cvs+ssh://:ext:bcornec@linuxcoe.cvs.sourceforge.net:/cvsroot/linuxcoe
|
---|
501 |
|
---|
502 | # All these URLs needs to be defined here as the are the entry point
|
---|
503 | # for how to build packages for the project
|
---|
504 | #
|
---|
505 | pbconfurl pb = svn+ssh://svn.project-builder.org/mondo/svn/pb/pbconf
|
---|
506 | pbconfurl mondorescue = svn+ssh://svn.project-builder.org/mondo/svn/project-builder/mondorescue/pbconf
|
---|
507 | pbconfurl collectl = svn+ssh://bruno@svn.mondorescue.org/mondo/svn/project-builder/collectl/pbconf
|
---|
508 | pbconfurl netperf = svn+ssh://svn.mondorescue.org/mondo/svn/project-builder/netperf/pbconf
|
---|
509 |
|
---|
510 | # Under that dir will take place everything related to pb
|
---|
511 | # If you want to use VMs/chroot/..., then use $ENV{'HOME'} to make it portable
|
---|
512 | # to your VMs/chroot/...
|
---|
513 | # if not defined then /var/cache
|
---|
514 | pbdefdir default = $ENV{'HOME'}/project-builder
|
---|
515 | pbdefdir pb = $ENV{'HOME'}
|
---|
516 | pbdefdir linuxcoe = $ENV{'HOME'}/LinuxCOE/cvs
|
---|
517 | pbdefdir mondorescue = $ENV{'HOME'}/mondo/svn
|
---|
518 |
|
---|
519 | # pbconfdir points to the directory where the CMS content of the pbconfurl is checked out
|
---|
520 | # If not defined, pbconfdir is under pbdefdir/pbproj/pbconf
|
---|
521 | pbconfdir linuxcoe = $ENV{'HOME'}/LinuxCOE/cvs/pbconf
|
---|
522 | pbconfdir mondorescue = $ENV{'HOME'}/mondo/svn/pbconf
|
---|
523 |
|
---|
524 | # pbdir points to the directory where the CMS content of the pburl is checked out
|
---|
525 | # If not defined, pbdir is under pbdefdir/pbproj
|
---|
526 | # Only defined if we have access to the dev of the project
|
---|
527 | pbdir linuxcoe = $ENV{'HOME'}/LinuxCOE/cvs
|
---|
528 | pbdir mondorescue = $ENV{'HOME'}/mondo/svn
|
---|
529 |
|
---|
530 | # -daemonize doesn't work with qemu 0.8.2
|
---|
531 | vmopt default = -m 384
|
---|
532 |
|
---|
533 | =head1 COMMAND DETAILS
|
---|
534 |
|
---|
535 | =head2 newproj
|
---|
536 |
|
---|
537 | The newproj command creates a new project-builder project. To run this command you first need to define two variables in your ~/.pbrc file:
|
---|
538 |
|
---|
539 | pbconfurl I<project> = file:///home/anderse/.git/project-builder-config/I<project>
|
---|
540 | pbdefdir default = $ENV{HOME}/cache-project-builder
|
---|
541 |
|
---|
542 | The first line defines the version controlled configuration information and the second defines the root directory for project-builder to use.
|
---|
543 |
|
---|
544 | You can then run the command:
|
---|
545 |
|
---|
546 | % pb -p I<$project> -r I<$version> newproj I<$pkg>
|
---|
547 |
|
---|
548 | to create the new project. Running the newproj command will then generate the file $pbdefdir/$project/pbconf/$version/$project.pb, and the directory $pbdefdir/$project/pbconf/$version/$pkg. You will need to edit those files to make the later commands work.
|
---|
549 |
|
---|
550 | =head2 cms2build
|
---|
551 |
|
---|
552 | The cms2build command takes your files from the content management system and makes the two tar files that are necessary for building files. You need to have run the newproj command first. Then there are several steps for running this command:
|
---|
553 |
|
---|
554 | =over 4
|
---|
555 |
|
---|
556 | =item Update your $project.pb configuration file.
|
---|
557 |
|
---|
558 | You need to set the pburl, pbrepo, pbwf, pbpackager, projver, projtag, testver, delivery, and defpkgdir lines as described in the configuration file. The pburl entry is used to find the source for your package. The pbrepo entry is used to build the .repo or .sources.list files for use by downloaders of the package. The pbwf entry indicates that the source tar file is named by I<package-name>-I<version>. The pbpackager entry will be stored in the packages and should be you or your team. The projver/projtag entries indicate the version of the software and the version of the packaging scripts. The testver entry when true indicates that the package is in a test version, so no log file is computed (can be long), and version is made up using a timstamp. The delivery entry gives the subdirectory under which the packages will be delivered on the repository, and the defpkgdir entry corresponds to the local subdirectory hosting the package content.
|
---|
559 |
|
---|
560 | For example:
|
---|
561 |
|
---|
562 | pburl Lintel = file:///home/anderse/projects/Lintel-0.2012.02.28.tar.gz
|
---|
563 | pbrepo Lintel = http://tesla.hpl.hp.com/opensource
|
---|
564 | pbwf Lintel = 1
|
---|
565 | pbpackager Lintel = Eric Anderson <eric.anderson4@hp.com>
|
---|
566 | projver Lintel = 0.2012.02.28
|
---|
567 | projtag Lintel = 1
|
---|
568 | testver Lintel = false
|
---|
569 | delivery Lintel = production
|
---|
570 | defpkgdir Lintel = Lintel-0.2012.02.28
|
---|
571 |
|
---|
572 | =item Create the build .tar.gz files:
|
---|
573 |
|
---|
574 | Then you need to take those files and create the initial tar files. Run a command like:
|
---|
575 |
|
---|
576 | % pb -p $project -r $version cms2build
|
---|
577 |
|
---|
578 | To create the $pbdefdir/$project/pbdelivery/$project-$version.{,pbconf}.tar.gz files, the $version-$projtag.pb and pbrc files in the same directory.
|
---|
579 |
|
---|
580 | =back
|
---|
581 |
|
---|
582 | =head2 build2pkg
|
---|
583 |
|
---|
584 | The build2pkg command takes the tar files created in the cms2build step and attempts to build binary packages for your current operating system. There are two steps:
|
---|
585 |
|
---|
586 | =over 4
|
---|
587 |
|
---|
588 | =item Update your filters and build files.
|
---|
589 |
|
---|
590 | You probably need to edit the files describing the build steps in one of the $pbdefdir/$project/pbconf/$version/$project/{deb,rpm,pkg} directories and the filters in $pbdefdir/$project/pbconf/$version/pbfilter. Note that you can define additional filters and transforms in the *.pbf files. The build files will be filtered by the filters defined in the *.pbf files to generate the inputs to the build step. Therefore, if you change those files, you need to re-run the cms2build step.
|
---|
591 |
|
---|
592 | =item Build the package.
|
---|
593 |
|
---|
594 | Then you can run a command like:
|
---|
595 |
|
---|
596 | % pb -p $project -r $version build2pkg
|
---|
597 |
|
---|
598 | To create the files in $project/pbbuild that comprise your binary package(s).
|
---|
599 |
|
---|
600 | =back
|
---|
601 |
|
---|
602 | =head2 newve
|
---|
603 |
|
---|
604 | The newve command creates a new virtual environment, i.e. a chrooted OS for building packages. Using a virtual environment is an efficient way to build packages on a related set of operating systems. The OS's have to be related because the kernel will be shared. Steps:
|
---|
605 |
|
---|
606 | =over 4
|
---|
607 |
|
---|
608 | =item Update ~/.pbrc
|
---|
609 |
|
---|
610 | Update your ~/.pbrc file to specify the vepath, velist, velogin, and vetype variables, e.g.:
|
---|
611 |
|
---|
612 | vepath default = $ENV{HOME}/cache-project-builder/chroot
|
---|
613 | velist default = debian-6.0-i386
|
---|
614 | velogin default = pb
|
---|
615 | vetype default = chroot
|
---|
616 |
|
---|
617 | If you are building for rpm style OS's, update the verpmtype option, and install the appropriate tool.
|
---|
618 |
|
---|
619 | verpmtype default = rpmbootstrap
|
---|
620 |
|
---|
621 | You may also choose to specify a mirror for the OS packages, and optionally http/ftp proxies. You can specify the proxies either through environment variables ($http_proxy/$ftp_proxy) or in the configuration file. The configuration file will be used if no corresponding environment variable has been set. For example, for debian and with a local squid proxy:
|
---|
622 |
|
---|
623 | rbsmirrorsrv debian = http://mirrors1.kernel.org/debian/
|
---|
624 | http_proxy default = http://localhost:3128/
|
---|
625 | ftp_proxy default = http://localhost:3128/
|
---|
626 |
|
---|
627 | =item Run the cms2build command
|
---|
628 |
|
---|
629 | If you have deleted your $package/pbdelivery directory, re-run the cms2build command as in the earlier step. This step is necessary to generate the I<package>/pbdelivery/pbrc file.
|
---|
630 |
|
---|
631 | =item Create the new virtual environment
|
---|
632 |
|
---|
633 | Initialize the new operating system. This step will install the core OS packages for the virtual environment, e.g.:
|
---|
634 |
|
---|
635 | % pb -v -p $project -m debian-6.0-i386 newve
|
---|
636 |
|
---|
637 | =back
|
---|
638 |
|
---|
639 | =head2 setupve
|
---|
640 |
|
---|
641 | The setupve command prepares a virtual environment for use by project builder. In particular it installs project-builder from the packages into the virtual environment. Two sub-steps are necessary:
|
---|
642 |
|
---|
643 | =over 4
|
---|
644 |
|
---|
645 | =item Update $project.pb
|
---|
646 |
|
---|
647 | You need to have a sshhost entry for setupve to work, so add one, even an invalid one, e.g.:
|
---|
648 |
|
---|
649 | sshhost $project = foo.example.org
|
---|
650 |
|
---|
651 | =item Setup the virtual environment
|
---|
652 |
|
---|
653 | % pb -v -p $project -m debian-6.0-i386 setupve
|
---|
654 |
|
---|
655 | If you prefer to install the current SVN version of project builder, you can substitute the setupve option by the sbx2setupv one.
|
---|
656 |
|
---|
657 | =back
|
---|
658 |
|
---|
659 | =head2 build2ve
|
---|
660 |
|
---|
661 | The build2ve command is similar to the build2pkg command in that it will take the sources created by cms2build and turn them into binary packages. The command has two differences. First, it creates the packages in a virtual environment, i.e. the one made by an earlier setupve setup. Second it copies the resulting packages to a repository and builds the repository meta-data needed.
|
---|
662 |
|
---|
663 | Three sub-steps are needed:
|
---|
664 |
|
---|
665 | =over 4
|
---|
666 |
|
---|
667 | =item Update $project.pb
|
---|
668 |
|
---|
669 | You need to have a valid sshdir and sshhost entry for build2ve to work, so add them. Note that you need to be able to ssh from the host you run the command on to the repository host, preferably without needing to type in a password, so using ssh-agent or having a special passwordless project-builder ssh key will make this step easier.
|
---|
670 |
|
---|
671 | sshhost $project = localhost
|
---|
672 | sshdir $project = $home/cache-project-builder/repos
|
---|
673 |
|
---|
674 | You may also need to specify additional repository files to use or rpms to install. Note the URL for repositories is not the URL of the repository, but the URL of a file that can be put in the yum.repos.d or apt.sources.d directory.
|
---|
675 |
|
---|
676 | addrepo centos-5-i386 = http://localhost/pb/centos-extras.repo,http://mirror.centos.org/centos/5/extras/i386/RPMS/chrpath-0.13-3.el5.centos.i386.rpm
|
---|
677 |
|
---|
678 | =item Update your filters and build files
|
---|
679 |
|
---|
680 | You may need to update your filter files (*.pbf) as in the build2pkg step if you are building for a new OS or architecture.
|
---|
681 |
|
---|
682 | =item Build the packages and copy them to the repository
|
---|
683 |
|
---|
684 | % pb -v -p $project -m debian-6.0-i386 build2ve
|
---|
685 |
|
---|
686 | =back
|
---|
687 |
|
---|
688 | *Debugging:* If the build fails (and you did not specify the --no-stop-on-error) option, then the virtual environment and scripts should still be present and configured to build the package. You can run a command like 'sudo setarch i386 chroot $path bash' in order to get into the environment. In your log you should see a command like that. From there you can go into the /home/pb directory as the pb user and run the same style of pb commands as you did when doing build2pkg. This will help you figure out what has gone wrong in the build in the virtual environment.
|
---|
689 |
|
---|
690 | =head1 AUTHORS
|
---|
691 |
|
---|
692 | The Project-Builder.org team L<http://trac.project-builder.org/> lead by Bruno Cornec L<mailto:bruno@project-builder.org>.
|
---|
693 |
|
---|
694 | =head1 COPYRIGHT
|
---|
695 |
|
---|
696 | Project-Builder.org is distributed under the GPL v2.0 license
|
---|
697 | described in the file C<COPYING> included with the distribution.
|
---|
698 |
|
---|
699 | =cut
|
---|
700 |
|
---|
701 | # ---------------------------------------------------------------------------
|
---|
702 |
|
---|
703 | my ($projectbuilderver,$projectbuilderrev) = pb_version_init();
|
---|
704 | my $appname = "pb";
|
---|
705 |
|
---|
706 | # Initialize the syntax string
|
---|
707 |
|
---|
708 | pb_syntax_init("$appname (aka project-builder.org) Version $projectbuilderver-$projectbuilderrev\n");
|
---|
709 |
|
---|
710 | GetOptions("help|?|h+" => \$opts{'h'},
|
---|
711 | "account|a=s" => \$opts{'a'},
|
---|
712 | "force|f" => \$opts{'f'},
|
---|
713 | "nographic|g" => \$opts{'g'},
|
---|
714 | "image|i=s" => \$opts{'i'},
|
---|
715 | "log-files|l=s" => \$opts{'l'},
|
---|
716 | "machines|mock|m=s" => \$opts{'m'},
|
---|
717 | "man" => \$opts{'man'},
|
---|
718 | "port|P=i" => \$opts{'P'},
|
---|
719 | "project|p=s" => \$opts{'p'},
|
---|
720 | "quiet|q" => \$opts{'q'},
|
---|
721 | "revision|r=s" => \$opts{'r'},
|
---|
722 | "rebuild" => \$opts{'rebuild'},
|
---|
723 | "snapshot|S" => \$opts{'S'},
|
---|
724 | "script|s=s" => \$opts{'s'},
|
---|
725 | "type|T:s" => \$opts{'T'},
|
---|
726 | "target|t:s" => \$opts{'t'},
|
---|
727 | "version|V=s" => \$opts{'V'},
|
---|
728 | "verbose|v+" => \$opts{'v'},
|
---|
729 | "keep|k" => \$opts{'k'},
|
---|
730 | "stop-on-error!" => \$Global::pb_stop_on_error,
|
---|
731 | ) || pb_syntax(-1,0);
|
---|
732 |
|
---|
733 | if (defined $opts{'l'}) {
|
---|
734 | open(pbLOG,"> $opts{'l'}") || die "Unable to log to $opts{'l'}: $!";
|
---|
735 | $pbLOG = \*pbLOG;
|
---|
736 | $pbdebug = 0 if ($pbdebug == -1);
|
---|
737 | }
|
---|
738 | pb_log_init($pbdebug, $pbLOG);
|
---|
739 |
|
---|
740 | if (defined $opts{'h'}) {
|
---|
741 | pb_syntax(0,$opts{'h'}-1);
|
---|
742 | }
|
---|
743 | if (defined $opts{'man'}) {
|
---|
744 | pb_syntax(0,2);
|
---|
745 | }
|
---|
746 | if (defined $opts{'v'}) {
|
---|
747 | $pbdebug = $opts{'v'};
|
---|
748 | }
|
---|
749 | if (defined $opts{'f'}) {
|
---|
750 | $pbforce=1;
|
---|
751 | }
|
---|
752 | if (defined $opts{'q'}) {
|
---|
753 | $pbdebug=-1;
|
---|
754 | }
|
---|
755 | if (defined $opts{'S'}) {
|
---|
756 | $pbsnap=1;
|
---|
757 | }
|
---|
758 | if (defined $opts{'k'}) {
|
---|
759 | $pbkeep=1;
|
---|
760 | }
|
---|
761 | pb_display_init("text","");
|
---|
762 |
|
---|
763 | # Handle root of the project if defined
|
---|
764 | if (defined $opts{'r'}) {
|
---|
765 | $ENV{'PBROOTDIR'} = $opts{'r'};
|
---|
766 | }
|
---|
767 | # Handle virtual machines if any
|
---|
768 | if (defined $opts{'m'}) {
|
---|
769 | $ENV{'PBV'} = $opts{'m'};
|
---|
770 | }
|
---|
771 | if (defined $opts{'s'}) {
|
---|
772 | $pbscript{'default'} = $opts{'s'};
|
---|
773 | die "option -s requires a script having a full path name" if ($pbscript{'default'} !~ /^\//);
|
---|
774 | }
|
---|
775 | if (defined $opts{'a'}) {
|
---|
776 | $ENV{'PBACCOUNT'} = $opts{'a'};
|
---|
777 | die "option -a requires a -s script option" if (not defined $pbscript{'default'});
|
---|
778 | }
|
---|
779 | if (defined $opts{'P'}) {
|
---|
780 | $pbport = $opts{'P'};
|
---|
781 | }
|
---|
782 | if (defined $opts{'V'}) {
|
---|
783 | $newver = $opts{'V'};
|
---|
784 | }
|
---|
785 | if (defined $opts{'i'}) {
|
---|
786 | $pbimage = $opts{'i'};
|
---|
787 | }
|
---|
788 | if (defined $opts{'t'}) {
|
---|
789 | $pbtarget = $opts{'t'};
|
---|
790 | }
|
---|
791 | if (defined $opts{'T'}) {
|
---|
792 | $vetype = $opts{'T'};
|
---|
793 | }
|
---|
794 |
|
---|
795 | # Get Action
|
---|
796 | $action = shift @ARGV;
|
---|
797 | die pb_syntax(-1,1) if (not defined $action);
|
---|
798 |
|
---|
799 | my ($filteredfiles, $supfiles, $defpkgdir, $extpkgdir);
|
---|
800 | my $pbinit = undef;
|
---|
801 | $pbinit = 1 if ($action =~ /^newproj$/);
|
---|
802 |
|
---|
803 | # Handles project name if any
|
---|
804 | # And get global params
|
---|
805 | ($filteredfiles, $supfiles, $defpkgdir, $extpkgdir) = pb_env_init($opts{'p'},$pbinit,$action,$pbkeep);
|
---|
806 |
|
---|
807 | #
|
---|
808 | # Check for command requirements
|
---|
809 | #
|
---|
810 | my ($req,$opt,$pbpara,$pbshowsudo,$pbstoponerr) = pb_conf_get_if("oscmd","oscmdopt","pbparallel","pbshowsudo","pbstoponerr");
|
---|
811 | pb_check_requirements($req,$opt,$appname);
|
---|
812 |
|
---|
813 | $Global::pb_show_sudo = 1 if ((defined $pbshowsudo) && (defined $pbshowsudo->{$ENV{'PBPROJ'}}) && ($pbshowsudo->{$ENV{'PBPROJ'}} =~ /true/oi));
|
---|
814 | $Global::pb_stop_on_error = 1 if ((defined $pbstoponerr) && (defined $pbstoponerr->{$ENV{'PBPROJ'}}) && ($pbstoponerr->{$ENV{'PBPROJ'}} =~ /true/oi));
|
---|
815 |
|
---|
816 | #
|
---|
817 | # Check if we can launch some actions in // with Parallel::ForkManager
|
---|
818 | #
|
---|
819 | my $pbparallel = $pbpara->{$appname} if (defined $pbpara);
|
---|
820 | if (not defined $pbparallel) {
|
---|
821 | eval
|
---|
822 | {
|
---|
823 | require Sys::CPU;
|
---|
824 | Sys::CPU->import();
|
---|
825 | };
|
---|
826 | if ($@) {
|
---|
827 | # Sys::CPU not found, defaulting to 1
|
---|
828 | pb_log(1,"ADVISE: Install Sys::CPU to benefit from automatic parallelism optimization.\nOr use pbparallel in your pb.conf file\nOnly 1 process at a time for the moment\n");
|
---|
829 | $pbparallel = 1;
|
---|
830 | } else {
|
---|
831 | # Using the number of cores
|
---|
832 | $pbparallel = Sys::CPU::cpu_count();
|
---|
833 | pb_log(1,"Using parallel mode with $pbparallel processes\n");
|
---|
834 | }
|
---|
835 | }
|
---|
836 |
|
---|
837 | eval
|
---|
838 | {
|
---|
839 | require Parallel::ForkManager;
|
---|
840 | Parallel::ForkManager->import();
|
---|
841 | };
|
---|
842 | # Parallel::ForkManager not found so no // actions
|
---|
843 | if ($@) {
|
---|
844 | $pbparallel = undef;
|
---|
845 | pb_log(1,"ADVISE: Install Parallel::ForkManager to benefit from automatic parallelism optimization.\nOnly 1 process at a time for the moment\n");
|
---|
846 | }
|
---|
847 |
|
---|
848 | pb_log(0,"Project: $ENV{'PBPROJ'}\n");
|
---|
849 | pb_log(0,"Action: $action\n");
|
---|
850 |
|
---|
851 | my $do_install = undef;
|
---|
852 | # Possibility to add a 2ins to some commands to launch automatic installation
|
---|
853 | $do_install = 1 if ($action =~ /2ins$/);
|
---|
854 |
|
---|
855 | # Act depending on action
|
---|
856 | if ($action =~ /^cms2build$/) {
|
---|
857 | pb_cms2build("CMS");
|
---|
858 | } elsif ($action =~ /^sbx2build$/) {
|
---|
859 | pb_cms2build("SandBox");
|
---|
860 | } elsif ($action =~ /^build2prep/) {
|
---|
861 | pb_build2prep();
|
---|
862 | } elsif ($action =~ /^build2pkg/) {
|
---|
863 | pb_build2pkg($do_install);
|
---|
864 | } elsif ($action =~ /^cms2pkg/) {
|
---|
865 | pb_cms2build("CMS");
|
---|
866 | pb_build2pkg($do_install);
|
---|
867 | } elsif ($action =~ /^sbx2pkg/) {
|
---|
868 | pb_cms2build("SandBox");
|
---|
869 | pb_build2pkg($do_install);
|
---|
870 | } elsif ($action =~ /^build2ssh$/) {
|
---|
871 | pb_build2ssh();
|
---|
872 | } elsif ($action =~ /^cms2ssh$/) {
|
---|
873 | pb_cms2build("CMS");
|
---|
874 | pb_build2ssh();
|
---|
875 | } elsif ($action =~ /^sbx2ssh$/) {
|
---|
876 | pb_cms2build("SandBox");
|
---|
877 | pb_build2ssh();
|
---|
878 | } elsif ($action =~ /^pkg2ssh$/) {
|
---|
879 | pb_pkg2ssh();
|
---|
880 | } elsif ($action =~ /^build2rm$/) {
|
---|
881 | pb_build2v("rm","build");
|
---|
882 | } elsif ($action =~ /^build2ve$/) {
|
---|
883 | pb_build2v("ve","build");
|
---|
884 | } elsif ($action =~ /^build2vm$/) {
|
---|
885 | pb_build2v("vm","build");
|
---|
886 | } elsif ($action =~ /^sbx2preprm$/) {
|
---|
887 | pb_cms2build("SandBox");
|
---|
888 | pb_build2v("rm","prep");
|
---|
889 | } elsif ($action =~ /^sbx2prepve$/) {
|
---|
890 | pb_cms2build("SandBox");
|
---|
891 | pb_build2v("ve","prep");
|
---|
892 | } elsif ($action =~ /^sbx2prepvm$/) {
|
---|
893 | pb_cms2build("SandBox");
|
---|
894 | pb_build2v("vm","prep");
|
---|
895 | } elsif ($action =~ /^preprm$/) {
|
---|
896 | pb_build2v("rm","prep");
|
---|
897 | } elsif ($action =~ /^prepve$/) {
|
---|
898 | pb_build2v("ve","prep");
|
---|
899 | } elsif ($action =~ /^prepvm$/) {
|
---|
900 | pb_build2v("vm","prep");
|
---|
901 | } elsif ($action =~ /^cms2rm$/) {
|
---|
902 | pb_cms2build("CMS");
|
---|
903 | pb_build2v("rm","build");
|
---|
904 | } elsif ($action =~ /^cms2ve$/) {
|
---|
905 | pb_cms2build("CMS");
|
---|
906 | pb_build2v("ve","build");
|
---|
907 | } elsif ($action =~ /^sbx2rm$/) {
|
---|
908 | pb_cms2build("SandBox");
|
---|
909 | pb_build2v("rm","build");
|
---|
910 | } elsif ($action =~ /^sbx2ve$/) {
|
---|
911 | pb_cms2build("SandBox");
|
---|
912 | pb_build2v("ve","build");
|
---|
913 | } elsif ($action =~ /^cms2vm$/) {
|
---|
914 | pb_cms2build("CMS");
|
---|
915 | pb_build2v("vm","build");
|
---|
916 | } elsif ($action =~ /^sbx2vm$/) {
|
---|
917 | pb_cms2build("SandBox");
|
---|
918 | pb_build2v("vm","build");
|
---|
919 | } elsif ($action =~ /^launchvm$/) {
|
---|
920 | pb_parallel_launchv(undef,"vm",undef,3,$pbimage);
|
---|
921 | } elsif ($action =~ /^launchve$/) {
|
---|
922 | pb_parallel_launchv(undef,"ve",undef,3,$pbimage);
|
---|
923 | } elsif ($action =~ /^script2vm$/) {
|
---|
924 | die "action script2vm requires a -s script option" if (not defined $pbscript{'default'});
|
---|
925 | pb_script2v(\%pbscript,"vm");
|
---|
926 | } elsif ($action =~ /^script2ve$/) {
|
---|
927 | die "action script2ve requires a -s script option" if (not defined $pbscript{'default'});
|
---|
928 | pb_script2v(\%pbscript,"ve");
|
---|
929 | } elsif ($action =~ /^script2rm$/) {
|
---|
930 | die "action script2rm requires a -s script option" if (not defined $pbscript{'default'});
|
---|
931 | pb_script2v(\%pbscript,"rm");
|
---|
932 | } elsif ($action =~ /^newver$/) {
|
---|
933 | pb_newver();
|
---|
934 | } elsif ($action =~ /^sbx2docker$/) {
|
---|
935 | my $savproj = $ENV{'PBPROJ'};
|
---|
936 | my $pkg;
|
---|
937 | my @pkg;
|
---|
938 | if ($#ARGV == -1) {
|
---|
939 | $pkg = pb_cms_get_pkg($defpkgdir,$extpkgdir);
|
---|
940 | @pkgs = @$pkg;
|
---|
941 | } else {
|
---|
942 | @pkgs = @ARGV;
|
---|
943 | }
|
---|
944 | #pb_log(1,"Packages to print:".Dumper(@pkgs)."\n");
|
---|
945 |
|
---|
946 | # Get the list of all VE we need to work on
|
---|
947 | my ($vm,$all);
|
---|
948 | if (not defined $ENV{'PBV'}) {
|
---|
949 | ($vm,$all) = pb_get2v("ve");
|
---|
950 | } else {
|
---|
951 | @$vm = split(/,/,$ENV{'PBV'});
|
---|
952 | }
|
---|
953 |
|
---|
954 | # TODO: parallelize
|
---|
955 | foreach my $v (@$vm) {
|
---|
956 | my $done = 0;
|
---|
957 | my $pbos = pb_distro_get_context($v);
|
---|
958 | # Docker images are (for now !) regular and use distro:ver
|
---|
959 | $pbimage = "$pbos->{'name'}".":$pbos->{'version'}";
|
---|
960 | # the first steps needs to be done with the pb project
|
---|
961 | $ENV{'PBPROJ'} = $appname;
|
---|
962 | my $found = pb_ve_docker_get_image("$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}");
|
---|
963 | if (($pbforce == 1) || (not defined $found)) {
|
---|
964 | # pb -p pb -m distro-ver-arch newve -i distro:ver
|
---|
965 | pb_log(0,"Docker container $pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'} not found, creating it\n");
|
---|
966 | pb_parallel_launchv(undef,"ve",undef,0,$pbimage);
|
---|
967 | }
|
---|
968 | $found = pb_ve_docker_get_image("$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}-pb");
|
---|
969 | if (($pbforce == 1) || (not defined $found)) {
|
---|
970 | pb_log(0,"Docker container $pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}-pb not found, creating it\n");
|
---|
971 | # pb -p pb -m distro-ver-arch setupve
|
---|
972 | if ($savproj eq "pb") {
|
---|
973 | pb_cms2build("SandBox");
|
---|
974 | $done = 1;
|
---|
975 | pb_setup2v("ve","SandBox");
|
---|
976 | } else {
|
---|
977 | pb_setup2v("ve");
|
---|
978 | }
|
---|
979 | }
|
---|
980 | # Now back to our original project
|
---|
981 | $ENV{'PBPROJ'} = $savproj;
|
---|
982 | $found = pb_ve_docker_get_image("$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}-pb-$ENV{'PBPROJ'}");
|
---|
983 | if (($pbforce == 1) || (not defined $found)) {
|
---|
984 | pb_log(0,"Docker container $pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}-pb-$ENV{'PBPROJ'} not found, creating it\n");
|
---|
985 | # pb -p pb -m distro-ver-arch prepve
|
---|
986 | if ($savproj eq "pb") {
|
---|
987 | pb_cms2build("SandBox");
|
---|
988 | $done = 1;
|
---|
989 | }
|
---|
990 | pb_build2v("ve","prep");
|
---|
991 | }
|
---|
992 | pb_log(0,"Instantiating a Docker container from $pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}-pb-$ENV{'PBPROJ'}\n");
|
---|
993 | if (($savproj ne "pb") || ($done == 0)) {
|
---|
994 | pb_cms2build("SandBox");
|
---|
995 | }
|
---|
996 | pb_build2v("ve","build");
|
---|
997 | }
|
---|
998 | } elsif ($action =~ /^newve$/) {
|
---|
999 | pb_parallel_launchv(undef,"ve",undef,0,$pbimage);
|
---|
1000 | } elsif ($action =~ /^newvm$/) {
|
---|
1001 | pb_parallel_launchv(undef,"vm",undef,0,$pbimage);
|
---|
1002 | pb_log(0, "Please ensure that sshd is running in your VM by default\n");
|
---|
1003 | pb_log(0, "and that it allows remote root login (PermitRootLogin yes in /etc/ssh/sshd_config)\n");
|
---|
1004 | pb_log(0, "Also ensure that network is up, firewalling correctly configured\n");
|
---|
1005 | pb_log(0, "and perl, Data::Dumper, tar, sudo, ntpdate and scp/ssh installed\n");
|
---|
1006 | pb_log(0, "You should then be able to login with ssh -p VMPORT root\@localhost (if VM started with pb)\n");
|
---|
1007 | } elsif ($action =~ /^setuprm$/) {
|
---|
1008 | pb_setup2v("rm");
|
---|
1009 | } elsif ($action =~ /^setupve$/) {
|
---|
1010 | pb_setup2v("ve");
|
---|
1011 | } elsif ($action =~ /^setupvm$/) {
|
---|
1012 | pb_setup2v("vm");
|
---|
1013 | } elsif ($action =~ /^sbx2setuprm$/) {
|
---|
1014 | die "This feature is limited to the pb project" if ($ENV{'PBPROJ'} ne $appname);
|
---|
1015 | pb_cms2build("SandBox");
|
---|
1016 | pb_setup2v("rm","SandBox");
|
---|
1017 | } elsif ($action =~ /^sbx2setupve$/) {
|
---|
1018 | die "This feature is limited to the pb project" if ($ENV{'PBPROJ'} ne $appname);
|
---|
1019 | pb_cms2build("SandBox");
|
---|
1020 | pb_setup2v("ve","SandBox");
|
---|
1021 | } elsif ($action =~ /^sbx2setupvm$/) {
|
---|
1022 | die "This feature is limited to the pb project" if ($ENV{'PBPROJ'} ne $appname);
|
---|
1023 | pb_cms2build("SandBox");
|
---|
1024 | pb_setup2v("vm","SandBox");
|
---|
1025 | } elsif ($action =~ /^build2setuprm$/) {
|
---|
1026 | die "This feature is limited to the pb project" if ($ENV{'PBPROJ'} ne $appname);
|
---|
1027 | pb_setup2v("rm","SandBox");
|
---|
1028 | } elsif ($action =~ /^build2setupve$/) {
|
---|
1029 | die "This feature is limited to the pb project" if ($ENV{'PBPROJ'} ne $appname);
|
---|
1030 | pb_setup2v("ve","SandBox");
|
---|
1031 | } elsif ($action =~ /^build2setupvm$/) {
|
---|
1032 | die "This feature is limited to the pb project" if ($ENV{'PBPROJ'} ne $appname);
|
---|
1033 | pb_setup2v("vm","SandBox");
|
---|
1034 | } elsif ($action =~ /^updaterm$/) {
|
---|
1035 | pb_update2v("rm");
|
---|
1036 | } elsif ($action =~ /^updateve$/) {
|
---|
1037 | pb_update2v("ve");
|
---|
1038 | } elsif ($action =~ /^updatevm$/) {
|
---|
1039 | pb_update2v("vm");
|
---|
1040 | } elsif ($action =~ /^snapve$/) {
|
---|
1041 | pb_snap2v("ve");
|
---|
1042 | } elsif ($action =~ /^snapvm$/) {
|
---|
1043 | pb_snap2v("vm");
|
---|
1044 | } elsif ($action =~ /^test2pkg$/) {
|
---|
1045 | pb_test2pkg();
|
---|
1046 | } elsif ($action =~ /^test2rm$/) {
|
---|
1047 | pb_build2v("rm","test");
|
---|
1048 | } elsif ($action =~ /^test2ve$/) {
|
---|
1049 | pb_build2v("ve","test");
|
---|
1050 | } elsif ($action =~ /^test2vm$/) {
|
---|
1051 | pb_build2v("vm","test");
|
---|
1052 | } elsif ($action =~ /^newproj$/) {
|
---|
1053 | # Nothing to do - already done in pb_env_init
|
---|
1054 | } elsif ($action =~ /^getconf$/) {
|
---|
1055 | my $pbos = pb_distro_get_context($ENV{'PBV'});
|
---|
1056 | pb_log(1,"Arguments to print:".Dumper(@ARGV)."\n");
|
---|
1057 | pb_distro_conf_print($pbos,@ARGV);
|
---|
1058 | } elsif ($action =~ /^getvar$/) {
|
---|
1059 | my $pbos = pb_distro_get_context($ENV{'PBV'});
|
---|
1060 | my $pkg;
|
---|
1061 | my @pkg;
|
---|
1062 | if ($#ARGV == -1) {
|
---|
1063 | $pkg = pb_cms_get_pkg($defpkgdir,$extpkgdir);
|
---|
1064 | @pkgs = @$pkg;
|
---|
1065 | } else {
|
---|
1066 | @pkgs = @ARGV;
|
---|
1067 | }
|
---|
1068 | pb_log(1,"Packages to print:".Dumper(@pkgs)."\n");
|
---|
1069 | pb_filter_var_print($pbos,@pkgs);
|
---|
1070 | } elsif ($action =~ /^clean$/) {
|
---|
1071 | pb_clean();
|
---|
1072 | } elsif ($action =~ /^announce$/) {
|
---|
1073 | # For announce only. Require avoids the systematic load of these modules
|
---|
1074 | require DBI;
|
---|
1075 | require DBD::SQLite;
|
---|
1076 |
|
---|
1077 | pb_announce("Announce");
|
---|
1078 | } elsif ($action =~ /^cleanssh$/) {
|
---|
1079 | pb_announce("Clean");
|
---|
1080 | } elsif ($action =~ /^checkssh$/) {
|
---|
1081 | pb_announce("Check");
|
---|
1082 | } elsif ($action =~ /^checkps$/) {
|
---|
1083 | my ($ptr2) = pb_conf_get("vmcmd");
|
---|
1084 | my $vmcmd = $ptr2->{$ENV{'PBPROJ'}};
|
---|
1085 | my ($vexist,$vmmport) = pb_check_ps($vmcmd,$ENV{'PBV'});
|
---|
1086 | if (! $vexist) {
|
---|
1087 | pb_log(0,"No VM found for $ENV{'PBV'}\n");
|
---|
1088 | } else {
|
---|
1089 | pb_log(0,"Found an existing VM for $ENV{'PBV'} PID: $vexist - SSH port: $vmmport\n");
|
---|
1090 | }
|
---|
1091 | } elsif ($action =~ /^sbx2webpkg$/) {
|
---|
1092 | require DBI;
|
---|
1093 | require DBD::SQLite;
|
---|
1094 |
|
---|
1095 | pb_cms2build("SandBox","Web");
|
---|
1096 | } elsif ($action =~ /^sbx2webssh$/) {
|
---|
1097 | require DBI;
|
---|
1098 | require DBD::SQLite;
|
---|
1099 |
|
---|
1100 | pb_cms2build("SandBox","Web");
|
---|
1101 | pb_send2target("Web");
|
---|
1102 | } elsif ($action =~ /^cms2webpkg$/) {
|
---|
1103 | require DBI;
|
---|
1104 | require DBD::SQLite;
|
---|
1105 |
|
---|
1106 | pb_cms2build("CMS","Web");
|
---|
1107 | } elsif ($action =~ /^cms2webssh$/) {
|
---|
1108 | require DBI;
|
---|
1109 | require DBD::SQLite;
|
---|
1110 |
|
---|
1111 | pb_cms2build("CMS","Web");
|
---|
1112 | pb_send2target("Web");
|
---|
1113 | } else {
|
---|
1114 | pb_log(0,"\'$action\' is not available\n");
|
---|
1115 | pb_syntax(-2,1);
|
---|
1116 | }
|
---|
1117 |
|
---|
1118 | pb_exit();
|
---|
1119 |
|
---|
1120 | sub pb_cms2build {
|
---|
1121 |
|
---|
1122 | my $param = shift;
|
---|
1123 | my $web = shift;
|
---|
1124 |
|
---|
1125 | my $pkg;
|
---|
1126 | my @pkgs;
|
---|
1127 | my $webdir;
|
---|
1128 |
|
---|
1129 | my %pkgs;
|
---|
1130 | my $pb; # Structure to store conf info
|
---|
1131 |
|
---|
1132 | die "pb_cms2build requires a parameter: SandBox or CMS" if (not defined $param);
|
---|
1133 |
|
---|
1134 | # If Website, then pkg is only the website
|
---|
1135 | if (defined $web) {
|
---|
1136 | ($webdir) = pb_conf_get("webdir");
|
---|
1137 | pb_log(2,"webdir: ".Dumper($webdir)."\n");
|
---|
1138 | $pkgs[0] = $webdir->{$ENV{'PBPROJ'}};
|
---|
1139 | $extpkgdir = $webdir;
|
---|
1140 | pb_log(0,"Package: $pkgs[0]\n");
|
---|
1141 | } else {
|
---|
1142 | $pkg = pb_cms_get_pkg($defpkgdir,$extpkgdir);
|
---|
1143 | @pkgs = @$pkg;
|
---|
1144 | }
|
---|
1145 |
|
---|
1146 | my ($scheme, $uri) = pb_cms_init($pbinit,$param);
|
---|
1147 |
|
---|
1148 | # We need 2 lines here
|
---|
1149 | my ($pkgv, $pkgt, $testver) = pb_conf_get_if("pkgver","pkgtag","testver");
|
---|
1150 |
|
---|
1151 | # declare packager and repo for filtering
|
---|
1152 | my ($tmp1, $tmp2) = pb_conf_get("pbpackager","pbrepo");
|
---|
1153 | $ENV{'PBPACKAGER'} = $tmp1->{$ENV{'PBPROJ'}};
|
---|
1154 | $ENV{'PBREPO'} = $tmp2->{$ENV{'PBPROJ'}};
|
---|
1155 | my ($delivery) = pb_conf_get_if("delivery");
|
---|
1156 | $delivery->{$ENV{'PBPROJ'}} = "" if (not defined $delivery->{$ENV{'PBPROJ'}});
|
---|
1157 |
|
---|
1158 | # If we deal with a test dir, we want to keep the date in tar file and dir name
|
---|
1159 | my $pbextdirpkg = "";
|
---|
1160 | my $pbextdir = "";
|
---|
1161 | if ((defined $testver) && (defined $testver->{$ENV{'PBPROJ'}}) && ($testver->{$ENV{'PBPROJ'}} =~ /true/i)) {
|
---|
1162 | $pbextdirpkg = strftime("%Y%m%d%H%M%S", @date);
|
---|
1163 | $pbextdir = ".0.".$pbextdirpkg;
|
---|
1164 | }
|
---|
1165 |
|
---|
1166 | foreach my $pbpkg (@pkgs) {
|
---|
1167 | $ENV{'PBPKG'} = $pbpkg;
|
---|
1168 | $pbrev = $ENV{'PBREVISION'};
|
---|
1169 |
|
---|
1170 | if ((defined $pkgv) && (defined $pkgv->{$pbpkg})) {
|
---|
1171 | $pbver = $pkgv->{$pbpkg};
|
---|
1172 | } else {
|
---|
1173 | $pbver = $ENV{'PBPROJVER'};
|
---|
1174 | }
|
---|
1175 | # If it's a test version, then build a tag to fit distributions need
|
---|
1176 | if ((defined $testver) && (defined $testver->{$ENV{'PBPROJ'}}) && ($testver->{$ENV{'PBPROJ'}} =~ /true/i)) {
|
---|
1177 | my $letter = "";
|
---|
1178 | if ($scheme =~ /svn/) {
|
---|
1179 | $letter = "s";
|
---|
1180 | } elsif ($scheme =~ /cvs/) {
|
---|
1181 | $letter = "c";
|
---|
1182 | } elsif ($scheme =~ /svk/) {
|
---|
1183 | $letter = "k";
|
---|
1184 | } elsif ($scheme =~ /hg/) {
|
---|
1185 | $letter = "h";
|
---|
1186 | } elsif ($scheme =~ /git/) {
|
---|
1187 | $letter = "g";
|
---|
1188 | } else {
|
---|
1189 | $letter = "f";
|
---|
1190 | }
|
---|
1191 | $pbtag = "0.$pbextdirpkg.$letter$pbrev";
|
---|
1192 | $ENV{'PBPROJTAG'} = $pbtag;
|
---|
1193 | } elsif ((defined $pkgt) && (defined $pkgt->{$pbpkg})) {
|
---|
1194 | $pbtag = $pkgt->{$pbpkg};
|
---|
1195 | } else {
|
---|
1196 | $pbtag = $ENV{'PBPROJTAG'};
|
---|
1197 | }
|
---|
1198 |
|
---|
1199 | pb_log(0,"\n");
|
---|
1200 | pb_log(0,"Management of $pbpkg $pbver-$pbtag\n");
|
---|
1201 | die "Unable to get env var PBDESTDIR" if (not defined $ENV{'PBDESTDIR'});
|
---|
1202 |
|
---|
1203 | my $dest = "$ENV{'PBDESTDIR'}/$pbpkg-$pbver$pbextdir";
|
---|
1204 | # Create the structure if needed
|
---|
1205 | pb_mkdir_p($dest);
|
---|
1206 | # Clean up dest if necessary. The export will recreate it
|
---|
1207 | pb_rm_rf($dest) if (-d $dest);
|
---|
1208 |
|
---|
1209 | # Export CMS tree for the concerned package to dest
|
---|
1210 | # And generate some additional files
|
---|
1211 | $OUTPUT_AUTOFLUSH=1;
|
---|
1212 |
|
---|
1213 | # computes in which dir we have to work
|
---|
1214 | my $dir = $defpkgdir->{$pbpkg};
|
---|
1215 | $dir = $extpkgdir->{$pbpkg} if (not defined $dir);
|
---|
1216 | $dir = $webdir->{$ENV{'PBPROJ'}} if (defined $web);
|
---|
1217 | die "Variable \$dir not defined. Check the package name first.\nIf this is the one expected, please report to dev team with log of a verbose run and this info ".Dumper($defpkgdir,$extpkgdir,$webdir) if (not defined $dir);
|
---|
1218 | pb_log(2,"def:".Dumper($defpkgdir)."ext: ".Dumper($extpkgdir)."dir: $dir\n");
|
---|
1219 |
|
---|
1220 | # Exporting content from CMS
|
---|
1221 | my $sourcedir = undef;
|
---|
1222 | my $sourceuri = $uri;
|
---|
1223 | if ($param eq "SandBox") {
|
---|
1224 | # Point to the local instance
|
---|
1225 | $sourcedir = "$ENV{'PBDIR'}/$dir";
|
---|
1226 | } else {
|
---|
1227 | # Get it from a subdir of the URI with same version as localy but different root
|
---|
1228 | # Only if using a real CMS
|
---|
1229 | my ($scheme, $account, $host, $port, $path) = pb_get_uri($uri);
|
---|
1230 | if (($scheme !~ /^file/) && ($scheme !~ /^(ht|f)tp/)) {
|
---|
1231 | $sourceuri = "$ENV{'PBDIR'}/$dir";
|
---|
1232 | $sourceuri =~ s|^$ENV{'PBPROJDIR'}/|$uri/|;
|
---|
1233 | }
|
---|
1234 | }
|
---|
1235 | my $preserve = pb_vcs_export($sourceuri,$sourcedir,$dest);
|
---|
1236 |
|
---|
1237 | # Generated fake content for test versions to speed up stuff
|
---|
1238 | my $chglog;
|
---|
1239 |
|
---|
1240 | # Get project info on authors and log file
|
---|
1241 | # TODO: Make it CMS aware
|
---|
1242 | $chglog = "$ENV{'PBROOTDIR'}/$pbpkg/pbcl";
|
---|
1243 | $chglog = "$ENV{'PBROOTDIR'}/pbcl" if (! -f $chglog);
|
---|
1244 | $chglog = undef if (! -f $chglog);
|
---|
1245 |
|
---|
1246 | # TODO: Make it CMS aware
|
---|
1247 | my $authors = "$ENV{'PBROOTDIR'}/$pbpkg/pbauthors";
|
---|
1248 | $authors = "$ENV{'PBROOTDIR'}/pbauthors" if (! -f $authors);
|
---|
1249 | $authors = "/dev/null" if (! -f $authors);
|
---|
1250 |
|
---|
1251 | # Extract cms log history and store it
|
---|
1252 | if ((defined $chglog) && (! -f "$dest/NEWS")) {
|
---|
1253 | pb_log(2,"Generating NEWS file from $chglog\n");
|
---|
1254 | copy($chglog,"$dest/NEWS") || die "Unable to create $dest/NEWS";
|
---|
1255 | }
|
---|
1256 | pb_cms_log($scheme,"$ENV{'PBDIR'}/$dir",$dest,$chglog,$authors,$testver);
|
---|
1257 |
|
---|
1258 | my %build;
|
---|
1259 | # We want to at least build for the underlying distro
|
---|
1260 | # except if a target was given, in which case we only build for it
|
---|
1261 | # if -t was passed without target then build for the native distro.
|
---|
1262 | my $pbos = pb_distro_get_context($pbtarget);
|
---|
1263 | my $tmpl = pb_get_distros($pbos,$pbtarget);
|
---|
1264 |
|
---|
1265 | # Setup $pb structure to allow filtering later on, on files using that structure
|
---|
1266 | $pb->{'tag'} = $pbtag;
|
---|
1267 | $pb->{'rev'} = $pbrev;
|
---|
1268 | $pb->{'ver'} = $pbver;
|
---|
1269 | $pb->{'pkg'} = $pbpkg;
|
---|
1270 | $pb->{'suf'} = $pbos->{'suffix'};
|
---|
1271 | $pb->{'realpkg'} = $pbpkg;
|
---|
1272 | $pb->{'date'} = $pbdate;
|
---|
1273 | $pb->{'defpkgdir'} = $defpkgdir;
|
---|
1274 | $pb->{'extpkgdir'} = $extpkgdir;
|
---|
1275 | $pb->{'chglog'} = $chglog;
|
---|
1276 | $pb->{'extdir'} = $pbextdir;
|
---|
1277 | $pb->{'packager'} = $ENV{'PBPACKAGER'};
|
---|
1278 | $pb->{'proj'} = $ENV{'PBPROJ'};
|
---|
1279 | $pb->{'repo'} = "$ENV{'PBREPO'}/$delivery->{$ENV{'PBPROJ'}}";
|
---|
1280 | $pb->{'patches'} = ();
|
---|
1281 | $pb->{'sources'} = ();
|
---|
1282 |
|
---|
1283 | my $tmpd = "$ENV{'PBTMP'}/cms.$$";
|
---|
1284 | pb_mkdir_p($tmpd) if (defined $pbparallel);
|
---|
1285 |
|
---|
1286 | # Get only all.pbf filter at this stage for pbinit
|
---|
1287 | my $ptr = pb_get_filters($pbpkg);
|
---|
1288 |
|
---|
1289 | # Do not do that for website
|
---|
1290 | if (not defined $web) {
|
---|
1291 | my %virt;
|
---|
1292 | # De-duplicate similar VM/VE/RM
|
---|
1293 | foreach my $d (split(/,/,$tmpl)) {
|
---|
1294 | # skip ill-formatted vms (name-ver-arch)
|
---|
1295 | next if ($d !~ /-/);
|
---|
1296 | $virt{$d} = $d;
|
---|
1297 | }
|
---|
1298 |
|
---|
1299 | # Try to use // processing here
|
---|
1300 | my $all_ok = 1;
|
---|
1301 | my $pm = new Parallel::ForkManager($pbparallel) if (defined $pbparallel);
|
---|
1302 | $pm->run_on_finish(sub { my ($pid, $code, $id, $signal, $dump) = @_;
|
---|
1303 | $all_ok = 0 unless (($code == 0) && ($signal == 0) && ($dump == 0)); }) if (defined $pbparallel);
|
---|
1304 |
|
---|
1305 | pb_log(0,"Preparing delivery ...\n");
|
---|
1306 | foreach my $v (keys %virt) {
|
---|
1307 | $pm->start and next if (defined $pbparallel);
|
---|
1308 |
|
---|
1309 | # Distro context
|
---|
1310 | my $pbos = pb_distro_get_context($v);
|
---|
1311 |
|
---|
1312 | $pb->{'pbos'} = $pbos;
|
---|
1313 | $pb->{'suf'} = $pbos->{'suffix'};
|
---|
1314 | pb_log(3,"DEBUG: pb: ".Dumper($pb)."\n");
|
---|
1315 |
|
---|
1316 | # Get all filters to apply
|
---|
1317 | $ptr = pb_get_filters($pbpkg,$pbos);
|
---|
1318 |
|
---|
1319 | pb_log(2,"DEBUG Filtering PBDATE => $pbdate, PBTAG => $pbtag, PBVER => $pbver\n");
|
---|
1320 |
|
---|
1321 | # We need to compute the real name of the package
|
---|
1322 | my $pbrealpkg = pb_cms_get_real_pkg($pbpkg,$pbos->{'type'});
|
---|
1323 | $pb->{'realpkg'} = $pbrealpkg;
|
---|
1324 | pb_log(1,"Virtual package $pbpkg has a real package name of $pbrealpkg on $pbos->{'name'}-$pbos->{'version'}\n") if ($pbrealpkg ne $pbpkg);
|
---|
1325 |
|
---|
1326 | # Filter build files from the less precise up to the most with overloading
|
---|
1327 | # Filter all files found, keeping the name, and generating in dest
|
---|
1328 |
|
---|
1329 | # Find all build files first relatively to PBROOTDIR
|
---|
1330 | # Find also all specific files referenced in the .pb conf file
|
---|
1331 | my %bfiles = ();
|
---|
1332 | my %pkgfiles = ();
|
---|
1333 | # Used in Filter.pm by pb_filter_file
|
---|
1334 |
|
---|
1335 | $build{$v} = "no";
|
---|
1336 | if (-d "$ENV{'PBROOTDIR'}/$pbpkg/$pbos->{'os'}") {
|
---|
1337 | pb_list_bfiles("$ENV{'PBROOTDIR'}/$pbpkg/$pbos->{'os'}",$pbpkg,\%bfiles,\%pkgfiles,$supfiles);
|
---|
1338 | $build{$v} = "yes";
|
---|
1339 | }
|
---|
1340 | if (-d "$ENV{'PBROOTDIR'}/$pbpkg/$pbos->{'type'}") {
|
---|
1341 | pb_list_bfiles("$ENV{'PBROOTDIR'}/$pbpkg/$pbos->{'type'}",$pbpkg,\%bfiles,\%pkgfiles,$supfiles);
|
---|
1342 | $build{$v} = "yes";
|
---|
1343 | }
|
---|
1344 | if (-d "$ENV{'PBROOTDIR'}/$pbpkg/$pbos->{'family'}") {
|
---|
1345 | pb_list_bfiles("$ENV{'PBROOTDIR'}/$pbpkg/$pbos->{'family'}",$pbpkg,\%bfiles,\%pkgfiles,$supfiles);
|
---|
1346 | $build{$v} = "yes";
|
---|
1347 | }
|
---|
1348 | if (-d "$ENV{'PBROOTDIR'}/$pbpkg/$pbos->{'name'}") {
|
---|
1349 | pb_list_bfiles("$ENV{'PBROOTDIR'}/$pbpkg/$pbos->{'name'}",$pbpkg,\%bfiles,\%pkgfiles,$supfiles);
|
---|
1350 | $build{$v} = "yes";
|
---|
1351 | }
|
---|
1352 | if (-d "$ENV{'PBROOTDIR'}/$pbpkg/$pbos->{'name'}-$pbos->{'version'}") {
|
---|
1353 | pb_list_bfiles("$ENV{'PBROOTDIR'}/$pbpkg/$pbos->{'name'}-$pbos->{'version'}",$pbpkg,\%bfiles,\%pkgfiles,$supfiles);
|
---|
1354 | $build{$v} = "yes";
|
---|
1355 | }
|
---|
1356 | if (-d "$ENV{'PBROOTDIR'}/$pbpkg/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}") {
|
---|
1357 | pb_list_bfiles("$ENV{'PBROOTDIR'}/$pbpkg/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}",$pbpkg,\%bfiles,\%pkgfiles,$supfiles);
|
---|
1358 | $build{$v} = "yes";
|
---|
1359 | }
|
---|
1360 | pb_log(2,"DEBUG($v) bfiles: ".Dumper(\%bfiles)."\n");
|
---|
1361 |
|
---|
1362 | if ($build{$v} ne "no") {
|
---|
1363 | # Apply now all the filters on all the files concerned
|
---|
1364 | # destination dir depends on the type of file
|
---|
1365 | # For patch support
|
---|
1366 | # TODO: Make it CMS aware
|
---|
1367 | $pb->{'patches'} = pb_list_sfiles("$ENV{'PBROOTDIR'}/$pbpkg/pbpatch", $pb->{'patches'}, $pbos, "$ENV{'PBROOTDIR'}/$pbpkg/pbextpatch");
|
---|
1368 | pb_log(2,"DEBUG($v) patches: ".Dumper($pb->{'patches'})."\n");
|
---|
1369 | # TODO: Make it CMS aware
|
---|
1370 | $pb->{'sources'} = pb_list_sfiles("$ENV{'PBROOTDIR'}/$pbpkg/pbsrc", $pb->{'sources'}, $pbos, "$ENV{'PBROOTDIR'}/$pbpkg/pbextsrc");
|
---|
1371 | pb_log(2,"DEBUG($v) sources: ".Dumper($pb->{'sources'})."\n");
|
---|
1372 |
|
---|
1373 | if (defined $pb->{'patches'}->{$v}) {
|
---|
1374 | # Filter potential patches (local + remote)
|
---|
1375 | pb_mkdir_p("$dest/pbconf/$v/pbpatch");
|
---|
1376 | # If Debian based distribution, then prepare what will be done at build time
|
---|
1377 | my ($patchcmd,$patchopt);
|
---|
1378 | if ($pbos->{'type'} eq "deb") {
|
---|
1379 | ($patchcmd,$patchopt) = pb_distro_get_param($pbos,pb_conf_get_if("ospatchcmd","ospatchopt"));
|
---|
1380 | open(SCRIPT,"> $dest/pbconf/$v/pbpatch/pbapplypatch") || die "Unable to create $dest/pbconf/$v/pbpatch/pbapplypatch";
|
---|
1381 | print SCRIPT "#!/bin/bash\n";
|
---|
1382 | print SCRIPT "set -x\n" if ($pbdebug gt 1);
|
---|
1383 | }
|
---|
1384 | foreach my $pf (split(/,/,$pb->{'patches'}->{$v})) {
|
---|
1385 | my $pp = basename($pf);
|
---|
1386 | pb_vcs_export($pf,undef,"$dest/pbconf/$v/pbpatch");
|
---|
1387 | pb_filter_file_inplace($ptr,"$dest/pbconf/$v/pbpatch/$pp",$pb);
|
---|
1388 | pb_system("gzip -9f $dest/pbconf/$v/pbpatch/$pp","","quiet");
|
---|
1389 | if ($pbos->{'type'} eq "deb") {
|
---|
1390 | # If Debian based distribution, then prepare what will be done at build time
|
---|
1391 | # by applying the patches that will be available under the debian/patches dir
|
---|
1392 | print SCRIPT "$patchcmd $patchopt \< debian/patches/$pp\n";
|
---|
1393 | }
|
---|
1394 | }
|
---|
1395 | if ($pbos->{'type'} eq "deb") {
|
---|
1396 | close(SCRIPT);
|
---|
1397 | chmod 0755,"$dest/pbconf/$v/pbpatch/pbapplypatch";
|
---|
1398 | }
|
---|
1399 | #pb_system("cat $dest/pbconf/$v/pbpatch/pbapplypatch","APPLY","verbose");
|
---|
1400 | }
|
---|
1401 | if (defined $pb->{'sources'}->{$v}) {
|
---|
1402 | pb_mkdir_p("$dest/pbconf/$v/pbsrc");
|
---|
1403 | foreach my $pf (split(/,/,$pb->{'sources'}->{$v})) {
|
---|
1404 | my $pp = basename($pf);
|
---|
1405 | pb_vcs_export($pf,undef,"$dest/pbconf/$v/pbsrc");
|
---|
1406 | pb_filter_file_inplace($ptr,"$dest/pbconf/$v/pbsrc/$pp",$pb);
|
---|
1407 | }
|
---|
1408 | }
|
---|
1409 | # Filter build files at the end, as they depend on patches and sources
|
---|
1410 | foreach my $f (keys %bfiles) {
|
---|
1411 | pb_filter_file("$ENV{'PBROOTDIR'}/$bfiles{$f}",$ptr,"$dest/pbconf/$v/$f",$pb);
|
---|
1412 | }
|
---|
1413 | foreach my $f (keys %pkgfiles) {
|
---|
1414 | pb_filter_file("$ENV{'PBROOTDIR'}/$pkgfiles{$f}",$ptr,"$dest/pbconf/$v/$f",$pb);
|
---|
1415 | }
|
---|
1416 | }
|
---|
1417 |
|
---|
1418 | if (defined $pbparallel) {
|
---|
1419 | # Communicate results back to parent
|
---|
1420 | my $str = "";
|
---|
1421 | $str .= "build $v = $build{$v}\n" if (defined $build{$v});
|
---|
1422 | $str .= "patches $v = $pb->{'patches'}->{$v}\n" if (defined $pb->{'patches'}->{$v});
|
---|
1423 | $str .= "sources $v = $pb->{'sources'}->{$v}\n" if (defined $pb->{'sources'}->{$v});
|
---|
1424 | pb_set_content("$tmpd/$$","$str");
|
---|
1425 | $pm->finish;
|
---|
1426 | }
|
---|
1427 | }
|
---|
1428 | # In the parent, we need to get the result from the children
|
---|
1429 | $pm->wait_all_children if (defined $pbparallel);
|
---|
1430 | die "Aborting, one or more of the children failed." if ((not $all_ok) && ($Global::pb_stop_on_error));
|
---|
1431 | my $made = "";
|
---|
1432 | my %h = ();
|
---|
1433 | my %tmp;
|
---|
1434 | my %tmp2;
|
---|
1435 | my $pt;
|
---|
1436 | my $k;
|
---|
1437 |
|
---|
1438 | foreach $k (<$tmpd/*>) {
|
---|
1439 | $made .= pb_get_content($k);
|
---|
1440 | }
|
---|
1441 | pb_rm_rf($tmpd);
|
---|
1442 | pb_log(3,"MADE:\n$made");
|
---|
1443 |
|
---|
1444 | # Rebuild local hashes
|
---|
1445 | foreach $k (split(/\n/,$made)) {
|
---|
1446 | if ($k =~ /^\s*([A-z0-9-_]+)\s+([[A-z0-9-_.]+)\s*=\s*(.+)$/) {
|
---|
1447 | $h{$1}->{$2}=$3;
|
---|
1448 | }
|
---|
1449 | }
|
---|
1450 | pb_log(2,"HASH: ".Dumper(%h));
|
---|
1451 |
|
---|
1452 | # Patches
|
---|
1453 | pb_log(0,"Delivered and compressed patches ");
|
---|
1454 | $pt = $h{'patches'};
|
---|
1455 | foreach $k (keys %$pt) {
|
---|
1456 | foreach my $v1 (split(/,/,$pt->{$k})) {
|
---|
1457 | $tmp{$v1} = "";
|
---|
1458 | }
|
---|
1459 | }
|
---|
1460 | if (keys %tmp) {
|
---|
1461 | foreach $k (keys %tmp) {
|
---|
1462 | pb_log(0,"$k ");
|
---|
1463 | }
|
---|
1464 | } else {
|
---|
1465 | pb_log(0,"N/A");
|
---|
1466 | }
|
---|
1467 | pb_log(0,"\n");
|
---|
1468 |
|
---|
1469 | # Sources
|
---|
1470 | pb_log(0,"Delivered additional sources ");
|
---|
1471 | $pt = $h{'sources'};
|
---|
1472 | foreach $k (keys %$pt) {
|
---|
1473 | foreach my $v1 (split(/,/,$pt->{$k})) {
|
---|
1474 | $tmp2{$v1} = "";
|
---|
1475 | }
|
---|
1476 | }
|
---|
1477 | if (keys %tmp2) {
|
---|
1478 | foreach $k (keys %tmp2) {
|
---|
1479 | pb_log(0,"$k ");
|
---|
1480 | }
|
---|
1481 | } else {
|
---|
1482 | pb_log(0,"N/A");
|
---|
1483 | }
|
---|
1484 | pb_log(0,"\n");
|
---|
1485 |
|
---|
1486 | # Build files
|
---|
1487 | my @found;
|
---|
1488 | my @notfound;
|
---|
1489 | $pt = $h{'build'};
|
---|
1490 | foreach my $b (keys %$pt) {
|
---|
1491 | push @found,$b if ($pt->{$b} =~ /yes/);
|
---|
1492 | push @notfound,$b if ($pt->{$b} =~ /no/);
|
---|
1493 | }
|
---|
1494 | pb_log(0,"Build files have been generated for ... ".join(',',sort(@found))."\n") if (@found);
|
---|
1495 | pb_log(0,"No Build files found for ".join(',',sort(@notfound))."\n") if (@notfound);
|
---|
1496 |
|
---|
1497 | } else {
|
---|
1498 | # Instead call News generation
|
---|
1499 | pb_web_news2html($dest);
|
---|
1500 | # And create an empty pbconf
|
---|
1501 | pb_mkdir_p("$dest/pbconf");
|
---|
1502 | # And prepare the pbscript to execute remotely
|
---|
1503 | open(SCRIPT,"> $ENV{'PBTMP'}/pbscript") || die "Unable to create $ENV{'PBTMP'}/pbscript";
|
---|
1504 | print SCRIPT "#!/bin/bash\n";
|
---|
1505 | print SCRIPT "#set -x\n";
|
---|
1506 | print SCRIPT "echo ... Extracting Website content\n";
|
---|
1507 | print SCRIPT "find . -type f | grep -Ev '^./$pbpkg-$pbver$pbextdir.tar.gz|^./pbscript' | xargs rm -f non-existent\n";
|
---|
1508 | print SCRIPT "find * -type d -depth | xargs rmdir 2> /dev/null \n";
|
---|
1509 | print SCRIPT "tar xfz $pbpkg-$pbver$pbextdir.tar.gz\n";
|
---|
1510 | print SCRIPT "mv $pbpkg-$pbver$pbextdir/* .\n";
|
---|
1511 | print SCRIPT "rm -f $pbpkg-$pbver$pbextdir.tar.gz\n";
|
---|
1512 | print SCRIPT "rmdir $pbpkg-$pbver$pbextdir\n";
|
---|
1513 | print SCRIPT "find . -type f -print0 | xargs -0 chmod 644\n";
|
---|
1514 | print SCRIPT "find . -type d -print0 | xargs -0 chmod 755\n";
|
---|
1515 | close(SCRIPT);
|
---|
1516 | chmod 0755,"$ENV{'PBTMP'}/pbscript";
|
---|
1517 | }
|
---|
1518 |
|
---|
1519 | # Apply filters to the non-build files
|
---|
1520 | my $liste ="";
|
---|
1521 | if (defined $filteredfiles->{$pbpkg}) {
|
---|
1522 | foreach my $f (split(/,/,$filteredfiles->{$pbpkg})) {
|
---|
1523 | pb_filter_file_inplace($ptr,"$dest/$f",$pb);
|
---|
1524 | $liste = "$f $liste";
|
---|
1525 | }
|
---|
1526 | }
|
---|
1527 | pb_log(2,"Files ".$liste."have been filtered\n");
|
---|
1528 |
|
---|
1529 | # TODO: Make it CMS aware
|
---|
1530 | # Execute the pbinit script if any
|
---|
1531 | if (-x "$ENV{'PBROOTDIR'}/$pbpkg/pbinit") {
|
---|
1532 | pb_filter_file("$ENV{'PBROOTDIR'}/$pbpkg/pbinit",$ptr,"$ENV{'PBTMP'}/pbinit",$pb);
|
---|
1533 | chmod 0755,"$ENV{'PBTMP'}/pbinit";
|
---|
1534 | pb_system("cd $dest ; $ENV{'PBTMP'}/pbinit","Executing init script from $ENV{'PBROOTDIR'}/$pbpkg/pbinit under $dest","verbose");
|
---|
1535 | }
|
---|
1536 |
|
---|
1537 | # Do we have additional script to run to prepare the environement for the project ?
|
---|
1538 | # Then include it in the pbconf delivery
|
---|
1539 | foreach my $pbvf (<$ENV{'PBROOTDIR'}/pbv*.pre>,<$ENV{'PBROOTDIR'}/pbv*.post>, <$ENV{'PBROOTDIR'}/pbtest*>) {
|
---|
1540 | if (-x "$pbvf") {
|
---|
1541 | my $target = "$ENV{'PBDESTDIR'}/".basename($pbvf);
|
---|
1542 | pb_filter_file("$pbvf",$ptr,$target,$pb);
|
---|
1543 | chmod 0755,"$target";
|
---|
1544 | }
|
---|
1545 | }
|
---|
1546 |
|
---|
1547 | # Prepare the dest directory for archive
|
---|
1548 | chdir "$ENV{'PBDESTDIR'}" || die "Unable to change dir to $ENV{'PBDESTDIR'}";
|
---|
1549 | if (defined $preserve) {
|
---|
1550 | # In that case we want to preserve the original tar file for checksum purposes
|
---|
1551 | # The one created is btw equivalent in that case to this one
|
---|
1552 | # Maybe check basename of both to be sure they are the same ?
|
---|
1553 | pb_log(0,"Preserving original tar file ");
|
---|
1554 | move("$preserve","$pbpkg-$pbver$pbextdir.tar.gz");
|
---|
1555 | } else {
|
---|
1556 | # Possibility to look at PBSRC to guess more the filename
|
---|
1557 | pb_system("tar cfz $pbpkg-$pbver$pbextdir.tar.gz --exclude=$pbpkg-$pbver$pbextdir/pbconf $pbpkg-$pbver$pbextdir","Creating $pbpkg tar files compressed");
|
---|
1558 | }
|
---|
1559 | pb_log(0,"Under $ENV{'PBDESTDIR'}/$pbpkg-$pbver$pbextdir.tar.gz\n");
|
---|
1560 | pb_system("tar cfz $pbpkg-$pbver$pbextdir.pbconf.tar.gz $pbpkg-$pbver$pbextdir/pbconf","Creating pbconf tar files compressed");
|
---|
1561 | pb_log(0,"Under $ENV{'PBDESTDIR'}/$pbpkg-$pbver$pbextdir.pbconf.tar.gz\n");
|
---|
1562 |
|
---|
1563 | # Keep track of version-tag per pkg
|
---|
1564 | $pkgs{$pbpkg} = "$pbver-$pbtag";
|
---|
1565 |
|
---|
1566 | # Final cleanup
|
---|
1567 | pb_rm_rf($dest) if (-d $dest);
|
---|
1568 | }
|
---|
1569 |
|
---|
1570 | # Keep track of per package version
|
---|
1571 | pb_log(2,"DEBUG pkgs: ".Dumper(%pkgs)."\n");
|
---|
1572 | open(PKG,"> $ENV{'PBDESTDIR'}/$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}.pb") || die "Unable to create $ENV{'PBDESTDIR'}/$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}.pb";
|
---|
1573 | foreach my $pbpkg (keys %pkgs) {
|
---|
1574 | print PKG "pbpkg $pbpkg = $pkgs{$pbpkg}\n";
|
---|
1575 | }
|
---|
1576 | close(PKG);
|
---|
1577 |
|
---|
1578 | # Keep track of what is generated by default
|
---|
1579 | # We need to store the dir and info on version-tag
|
---|
1580 | # Base our content on the existing .pb file
|
---|
1581 | copy("$ENV{'PBROOTDIR'}/$ENV{'PBPROJ'}.pb","$ENV{'PBDESTDIR'}/pbrc");
|
---|
1582 | open(LAST,">> $ENV{'PBDESTDIR'}/pbrc") || die "Unable to create $ENV{'PBDESTDIR'}/pbrc";
|
---|
1583 | print LAST "pbroot $ENV{'PBPROJ'} = $ENV{'PBROOTDIR'}\n";
|
---|
1584 | print LAST "projver $ENV{'PBPROJ'} = $ENV{'PBPROJVER'}\n";
|
---|
1585 | print LAST "projtag $ENV{'PBPROJ'} = $ENV{'PBPROJTAG'}\n";
|
---|
1586 | print LAST "pbpackager $ENV{'PBPROJ'} = $ENV{'PBPACKAGER'}\n";
|
---|
1587 | print LAST "pbextdir $ENV{'PBPROJ'} = $pbextdir\n";
|
---|
1588 | close(LAST);
|
---|
1589 | }
|
---|
1590 |
|
---|
1591 | sub pb_test2pkg {
|
---|
1592 | # Get the running distro to test on
|
---|
1593 | my $pbos = pb_distro_get_context();
|
---|
1594 |
|
---|
1595 | # Get list of packages to test
|
---|
1596 | # Get content saved in cms2build
|
---|
1597 | my $ptr = pb_get_pkg();
|
---|
1598 | @pkgs = @$ptr;
|
---|
1599 |
|
---|
1600 | # Additional potential repo
|
---|
1601 | pb_distro_setuprepo($pbos);
|
---|
1602 | foreach my $pbpkg (@pkgs) {
|
---|
1603 | # We need to install the package to test, and deps brought with it
|
---|
1604 | pb_distro_installdeps(undef,$pbos,$pbpkg);
|
---|
1605 | pb_system("$ENV{'PBDESTDIR'}/pbtest","Launching test for $pbpkg","verbose");
|
---|
1606 | }
|
---|
1607 | }
|
---|
1608 |
|
---|
1609 | sub pb_build2prep {
|
---|
1610 |
|
---|
1611 | pb_log(0,"INFO: ------ Starting to prepare build environement ------\n");
|
---|
1612 |
|
---|
1613 | # Get the running distro to build on
|
---|
1614 | my $pbos = pb_distro_get_context();
|
---|
1615 |
|
---|
1616 | # If needed we may add repository to the build env
|
---|
1617 | pb_distro_setuprepo($pbos);
|
---|
1618 |
|
---|
1619 | # Get list of packages to build
|
---|
1620 | my $ptr = pb_get_pkg();
|
---|
1621 | @pkgs = @$ptr;
|
---|
1622 |
|
---|
1623 | # Get content saved in cms2build
|
---|
1624 | my ($pkg) = pb_conf_read("$ENV{'PBDESTDIR'}/$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}.pb","pbpkg");
|
---|
1625 | $pkg = { } if (not defined $pkg);
|
---|
1626 | my $pbextdir = pb_get_extdir();
|
---|
1627 |
|
---|
1628 | foreach my $pbpkg (@pkgs) {
|
---|
1629 | my $vertag = $pkg->{$pbpkg};
|
---|
1630 | my @buildfiles;
|
---|
1631 | pb_log(2,"Vertag: $vertag\n");
|
---|
1632 | # get the version of the current package - maybe different
|
---|
1633 | ($pbver,$pbtag) = split(/-/,$vertag);
|
---|
1634 |
|
---|
1635 | my $src2="$ENV{'PBDESTDIR'}/$pbpkg-$pbver$pbextdir.pbconf.tar.gz";
|
---|
1636 | pb_log(2,"Pbconf file: $src2\n");
|
---|
1637 | pb_log(2,"Working directory: $ENV{'PBBUILDDIR'}\n");
|
---|
1638 |
|
---|
1639 | my $ftype;
|
---|
1640 | if ($pbos->{'type'} eq "rpm") {
|
---|
1641 | $ftype = "\.spec\$";
|
---|
1642 | } elsif ($pbos->{'type'} eq "deb") {
|
---|
1643 | $ftype = "\/control\$";
|
---|
1644 | } elsif ($pbos->{'type'} eq "ebuild") {
|
---|
1645 | $ftype = "\.ebuild\$";
|
---|
1646 | } elsif ($pbos->{'type'} eq "tgz") {
|
---|
1647 | $ftype = "\/pbslack\$";
|
---|
1648 | } elsif ($pbos->{'type'} eq "pkg") {
|
---|
1649 | # Solaris - no deps management up to Solaris 10
|
---|
1650 | } elsif ($pbos->{'type'} eq "hpux") {
|
---|
1651 | # TODO: HP-UX
|
---|
1652 | } else {
|
---|
1653 | die "Unknown OS type format $pbos->{'type'}";
|
---|
1654 | }
|
---|
1655 |
|
---|
1656 | @buildfiles = pb_extract_build_files($src2,"$pbpkg-$pbver$pbextdir/pbconf/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}/","$ENV{'PBTMP'}","spec");
|
---|
1657 | pb_log(2,"buildfiles ".Dumper(\@buildfiles)."\n");
|
---|
1658 | foreach my $f (@buildfiles) {
|
---|
1659 | if ($f =~ /$ftype/) {
|
---|
1660 | pb_distro_installdeps($f,$pbos);
|
---|
1661 | }
|
---|
1662 | }
|
---|
1663 | }
|
---|
1664 | pb_log(0,"INFO: ------ Finished preparing build environment ------\n");
|
---|
1665 | }
|
---|
1666 |
|
---|
1667 | sub pb_build2pkg {
|
---|
1668 |
|
---|
1669 | my $do_install = shift;
|
---|
1670 |
|
---|
1671 | pb_log(0,"INFO: ------ Starting to build package ------\n");
|
---|
1672 | # Get the running distro to build on
|
---|
1673 | my $pbos = pb_distro_get_context();
|
---|
1674 |
|
---|
1675 | # Get list of packages to build
|
---|
1676 | my $ptr = pb_get_pkg();
|
---|
1677 | @pkgs = @$ptr;
|
---|
1678 |
|
---|
1679 | # Get content saved in cms2build
|
---|
1680 | my ($pkg) = pb_conf_read("$ENV{'PBDESTDIR'}/$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}.pb","pbpkg");
|
---|
1681 | $pkg = { } if (not defined $pkg);
|
---|
1682 | my $pbextdir = pb_get_extdir();
|
---|
1683 |
|
---|
1684 | pb_mkdir_p("$ENV{'PBBUILDDIR'}") if (! -d "$ENV{'PBBUILDDIR'}");
|
---|
1685 | chdir "$ENV{'PBBUILDDIR'}" || die "Unable to chdir to $ENV{'PBBUILDDIR'}";
|
---|
1686 | my $made = ""; # pkgs made during build
|
---|
1687 | my $pm;
|
---|
1688 | my $all_ok = 1;
|
---|
1689 |
|
---|
1690 | if (defined $pbparallel) {
|
---|
1691 | $pm = new Parallel::ForkManager($pbparallel);
|
---|
1692 | $pm->run_on_finish(sub { my ($pid, $code, $id, $signal, $dump) = @_;
|
---|
1693 | $all_ok = 0 unless (($code == 0) && ($signal == 0) && ($dump == 0)); });
|
---|
1694 | }
|
---|
1695 |
|
---|
1696 | # We need to communicate info back from the children if parallel so prepare a dir for that
|
---|
1697 | my $tmpd = "$ENV{'PBTMP'}/build.$$";
|
---|
1698 | pb_mkdir_p($tmpd) if (defined $pbparallel);
|
---|
1699 |
|
---|
1700 | # Do it here as this doesn't work in // mode as it dies if it fails
|
---|
1701 | if ($pbos->{'type'} eq "rpm") {
|
---|
1702 | foreach my $d ('RPMS','SRPMS','SPECS','SOURCES','BUILD') {
|
---|
1703 | if (! -d "$ENV{'PBBUILDDIR'}/$d") {
|
---|
1704 | pb_mkdir_p("$ENV{'PBBUILDDIR'}/$d");
|
---|
1705 | }
|
---|
1706 | }
|
---|
1707 | }
|
---|
1708 |
|
---|
1709 | foreach my $pbpkg (@pkgs) {
|
---|
1710 | $pm->start and next if (defined $pbparallel);
|
---|
1711 |
|
---|
1712 | my $vertag = $pkg->{$pbpkg};
|
---|
1713 | pb_log(2,"Vertag: $vertag\n");
|
---|
1714 | # get the version of the current package - maybe different
|
---|
1715 | ($pbver,$pbtag) = split(/-/,$vertag);
|
---|
1716 |
|
---|
1717 | my $src="$ENV{'PBDESTDIR'}/$pbpkg-$pbver$pbextdir.tar.gz";
|
---|
1718 | my $src2="$ENV{'PBDESTDIR'}/$pbpkg-$pbver$pbextdir.pbconf.tar.gz";
|
---|
1719 | pb_log(2,"Source file: $src\n");
|
---|
1720 | pb_log(2,"Pbconf file: $src2\n");
|
---|
1721 |
|
---|
1722 | pb_log(2,"Working directory: $ENV{'PBBUILDDIR'}\n");
|
---|
1723 | if ($pbos->{'type'} eq "rpm") {
|
---|
1724 | # Remove in case a previous link/file was there
|
---|
1725 | unlink "$ENV{'PBBUILDDIR'}/SOURCES/".basename($src);
|
---|
1726 | symlink "$src","$ENV{'PBBUILDDIR'}/SOURCES/".basename($src) || die "Unable to symlink $src in $ENV{'PBBUILDDIR'}/SOURCES";
|
---|
1727 | # We need to first extract the spec file
|
---|
1728 | my @specfile = pb_extract_build_files($src2,"$pbpkg-$pbver$pbextdir/pbconf/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}/","$ENV{'PBBUILDDIR'}/SPECS","spec");
|
---|
1729 |
|
---|
1730 | # We need to handle potential patches to upstream sources
|
---|
1731 | pb_extract_build_files($src2,"$pbpkg-$pbver$pbextdir/pbconf/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}/pbpatch/","$ENV{'PBBUILDDIR'}/SOURCES","patch");
|
---|
1732 |
|
---|
1733 | # We need to handle potential additional sources to upstream sources
|
---|
1734 | pb_extract_build_files($src2,"$pbpkg-$pbver$pbextdir/pbconf/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}/pbsrc/","$ENV{'PBBUILDDIR'}/SOURCES","src");
|
---|
1735 |
|
---|
1736 | pb_log(2,"specfile: ".Dumper(\@specfile)."\n");
|
---|
1737 | # set LANGUAGE to check for correct log messages
|
---|
1738 | $ENV{'LANGUAGE'}="C";
|
---|
1739 | # Older Redhat use _target_platform in %configure incorrectly
|
---|
1740 | my $specialdef = "";
|
---|
1741 | if (($pbos->{'name'} eq "rhel") && ($pbos->{'version'} eq "2.1")) {
|
---|
1742 | $specialdef = "--define \'_target_platform \"\"\'";
|
---|
1743 | }
|
---|
1744 | my $buildcmd = "rpmbuild";
|
---|
1745 | if (($pbos->{'name'} eq "redhat") && ($pbos->{'version'} =~ /^6/)){
|
---|
1746 | $buildcmd = "rpm" ;
|
---|
1747 | }
|
---|
1748 |
|
---|
1749 | foreach my $f (@specfile) {
|
---|
1750 | if ($f =~ /\.spec$/) {
|
---|
1751 | pb_system("$buildcmd $specialdef --define \"packager $ENV{'PBPACKAGER'}\" --define \"_topdir $ENV{'PBBUILDDIR'}\" -ba $f","Building package with $f under $ENV{'PBBUILDDIR'}","verbose");
|
---|
1752 | last;
|
---|
1753 | }
|
---|
1754 | }
|
---|
1755 | # Get the name of the generated packages
|
---|
1756 | open(LOG,"$ENV{'PBTMP'}/system.$$.log") || die "Unable to open $ENV{'PBTMP'}/system.$$.log";
|
---|
1757 | while (<LOG>) {
|
---|
1758 | chomp($_);
|
---|
1759 | next if ($_ !~ /^Wrote:/);
|
---|
1760 | s|.*/([S]*RPMS.*)|$1|;
|
---|
1761 | $made .=" $_";
|
---|
1762 | }
|
---|
1763 | close(LOG);
|
---|
1764 |
|
---|
1765 | } elsif ($pbos->{'type'} eq "deb") {
|
---|
1766 | pb_system("tar xfz $src","Extracting sources");
|
---|
1767 | pb_system("tar xfz $src2","Extracting pbconf");
|
---|
1768 |
|
---|
1769 | chdir "$pbpkg-$pbver$pbextdir" || die "Unable to chdir to $pbpkg-$pbver$pbextdir";
|
---|
1770 | pb_rm_rf("debian");
|
---|
1771 | my $confdir = "pbconf/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}";
|
---|
1772 | die "Configuration directory $confdir is not a directory\nIs os description listed in your {ve,vm,rm}list values?" if (not -d $confdir);
|
---|
1773 | symlink "$confdir","debian" || die "Unable to symlink 'debian' to $confdir";
|
---|
1774 | chmod 0755,"debian/rules";
|
---|
1775 |
|
---|
1776 | # We need to handle potential patches to upstream sources
|
---|
1777 | pb_mkdir_p("debian/patches");
|
---|
1778 | my @f = pb_extract_build_files($src2,"$pbpkg-$pbver$pbextdir/pbconf/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}/pbpatch/","debian/patches","patch");
|
---|
1779 |
|
---|
1780 | # By default we use format 1.0 - Cf man dpkg-source
|
---|
1781 | my $debsrcfmt = "1.0";
|
---|
1782 | my $debsrcfile = "debian/source/format";
|
---|
1783 | if (-f $debsrcfile) {
|
---|
1784 | $debsrcfmt = pb_get_content($debsrcfile);
|
---|
1785 | }
|
---|
1786 |
|
---|
1787 | if ($debsrcfmt =~ /^3.*quilt/) {
|
---|
1788 | # If we use quilt to manage patches, we then setup the env correctly
|
---|
1789 | # as per http://pkg-perl.alioth.debian.org/howto/quilt.html
|
---|
1790 | # Generate Debian patch series for quilt
|
---|
1791 | open(SERIE,"> debian/patches/series") || die "Unable to write in debian/patches/series";
|
---|
1792 | $ENV{'QUILT_PATCHES'}="debian/patches";
|
---|
1793 | } else {
|
---|
1794 | # If we use dpatch to manage patches, we then setup the 00list file as well
|
---|
1795 | open(SERIE,"> debian/patches/00list") || die "Unable to write in debian/patches/00list";
|
---|
1796 | }
|
---|
1797 | foreach my $f (sort @f) {
|
---|
1798 | # Skip the script made to apply the patches to the Debian tree
|
---|
1799 | next if ($f =~ /pbapplypatch/);
|
---|
1800 | # We also need to uncompress them
|
---|
1801 | pb_system("gzip -d $f","","quiet");
|
---|
1802 | $f =~ s/\.gz$//;
|
---|
1803 | print SERIE "$f\n";
|
---|
1804 | }
|
---|
1805 | close(SERIE);
|
---|
1806 | if (@f) {
|
---|
1807 | # We have patches...
|
---|
1808 | my $patch_file = "debian/patches/pbapplypatch";
|
---|
1809 | if (($debsrcfmt =~ /^1.*/) && (-x $patch_file)) {
|
---|
1810 | # In that case we need to apply the patches ourselves locally
|
---|
1811 | pb_system("cat $patch_file","APPLY","verbose");
|
---|
1812 | pb_system("$patch_file","Applying patches to $pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'} tree");
|
---|
1813 | }
|
---|
1814 | # ...so modify the name of files to be Debian compliant
|
---|
1815 | move("../$src","../$pbpkg-$pbver$pbextdir.orig.tar.gz");
|
---|
1816 | }
|
---|
1817 |
|
---|
1818 | # We need to handle potential additional sources to upstream sources
|
---|
1819 | #pb_extract_build_files($src2,"$pbpkg-$pbver$pbextdir/pbconf/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}/pbsrc/","$ENV{'PBBUILDDIR'}/debian","src");
|
---|
1820 |
|
---|
1821 | pb_system("dpkg-buildpackage -us -uc -rfakeroot","Building package","verbose");
|
---|
1822 |
|
---|
1823 | # Get the name of the generated packages
|
---|
1824 | open(LOG,"$ENV{'PBTMP'}/system.$$.log") || die "Unable to open $ENV{'PBTMP'}/system.$$.log";
|
---|
1825 | while (<LOG>) {
|
---|
1826 | chomp();
|
---|
1827 | pb_log(3,"In loop: $_\n");
|
---|
1828 | next unless (/^dpkg-deb.*\.\.\/(.*_.*\.deb).*/);
|
---|
1829 | my $tmp = $1;
|
---|
1830 | #doesn't work in my case
|
---|
1831 | #die "Missing file $tmp" if (not -f "../$tmp");
|
---|
1832 | $made = "$made $tmp";
|
---|
1833 | }
|
---|
1834 | close(LOG);
|
---|
1835 | pb_log(2,"Now made is: $made\n");
|
---|
1836 |
|
---|
1837 | open(CTRL,"debian/control") || die "Unable to open debian/control: $!";
|
---|
1838 | while (<CTRL>) {
|
---|
1839 | pb_log(3,"In loop: $_\n");
|
---|
1840 | next unless (/^Source: (\S+)/o);
|
---|
1841 | my $tmp = $1;
|
---|
1842 | $made = "$made $tmp"."_*.dsc $tmp"."_*.tar.gz $tmp"."_*.changes";
|
---|
1843 | #doesn't work in my case
|
---|
1844 | #foreach my $glob (("$1\_*.changes", "$1\_*.dsc", "$1\_*.tar.gz")) {
|
---|
1845 | #my @file = glob($glob);
|
---|
1846 | #die "Missing file for $glob" unless (@file > 0);
|
---|
1847 | #die "Too many files for $glob" if (@file > 1);
|
---|
1848 | #die "Missing file $file[0]" if (not -f $file[0]);
|
---|
1849 | #$made .= " $file[0]";
|
---|
1850 | #}
|
---|
1851 | }
|
---|
1852 | close(CTRL);
|
---|
1853 | #pb_display_file("$ENV{'PBTMP'}/system.$$.log");
|
---|
1854 | pb_log(2,"Finally made is: $made\n");
|
---|
1855 |
|
---|
1856 | chdir ".." || die "Unable to chdir to parent dir";
|
---|
1857 | pb_rm_rf("$pbpkg-$pbver$pbextdir");
|
---|
1858 | } elsif ($pbos->{'type'} eq "ebuild") {
|
---|
1859 | my @ebuildfile;
|
---|
1860 | # For gentoo we need to take pb as subsystem name
|
---|
1861 | # We put every apps here under sys-apps. hope it's correct
|
---|
1862 | # We use pb's home dir in order to have a single OVERLAY line
|
---|
1863 | my $tmpe = "$ENV{'HOME'}/portage/pb/sys-apps/$pbpkg";
|
---|
1864 | pb_mkdir_p($tmpe) if (! -d "$tmpe");
|
---|
1865 | pb_mkdir_p("$ENV{'HOME'}/portage/distfiles") if (! -d "$ENV{'HOME'}/portage/distfiles");
|
---|
1866 |
|
---|
1867 | # We need to first extract the ebuild file
|
---|
1868 | @ebuildfile = pb_extract_build_files($src2,"$pbpkg-$pbver$pbextdir/pbconf/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}/","$tmpe","ebuild");
|
---|
1869 |
|
---|
1870 | # Prepare the build env for gentoo
|
---|
1871 | my $found = 0;
|
---|
1872 | my $pbbd = $ENV{'HOME'};
|
---|
1873 | $pbbd =~ s|/|\\/|g;
|
---|
1874 | if (-r "/etc/make.conf") {
|
---|
1875 | open(MAKE,"/etc/make.conf");
|
---|
1876 | while (<MAKE>) {
|
---|
1877 | $found = 1 if (/$pbbd\/portage/);
|
---|
1878 | }
|
---|
1879 | close(MAKE);
|
---|
1880 | }
|
---|
1881 | if ($found == 0) {
|
---|
1882 | pb_system("sudo sh -c 'echo PORTDIR_OVERLAY=\"$ENV{'HOME'}/portage\" >> /etc/make.conf'");
|
---|
1883 | }
|
---|
1884 | #$found = 0;
|
---|
1885 | #if (-r "/etc/portage/package.keywords") {
|
---|
1886 | #open(KEYW,"/etc/portage/package.keywords");
|
---|
1887 | #while (<KEYW>) {
|
---|
1888 | #$found = 1 if (/portage\/pb/);
|
---|
1889 | #}
|
---|
1890 | #close(KEYW);
|
---|
1891 | #}
|
---|
1892 | #if ($found == 0) {
|
---|
1893 | #pb_system("sudo sh -c \"echo portage/pb >> /etc/portage/package.keywords\"");
|
---|
1894 | #}
|
---|
1895 |
|
---|
1896 | # Build
|
---|
1897 | foreach my $f (@ebuildfile) {
|
---|
1898 | if ($f =~ /\.ebuild$/) {
|
---|
1899 | move($f,"$tmpe/$pbpkg-$pbver.ebuild");
|
---|
1900 | pb_system("cd $tmpe ; ebuild $pbpkg-$pbver.ebuild clean ; ebuild $pbpkg-$pbver.ebuild digest ; ebuild $pbpkg-$pbver.ebuild package","verbose");
|
---|
1901 | # Now move it where pb expects it
|
---|
1902 | pb_mkdir_p("$ENV{'PBBUILDDIR'}/portage/pb/sys-apps/$pbpkg");
|
---|
1903 | if ($pbtag eq 0) {
|
---|
1904 | # This is assumed to be a test version
|
---|
1905 | my $nver = substr($pbver,0,-14);
|
---|
1906 | my $ntag = substr($pbver,-14);
|
---|
1907 | my $ebtg = "portage/pb/sys-apps/$pbpkg/$pbpkg-$nver"."_p$ntag.ebuild";
|
---|
1908 | move("$tmpe/$pbpkg-$pbver.ebuild","$ENV{'PBBUILDDIR'}/$ebtg");
|
---|
1909 | $made="$made $ebtg";
|
---|
1910 | } else {
|
---|
1911 | my $ebtg = "portage/pb/sys-apps/$pbpkg/$pbpkg-$pbver-r$pbtag.ebuild";
|
---|
1912 | move("$tmpe/$pbpkg-$pbver.ebuild","$ENV{'PBBUILDDIR'}/$ebtg");
|
---|
1913 | $made="$made $ebtg";
|
---|
1914 | }
|
---|
1915 | }
|
---|
1916 | }
|
---|
1917 |
|
---|
1918 | } elsif ($pbos->{'type'} eq "tgz") {
|
---|
1919 | # Slackware family
|
---|
1920 | $made="$made $pbpkg/$pbpkg-$pbver-*-$pbtag.tgz";
|
---|
1921 |
|
---|
1922 | pb_system("tar xfz $src","Extracting sources");
|
---|
1923 | pb_system("tar xfz $src2","Extracting pbconf");
|
---|
1924 | chdir "$pbpkg-$pbver$pbextdir" || die "Unable to chdir to $pbpkg-$pbver$pbextdir";
|
---|
1925 | symlink "pbconf/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}","install" || die "Unable to symlink to pbconf/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}";
|
---|
1926 | if (-x "install/pbslack") {
|
---|
1927 | pb_system("./install/pbslack","Building software");
|
---|
1928 | pb_system("sudo /sbin/makepkg -p -l y -c y $pbpkg","Packaging $pbpkg","verbose");
|
---|
1929 | }
|
---|
1930 | chdir ".." || die "Unable to chdir to parent dir";
|
---|
1931 | pb_rm_rf("$pbpkg-$pbver$pbextdir");
|
---|
1932 | } elsif ($pbos->{'type'} eq "pkg") {
|
---|
1933 | # Solaris
|
---|
1934 | $made="$made $pbpkg-$pbver-$pbtag.pkg.gz";
|
---|
1935 | my $pkgdestdir="$ENV{'PBBUILDDIR'}/install";
|
---|
1936 |
|
---|
1937 | # Will host resulting packages
|
---|
1938 | pb_mkdir_p("$pbos->{'type'}");
|
---|
1939 | pb_mkdir_p("$pkgdestdir/pbdelivery");
|
---|
1940 | pb_system("tar xfz $src","Extracting sources under $ENV{'PBBUILDDIR'}");
|
---|
1941 | pb_system("tar xfz $src2","Extracting pbconf under $ENV{'PBBUILDDIR'}");
|
---|
1942 | # We need to handle potential patches to upstream sources
|
---|
1943 | pb_extract_build_files($src2,"$pbpkg-$pbver$pbextdir/pbconf/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}/pbpatch/","$ENV{'PBBUILDDIR'}","patch");
|
---|
1944 |
|
---|
1945 | # We need to handle potential additional sources to upstream sources
|
---|
1946 | pb_extract_build_files($src2,"$pbpkg-$pbver$pbextdir/pbconf/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}/pbsrc/","$ENV{'PBBUILDDIR'}","src");
|
---|
1947 |
|
---|
1948 | chdir "$pbpkg-$pbver$pbextdir" || die "Unable to chdir to $pbpkg-$pbver$pbextdir";
|
---|
1949 | if (-f "pbconf/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}/pbbuild") {
|
---|
1950 | chmod 0755,"pbconf/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}/pbbuild";
|
---|
1951 | # pkginfo file is mandatory
|
---|
1952 | die "Unable to find pkginfo file in pbconf/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}" if (! -f "pbconf/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}/pkginfo");
|
---|
1953 | # Build
|
---|
1954 | pb_system("pbconf/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}/pbbuild $pkgdestdir/pbdelivery","Building software and installing under $pkgdestdir/pbdelivery");
|
---|
1955 | # Copy complementary files
|
---|
1956 | if (-f "pbconf/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}/prototype") {
|
---|
1957 | copy("pbconf/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}/prototype", $pkgdestdir)
|
---|
1958 | } else {
|
---|
1959 | # No prototype provided, calculating it
|
---|
1960 | open(PROTO,"> $pkgdestdir/prototype") || die "Unable to create prototype file";
|
---|
1961 | print PROTO "i pkginfo\n";
|
---|
1962 | print PROTO "i depend\n" if (-f "pbconf/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}/depend");
|
---|
1963 | $ENV{'PBSOLDESTDIR'} = "$pkgdestdir/pbdelivery";
|
---|
1964 | find(\&create_solaris_prototype, "$pkgdestdir/pbdelivery");
|
---|
1965 | }
|
---|
1966 | copy("pbconf/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}/depend", $pkgdestdir) if (-f "pbconf/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}/depend");
|
---|
1967 | copy("pbconf/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}/pkginfo", $pkgdestdir);
|
---|
1968 | pb_system("cd $pkgdestdir/pbdelivery ; pkgmk -o -f ../prototype -r $pkgdestdir/pbdelivery -d $ENV{'PBBUILDDIR'}/$pbos->{'type'}","Packaging $pbpkg","verbose");
|
---|
1969 | pb_system("cd $ENV{'PBBUILDDIR'}/$pbos->{'type'} ; echo \"\" | pkgtrans -o -n -s $ENV{'PBBUILDDIR'}/$pbos->{'type'} $ENV{'PBBUILDDIR'}/$pbpkg-$pbver-$pbtag.pkg all","Transforming $pbpkg","verbose");
|
---|
1970 | pb_system("cd $ENV{'PBBUILDDIR'} ; gzip -9f $pbpkg-$pbver-$pbtag.pkg","Compressing $pbpkg-$pbver-$pbtag.pkg","verbose");
|
---|
1971 | } else {
|
---|
1972 | pb_log(0,"No pbconf/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}/pbbuild file found for $pbpkg-$pbver\n");
|
---|
1973 | }
|
---|
1974 | chdir ".." || die "Unable to chdir to parent dir";
|
---|
1975 | pb_rm_rf("$pbpkg-$pbver$pbextdir","$ENV{'PBBUILDDIR'}/$pbos->{'type'}","$pkgdestdir");
|
---|
1976 | } elsif ($pbos->{'type'} eq "hpux") {
|
---|
1977 | # HP-UX
|
---|
1978 | pb_system("tar xfz $src","Extracting sources");
|
---|
1979 | pb_system("tar xfz $src2","Extracting pbconf");
|
---|
1980 |
|
---|
1981 | chdir "$pbpkg-$pbver$pbextdir" || die "Unable to chdir to $pbpkg-$pbver$pbextdir";
|
---|
1982 | pb_system("buildpackage ","Building package","verbose");
|
---|
1983 | # Get the name of the generated packages
|
---|
1984 | open(LOG,"$ENV{'PBTMP'}/system.$$.log") || die "Unable to open $ENV{'PBTMP'}/system.$$.log";
|
---|
1985 | while (<LOG>) {
|
---|
1986 | chomp();
|
---|
1987 | my $tmp = $_;
|
---|
1988 | next if ($tmp !~ /^SD BUILD.*:/);
|
---|
1989 | $tmp =~ s|.*../(.*)_(.*).sd.*|$1|;
|
---|
1990 | $made = "$made $tmp"."_*.sd";
|
---|
1991 | }
|
---|
1992 | close(LOG);
|
---|
1993 | $made="$made $pbpkg-$pbver-$pbtag.sd";
|
---|
1994 |
|
---|
1995 | chdir ".." || die "Unable to chdir to parent dir";
|
---|
1996 | pb_rm_rf("$pbpkg-$pbver$pbextdir");
|
---|
1997 | } else {
|
---|
1998 | die "Unknown OS type format $pbos->{'type'}";
|
---|
1999 | }
|
---|
2000 | if (defined $pbparallel) {
|
---|
2001 | # Communicate results back to parent
|
---|
2002 | pb_set_content("$tmpd/$$",$made);
|
---|
2003 | $pm->finish;
|
---|
2004 | }
|
---|
2005 | }
|
---|
2006 | if (defined $pbparallel) {
|
---|
2007 | # In the parent, we need to get the result from the children
|
---|
2008 | $pm->wait_all_children;
|
---|
2009 | foreach my $f (<$tmpd/*>) {
|
---|
2010 | $made .= " ".pb_get_content($f);
|
---|
2011 | }
|
---|
2012 | die "Aborting, one or more of the children failed." if ((not $all_ok) && ($Global::pb_stop_on_error));
|
---|
2013 | pb_rm_rf($tmpd);
|
---|
2014 | }
|
---|
2015 |
|
---|
2016 | # Sign packages
|
---|
2017 | pb_sign_pkgs($pbos,$made);
|
---|
2018 | pb_log(0,"INFO: ------ Finished building package ------\n");
|
---|
2019 |
|
---|
2020 | # Find the appropriate check cmd/opts
|
---|
2021 | my ($chkcmd,$chkopt) = pb_distro_get_param($pbos,pb_conf_get_if("oschkcmd","oschkopt"));
|
---|
2022 |
|
---|
2023 | my $ret = "";
|
---|
2024 | # Packages check if needed
|
---|
2025 | if ($pbos->{'type'} eq "rpm") {
|
---|
2026 | if ((defined $chkcmd) && (-x $chkcmd)) {
|
---|
2027 | my $cmd = "$chkcmd";
|
---|
2028 | $cmd .= " $chkopt" if (defined $chkopt);
|
---|
2029 | $cmd .= " $made";
|
---|
2030 | my $ret = pb_system("$cmd","Checking validity of rpms with $chkcmd","mayfailverbose");
|
---|
2031 | pb_log(0,"ERROR: when checking packages validity\n") if ($ret ne 0);
|
---|
2032 | }
|
---|
2033 | my $rpms ="";
|
---|
2034 | my $srpms ="";
|
---|
2035 | foreach my $f (split(/ /,$made)) {
|
---|
2036 | $rpms .= "$ENV{'PBBUILDDIR'}/$f " if ($f =~ /^RPMS\//);
|
---|
2037 | $srpms .= "$ENV{'PBBUILDDIR'}/$f " if ($f =~ /^SRPMS\//);
|
---|
2038 | }
|
---|
2039 | pb_log(0,"SRPM packages generated: $srpms\n");
|
---|
2040 | pb_log(0,"RPM packages generated: $rpms\n");
|
---|
2041 | $ret = $rpms;
|
---|
2042 | } elsif ($pbos->{'type'} eq "deb") {
|
---|
2043 | my $made2 = "";
|
---|
2044 | foreach my $f (split(/ /,$made)) {
|
---|
2045 | $made2 .= "$f " if ($f =~ /\.deb$/);
|
---|
2046 | }
|
---|
2047 | if ((defined $chkcmd) && (-x $chkcmd)) {
|
---|
2048 | my $ret = pb_system("$chkcmd $chkopt $made2","Checking validity of debs with $chkcmd","mayfail");
|
---|
2049 | pb_log(0,"ERROR: when checking packages validity\n") if ($ret ne 0);
|
---|
2050 | }
|
---|
2051 | pb_log(0,"deb packages generated: $made2\n");
|
---|
2052 | $ret = $made2;
|
---|
2053 | } else {
|
---|
2054 | pb_log(0,"No check done for $pbos->{'type'} yet\n");
|
---|
2055 | pb_log(0,"Packages generated: $made\n");
|
---|
2056 | $ret = $made;
|
---|
2057 | }
|
---|
2058 |
|
---|
2059 | # Keep track of what is generated so that we can get them back from VMs/RMs
|
---|
2060 | my $pbstore = "$ENV{'PBBUILDDIR'}/pbgen-$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}-$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}";
|
---|
2061 | open(KEEP,"> $pbstore") || die "Unable to create $pbstore $!";
|
---|
2062 | print KEEP "$made\n";
|
---|
2063 | close(KEEP);
|
---|
2064 | pb_distro_installdeps(undef,$pbos,$ret) if ($do_install);
|
---|
2065 | }
|
---|
2066 |
|
---|
2067 | sub pb_build2ssh {
|
---|
2068 | pb_send2target("Sources");
|
---|
2069 | pb_send2target("CPAN");
|
---|
2070 | }
|
---|
2071 |
|
---|
2072 | sub pb_pkg2ssh {
|
---|
2073 | pb_send2target("Packages");
|
---|
2074 | }
|
---|
2075 |
|
---|
2076 | # By default deliver to the the public site hosting the
|
---|
2077 | # ftp structure (or whatever) or a VM/VE/RM
|
---|
2078 | sub pb_send2target {
|
---|
2079 |
|
---|
2080 | my $cmt = shift;
|
---|
2081 | my $pbscript1 = shift;
|
---|
2082 | my $v = shift;
|
---|
2083 | my $vexist = shift; # 0 is FALSE
|
---|
2084 | my $vpid = shift; # 0 is FALSE
|
---|
2085 | my $snapme = shift; # 0 is FALSE
|
---|
2086 | my $pbstep = shift; # 3 is usage of container
|
---|
2087 |
|
---|
2088 | $vexist = 0 if (not defined $vexist);
|
---|
2089 | $vpid = 0 if (not defined $vpid);
|
---|
2090 | $snapme = 0 if (not defined $snapme);
|
---|
2091 | $pbstep = 3 if (not defined $pbstep);
|
---|
2092 | pb_log(2,"DEBUG: pb_send2target($cmt,".Dumper($v).",$vexist,$vpid)\n");
|
---|
2093 | my $host = "sshhost";
|
---|
2094 | my $login = "sshlogin";
|
---|
2095 | my $dir = "sshdir";
|
---|
2096 | my $port = "sshport";
|
---|
2097 | my $conf = "sshconf";
|
---|
2098 | my $tmout = undef;
|
---|
2099 | my $path = undef;
|
---|
2100 |
|
---|
2101 | if ($cmt =~ /^VM/) {
|
---|
2102 | $login = "vmlogin";
|
---|
2103 | $dir = "pbdefdir";
|
---|
2104 | # Specific VM
|
---|
2105 | $tmout = "vmtmout";
|
---|
2106 | $path = "vmpath";
|
---|
2107 | $host = "vmhost";
|
---|
2108 | $port = "vmport";
|
---|
2109 | } elsif ($cmt =~ /^RM/) {
|
---|
2110 | $login = "rmlogin";
|
---|
2111 | $dir = "pbdefdir";
|
---|
2112 | # Specific RM
|
---|
2113 | $tmout = "rmtmout";
|
---|
2114 | $path = "rmpath";
|
---|
2115 | $host = "rmhost";
|
---|
2116 | $port = "rmport";
|
---|
2117 | } elsif ($cmt =~ /^VE/) {
|
---|
2118 | $login = "velogin";
|
---|
2119 | $dir = "pbdefdir";
|
---|
2120 | # Specific VE
|
---|
2121 | $path = "vepath";
|
---|
2122 | $conf = "rbsconf";
|
---|
2123 | } elsif ($cmt eq "Web") {
|
---|
2124 | $host = "websshhost";
|
---|
2125 | $login = "websshlogin";
|
---|
2126 | $dir = "websshdir";
|
---|
2127 | $port = "websshport";
|
---|
2128 | } elsif ($cmt eq "CPAN") {
|
---|
2129 | $host = "cpanpause";
|
---|
2130 | $login = "";
|
---|
2131 | $dir = "cpandir";
|
---|
2132 | $port = "";
|
---|
2133 | }
|
---|
2134 | my $cmd = "";
|
---|
2135 | my $src = "";
|
---|
2136 | my $cpanpkg = 0;
|
---|
2137 | my $pbos;
|
---|
2138 |
|
---|
2139 | my $pbextdir = pb_get_extdir();
|
---|
2140 |
|
---|
2141 | if ($cmt ne "Announce") {
|
---|
2142 | # Get list of packages to build
|
---|
2143 | my $ptr = pb_get_pkg();
|
---|
2144 | my @pkgs = ();
|
---|
2145 | @pkgs = @$ptr if (defined $ptr);
|
---|
2146 |
|
---|
2147 | # Get the running distro to consider
|
---|
2148 | $pbos = pb_distro_get_context($v);
|
---|
2149 |
|
---|
2150 | my $pkg = { };
|
---|
2151 | # Get content saved in cms2build
|
---|
2152 | if ((defined $ENV{'PBDESTDIR'}) && (defined $ENV{'PBPROJVER'}) && (defined $ENV{'PBPROJTAG'}) && (-f "$ENV{'PBDESTDIR'}/$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}.pb")) {
|
---|
2153 | ($pkg) = pb_conf_read("$ENV{'PBDESTDIR'}/$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}.pb","pbpkg");
|
---|
2154 | $pkg = { } if (not defined $pkg);
|
---|
2155 | }
|
---|
2156 |
|
---|
2157 | pb_mkdir_p("$ENV{'PBBUILDDIR'}") if (! -d "$ENV{'PBBUILDDIR'}");
|
---|
2158 | chdir "$ENV{'PBBUILDDIR'}" || die "Unable to chdir to $ENV{'PBBUILDDIR'}";
|
---|
2159 | foreach my $pbpkg (@pkgs) {
|
---|
2160 | my $vertag = $pkg->{$pbpkg};
|
---|
2161 | # get the version of the current package - maybe different
|
---|
2162 | pb_log(2,"Vertag: $vertag\n");
|
---|
2163 | ($pbver,$pbtag) = split(/-/,$vertag);
|
---|
2164 |
|
---|
2165 | if (($cmt eq "Sources") || ($cmt =~ /(V[EM]|RM)(build|prep)/)) {
|
---|
2166 | $src = "$src $ENV{'PBDESTDIR'}/$pbpkg-$pbver$pbextdir.tar.gz $ENV{'PBDESTDIR'}/$pbpkg-$pbver$pbextdir.pbconf.tar.gz";
|
---|
2167 | my $cmd2 = "ln -sf $pbpkg-$pbver$pbextdir.tar.gz $pbpkg-latest.tar.gz";
|
---|
2168 | #my $cmd2 = "ln $pbpkg-$pbver$pbextdir.tar.gz $pbpkg-latest.tar.gz 2> /dev/null ; if [ \$? -ne 0 ]; then ln -sf $pbpkg-$pbver$pbextdir.tar.gz $pbpkg-latest.tar.gz ; fi";
|
---|
2169 | if ($cmd eq "") {
|
---|
2170 | $cmd = $cmd2;
|
---|
2171 | } else {
|
---|
2172 | $cmd = "$cmd ; $cmd2";
|
---|
2173 | }
|
---|
2174 | } elsif ($cmt eq "Web") {
|
---|
2175 | $src = "$src $ENV{'PBDESTDIR'}/$pbpkg-$pbver$pbextdir.tar.gz"
|
---|
2176 | }
|
---|
2177 |
|
---|
2178 | # Do we have one perl package
|
---|
2179 | my @nametype = pb_conf_get_if("namingtype");
|
---|
2180 | my $type = $nametype[0]->{$pbpkg};
|
---|
2181 | if ((defined $type) && ($type eq "perl")) {
|
---|
2182 | $cpanpkg = 1;
|
---|
2183 | }
|
---|
2184 | }
|
---|
2185 | # Adds conf file for availability of conf elements
|
---|
2186 | pb_conf_add("$ENV{'PBROOTDIR'}/$ENV{'PBPROJ'}.pb") if ((defined $ENV{'PBROOTDIR'}) && (-f "$ENV{'PBROOTDIR'}/$ENV{'PBPROJ'}.pb"));
|
---|
2187 | }
|
---|
2188 | my ($rbsconf,$testver,$delivery) = pb_conf_get_if($conf,"testver","delivery");
|
---|
2189 | if ($cmt eq "CPAN") {
|
---|
2190 | # Do not deliver on Pause if this is a test version
|
---|
2191 | return if (not defined $testver);
|
---|
2192 | return if ($testver =~ /true/);
|
---|
2193 | # Do not deliver on Pause if this is not a perl package
|
---|
2194 | return if ($cpanpkg == 0);
|
---|
2195 | }
|
---|
2196 |
|
---|
2197 | if ($cmt =~ /(V[EM]|RM)(build|prep)/) {
|
---|
2198 | $src="$src $ENV{'PBROOTDIR'}/$ENV{'PBPROJ'}.pb $ENV{'PBDESTDIR'}/$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}.pb $ENV{'PBETC'} $ENV{'PBDESTDIR'}/pbrc $pbscript{$v}";
|
---|
2199 | } elsif ($cmt =~ /(V[EM]|RM)Script/) {
|
---|
2200 | $src="$src $pbscript{$v}";
|
---|
2201 | } elsif ($cmt =~ /(V[EM]|RM)test/) {
|
---|
2202 | $src="$src $ENV{'PBROOTDIR'}/$ENV{'PBPROJ'}.pb $ENV{'PBDESTDIR'}/$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}.pb $ENV{'PBETC'} $ENV{'PBDESTDIR'}/pbrc $pbscript{$v} $ENV{'PBDESTDIR'}/pbtest";
|
---|
2203 | } elsif (($cmt eq "Announce") || ($cmt eq "Web") || ($cmt eq "CPAN")) {
|
---|
2204 | $src="$src $ENV{'PBTMP'}/pbscript";
|
---|
2205 | } elsif ($cmt eq "Packages") {
|
---|
2206 | # Get package list from file made during build2pkg
|
---|
2207 | open(KEEP,"$ENV{'PBBUILDDIR'}/pbgen-$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}-$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}") || die "Unable to read $ENV{'PBBUILDDIR'}/pbgen-$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}-$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}";
|
---|
2208 | $src = <KEEP>;
|
---|
2209 | chomp($src);
|
---|
2210 | close(KEEP);
|
---|
2211 | }
|
---|
2212 | if (($cmt eq "Sources") || ($cmt eq "Packages") || ($cmt eq "CPAN")) {
|
---|
2213 | my ($pbpackager) = pb_conf_get("pbpackager");
|
---|
2214 | $ENV{'PBPACKAGER'} = $pbpackager->{$ENV{'PBPROJ'}};
|
---|
2215 | pb_log(0,"Exporting public key for $ENV{'PBPACKAGER'}\n");
|
---|
2216 | # Using pb_system is not working due to redirection below
|
---|
2217 | system("gpg --export -a \'$ENV{'PBPACKAGER'}\' > $ENV{'PBDESTDIR'}/$ENV{'PBPROJ'}.pubkey");
|
---|
2218 | chmod 0644,"$ENV{'PBDESTDIR'}/$ENV{'PBPROJ'}.pubkey";
|
---|
2219 | $src = "$src $ENV{'PBDESTDIR'}/$ENV{'PBPROJ'}.pubkey";
|
---|
2220 | }
|
---|
2221 | # Remove potential leading spaces (cause problem with basename)
|
---|
2222 | $src =~ s/^ *//;
|
---|
2223 | my $basesrc = "";
|
---|
2224 | foreach my $i (split(/ +/,$src)) {
|
---|
2225 | $basesrc .= " ".basename($i);
|
---|
2226 | }
|
---|
2227 |
|
---|
2228 | pb_log(2,"values: ".Dumper(($host,$login,$dir,$port,$tmout,$path,$conf))."\n");
|
---|
2229 | my ($sshhost,$sshdir) = pb_conf_get($host,$dir);
|
---|
2230 | # Not mandatory...
|
---|
2231 | $delivery->{$ENV{'PBPROJ'}} = "" if (not defined $delivery->{$ENV{'PBPROJ'}});
|
---|
2232 | my ($sshlogin,$sshport) = pb_conf_get_if($login,$port);
|
---|
2233 | $sshport->{$ENV{PBPROJ}} = 22 if (not defined $sshport->{$ENV{PBPROJ}});
|
---|
2234 | $sshlogin->{$ENV{PBPROJ}} = getpwuid($UID) if (not defined $sshlogin->{$ENV{PBPROJ}});
|
---|
2235 | my ($vtmout,$vepath);
|
---|
2236 | # ...Except those in virtual context
|
---|
2237 | if ($cmt =~ /^VE/) {
|
---|
2238 | ($vepath) = pb_conf_get($path);
|
---|
2239 | }
|
---|
2240 | if ($cmt =~ /^(V|R)M/) {
|
---|
2241 | $vtmout = pb_distro_get_param($pbos,pb_conf_get_if($tmout));
|
---|
2242 | }
|
---|
2243 | my $remhost = $sshhost->{$ENV{'PBPROJ'}};
|
---|
2244 | my $remdir = $sshdir->{$ENV{'PBPROJ'}};
|
---|
2245 | if ($cmt =~ /^V[EM]|RM/) {
|
---|
2246 | # In that case our real host is in the xxhost with the OS as key, not project as above
|
---|
2247 | $remhost = pb_distro_get_param($pbos,$sshhost);
|
---|
2248 | }
|
---|
2249 | pb_log(2,"ssh: ".Dumper(($remhost,$sshlogin,$remdir,$sshport,$vepath,$rbsconf))."\n");
|
---|
2250 | pb_log(2,"ssh: ".Dumper($vtmout)."\n") if (defined $vtmout);
|
---|
2251 |
|
---|
2252 | my $mac;
|
---|
2253 | if ($cmt !~ /^VE/) {
|
---|
2254 | $mac = "$sshlogin->{$ENV{'PBPROJ'}}\@$remhost";
|
---|
2255 | # Overwrite account value if passed as parameter or forced
|
---|
2256 | $mac = "$ENV{'PBACCOUNT'}\@$remhost" if (defined $ENV{'PBACCOUNT'});
|
---|
2257 | } else {
|
---|
2258 | # VE
|
---|
2259 | $mac = $sshlogin->{$ENV{'PBPROJ'}};
|
---|
2260 | # Overwrite account value if passed as parameter (typically for setup2v)
|
---|
2261 | $mac = $ENV{'PBACCOUNT'} if (defined $ENV{'PBACCOUNT'});
|
---|
2262 | }
|
---|
2263 | pb_log(2, "DEBUG: mac: $mac\n");
|
---|
2264 |
|
---|
2265 | my $tdir;
|
---|
2266 | my $bdir;
|
---|
2267 | if (($cmt eq "Sources") || ($cmt =~ /(V[EM]|RM)Script/)) {
|
---|
2268 | $tdir = "$remdir/$delivery->{$ENV{'PBPROJ'}}/src";
|
---|
2269 | } elsif ($cmt eq "CPAN") {
|
---|
2270 | $tdir = "$remdir";
|
---|
2271 | } elsif ($cmt =~ /(V[EM]|RM)(build|test|prep)/) {
|
---|
2272 | $tdir = $remdir."/$ENV{'PBPROJ'}/pbdelivery";
|
---|
2273 | $bdir = $remdir."/$ENV{'PBPROJ'}/pbbuild";
|
---|
2274 | # Remove a potential $ENV{'HOME'} as bdir should be relative to pb's home
|
---|
2275 | $bdir =~ s|\$ENV.+\}/||;
|
---|
2276 | } elsif ($cmt eq "Announce") {
|
---|
2277 | $tdir = "$remdir/$delivery->{$ENV{'PBPROJ'}}";
|
---|
2278 | } elsif ($cmt eq "Web") {
|
---|
2279 | $tdir = "$remdir/$delivery->{$ENV{'PBPROJ'}}";
|
---|
2280 | } elsif ($cmt eq "Packages") {
|
---|
2281 | if (($pbos->{'type'} eq "rpm") || ($pbos->{'type'} eq "pkg") || ($pbos->{'type'} eq "hpux") || ($pbos->{'type'} eq "tgz")) {
|
---|
2282 | # put packages under an arch subdir
|
---|
2283 | $tdir = "$remdir/$delivery->{$ENV{'PBPROJ'}}/$pbos->{'name'}/$pbos->{'version'}/$pbos->{'arch'}";
|
---|
2284 | } elsif (($pbos->{'type'} eq "deb") || ($pbos->{'type'} eq "ebuild")) {
|
---|
2285 | # No need for an arch subdir
|
---|
2286 | $tdir = "$remdir/$delivery->{$ENV{'PBPROJ'}}/$pbos->{'name'}/$pbos->{'version'}";
|
---|
2287 | } else {
|
---|
2288 | die "Please teach the dev team where to deliver ($pbos->{'type'} type of packages\n";
|
---|
2289 | }
|
---|
2290 |
|
---|
2291 | my $repodir = $tdir;
|
---|
2292 | $repodir =~ s|^$remdir/||;
|
---|
2293 | my $repotag = "";
|
---|
2294 | $repotag = "-$delivery->{$ENV{'PBPROJ'}}" if ($delivery->{$ENV{'PBPROJ'}} ne "");
|
---|
2295 |
|
---|
2296 | my ($pbrepo) = pb_conf_get("pbrepo");
|
---|
2297 |
|
---|
2298 | # Script for Repository management
|
---|
2299 | $pbscript{$v} = "$ENV{'PBBUILDDIR'}/pbscript.$$";
|
---|
2300 |
|
---|
2301 | # That script should be part of the delivery
|
---|
2302 | $src = "$src $pbscript{$v}";
|
---|
2303 | open(PBS,"> $pbscript{$v}") || die "Unable to create $pbscript{$v}";
|
---|
2304 | if ($pbos->{'type'} eq "rpm") {
|
---|
2305 | my $pbsha = pb_distro_get_param($pbos,pb_conf_get("ossha"));
|
---|
2306 | my $gpgcheck = pb_conf_get_if("pbgpgcheck");
|
---|
2307 | my $pbgpgcheck;
|
---|
2308 | $pbgpgcheck = $gpgcheck->{$ENV{PBPROJ}} if (defined $gpgcheck);
|
---|
2309 | # By default force GPG check in repo even if we support signature of packages to fail. This is a best practice
|
---|
2310 | $pbgpgcheck = 1 if (not defined $pbgpgcheck);
|
---|
2311 | # Also make a pbscript to generate dnf/yum/urpmi bases
|
---|
2312 | print PBS << "EOF";
|
---|
2313 | #!/bin/bash
|
---|
2314 | # Prepare a script to ease dnf/yum setup
|
---|
2315 | EOF
|
---|
2316 | print PBS "set -x\n" if ($pbdebug gt 1);
|
---|
2317 | print PBS << "EOF";
|
---|
2318 | cat > $ENV{'PBPROJ'}$repotag.repo << EOT
|
---|
2319 | [$ENV{'PBPROJ'}$repotag]
|
---|
2320 | name=$pbos->{'name'} $pbos->{'version'} $pbos->{'arch'} - $ENV{'PBPROJ'} Vanilla Packages
|
---|
2321 | baseurl=$pbrepo->{$ENV{'PBPROJ'}}/$repodir
|
---|
2322 | enabled=1
|
---|
2323 | gpgcheck=$pbgpgcheck
|
---|
2324 | gpgkey=$pbrepo->{$ENV{'PBPROJ'}}/$repodir/$ENV{'PBPROJ'}.pubkey
|
---|
2325 | EOT
|
---|
2326 | chmod 644 $ENV{'PBPROJ'}$repotag.repo
|
---|
2327 |
|
---|
2328 | # Clean up old repo content
|
---|
2329 | rm -rf headers/ repodata/
|
---|
2330 | # Create yum repo
|
---|
2331 | if [ -x /usr/bin/yum-arch ]; then
|
---|
2332 | yum-arch .
|
---|
2333 | fi
|
---|
2334 | # Create repodata
|
---|
2335 | createrepo -s $pbsha .
|
---|
2336 | # Link to the key
|
---|
2337 | if [ -s $ENV{'PBPROJ'}.pubkey ]; then
|
---|
2338 | # Avoiding creating empty repomd.xml.key; fails on opensuse 12.1
|
---|
2339 | (cd repodata ; ln -sf ../$ENV{'PBPROJ'}.pubkey repomd.xml.key)
|
---|
2340 | fi
|
---|
2341 | # sign the repomd (at least useful for SLES - which requires a local key)
|
---|
2342 | # gpg -a --detach-sign repodata/repomd.xml
|
---|
2343 | # SLES also looks for media.1/info.txt
|
---|
2344 | EOF
|
---|
2345 | if ($pbos->{'family'} eq "md") {
|
---|
2346 | # For Mandriva add urpmi management
|
---|
2347 | print PBS << "EOF";
|
---|
2348 | # Prepare a script to ease urpmi setup
|
---|
2349 | cat > $ENV{'PBPROJ'}$repotag.addmedia << EOT
|
---|
2350 | #rpm --import $pbrepo->{$ENV{'PBPROJ'}}/$repodir/$ENV{'PBPROJ'}.pubkey
|
---|
2351 | urpmi.addmedia $ENV{'PBPROJ'}$repotag $pbrepo->{$ENV{'PBPROJ'}}/$repodir with media_info/hdlist.cz
|
---|
2352 | EOT
|
---|
2353 | chmod 755 $ENV{'PBPROJ'}$repotag.addmedia
|
---|
2354 |
|
---|
2355 | # Clean up old repo content
|
---|
2356 | rm -f hdlist.cz synthesis.hdlist.cz
|
---|
2357 | # Create urpmi repo
|
---|
2358 | genhdlist2 --clean .
|
---|
2359 | if [ \$\? -ne 0 ]; then
|
---|
2360 | genhdlist .
|
---|
2361 | fi
|
---|
2362 | (cd media_info ; ln -sf ../$ENV{'PBPROJ'}.pubkey pubkey)
|
---|
2363 | EOF
|
---|
2364 | }
|
---|
2365 | if ($pbos->{'name'} eq "fedora") {
|
---|
2366 | # Extract the spec file to please Fedora maintainers :-(
|
---|
2367 | print PBS << "EOF";
|
---|
2368 | for p in $basesrc; do
|
---|
2369 | echo \$p | grep -q 'src.rpm'
|
---|
2370 | if [ \$\? -eq 0 ]; then
|
---|
2371 | rpm2cpio \$p | cpio -ivdum --quiet '*.spec'
|
---|
2372 | fi
|
---|
2373 | done
|
---|
2374 | EOF
|
---|
2375 | }
|
---|
2376 | if ($pbos->{'family'} eq "novell") {
|
---|
2377 | # Add ymp scripts for one-click install on SuSE
|
---|
2378 | print PBS << "EOF";
|
---|
2379 | # Prepare a script to ease SuSE one-click install
|
---|
2380 | # Cf: http://de.opensuse.org/1-Klick-Installation/ISV
|
---|
2381 | #
|
---|
2382 | cat > $ENV{'PBPROJ'}$repotag.ymp << EOT
|
---|
2383 | <?xml version="1.0" encoding="utf-8"?>
|
---|
2384 | <!-- vim: set sw=2 ts=2 ai et: -->
|
---|
2385 | <metapackage xmlns:os="http://opensuse.org/Standards/One_Click_Install" xmlns="http://opensuse.org/Standards/One_Click_Install">
|
---|
2386 | <group><!-- The group of software, typically one for project-builder.org -->
|
---|
2387 | <name>$ENV{'PBPROJ'} Bundle</name> <!-- Name of the software group -->
|
---|
2388 | <summary>Software bundle for the $ENV{'PBPROJ'}$repotag project</summary> <!--This message is shown to the user and should describe the whole bundle -->
|
---|
2389 | <description>This is the summary of the $ENV{'PBPROJ'}$repotag Project
|
---|
2390 |
|
---|
2391 | Details are available on a per package basis below
|
---|
2392 |
|
---|
2393 | </description><!--This is also shown to the user -->
|
---|
2394 | <remainSubscribed>false</remainSubscribed> <!-- Don't know what it mean -->
|
---|
2395 | <repositories><!-- List of needed repositories -->
|
---|
2396 | <repository>
|
---|
2397 | <name>$ENV{'PBPROJ'}$repotag Repository</name> <!-- Name of the repository -->
|
---|
2398 | <summary>This repository contains the $ENV{'PBPROJ'}$repotag project packages.</summary> <!-- Summary of the repository -->
|
---|
2399 | <description>This repository contains the $ENV{'PBPROJ'}$repotag project packages.</description><!-- This description is shown to the user -->
|
---|
2400 | <url>$pbrepo->{$ENV{'PBPROJ'}}/$repodir</url><!--URL of repository, which is added -->
|
---|
2401 | </repository>
|
---|
2402 | </repositories>
|
---|
2403 | <software><!-- A List of packages, which should be added through the one-click-installation -->
|
---|
2404 | EOT
|
---|
2405 | for p in $basesrc; do
|
---|
2406 | sum=`rpm -q --qf '%{SUMMARY}' \$p`
|
---|
2407 | name=`rpm -q --qf '%{NAME}' \$p`
|
---|
2408 | desc=`rpm -q --qf '%{description}' \$p`
|
---|
2409 | cat >> $ENV{'PBPROJ'}$repotag.ymp << EOT
|
---|
2410 | <item>
|
---|
2411 | <name>\$name</name><!-- Name of the package, is shown to the user and used to identify the package at the repository -->
|
---|
2412 | <summary>\$sum</summary> <!-- Summary of the package -->
|
---|
2413 | <description>\$desc</description> <!-- Description, is shown to the user -->
|
---|
2414 | </item>
|
---|
2415 | EOT
|
---|
2416 | done
|
---|
2417 | cat >> $ENV{'PBPROJ'}$repotag.ymp << EOT
|
---|
2418 | </software>
|
---|
2419 | </group>
|
---|
2420 | </metapackage>
|
---|
2421 | EOT
|
---|
2422 | chmod 644 $ENV{'PBPROJ'}$repotag.ymp
|
---|
2423 | EOF
|
---|
2424 | }
|
---|
2425 | } elsif ($pbos->{'type'} eq "deb") {
|
---|
2426 | # Also make a pbscript to generate apt bases
|
---|
2427 | # Cf: http://www.debian.org/doc/manuals/repository-howto/repository-howto.fr.html
|
---|
2428 | # This dirname removes ver
|
---|
2429 | my $debarch = pb_get_debarch($pbos);
|
---|
2430 | my $projcomponent = pb_get_debpc($pbos);
|
---|
2431 | my $rpd = dirname("$pbrepo->{$ENV{'PBPROJ'}}/$repodir");
|
---|
2432 | # Remove extra . in path to fix #522
|
---|
2433 | $rpd =~ s|/./|/|g;
|
---|
2434 | print PBS << "EOF";
|
---|
2435 | #!/bin/bash
|
---|
2436 | # Prepare a script to ease apt setup
|
---|
2437 | cat > $ENV{'PBPROJ'}$repotag.sources.list << EOT
|
---|
2438 | deb $rpd $pbos->{'version'} $projcomponent
|
---|
2439 | deb-src $rpd $pbos->{'version'} $projcomponent
|
---|
2440 | EOT
|
---|
2441 | chmod 644 $ENV{'PBPROJ'}$repotag.sources.list
|
---|
2442 |
|
---|
2443 | # Up two levels to deal with the dist dir cross versions
|
---|
2444 | cd ..
|
---|
2445 | mkdir -p dists/$pbos->{'version'}/$projcomponent/binary-$debarch dists/$pbos->{'version'}/$projcomponent/source
|
---|
2446 |
|
---|
2447 | # Prepare a script to create apt info file
|
---|
2448 | # Reuse twice after
|
---|
2449 | TMPD=`mktemp -d /tmp/pb.XXXXXXXXXX` || exit 1
|
---|
2450 | mkdir -p \$TMPD
|
---|
2451 | cat > \$TMPD/Release << EOT
|
---|
2452 | Archive: unstable
|
---|
2453 | Component: $projcomponent
|
---|
2454 | Origin: $ENV{'PBPROJ'}$repotag
|
---|
2455 | Label: $ENV{'PBPROJ'}$repotag dev repository $pbrepo->{$ENV{'PBPROJ'}}
|
---|
2456 | EOT
|
---|
2457 |
|
---|
2458 | echo "Creating Packages metadata"
|
---|
2459 | for i in dists/$pbos->{version}/$projcomponent/binary-*; do
|
---|
2460 | arch=`basename \$i | sed 's/binary-//'`
|
---|
2461 | echo "Packages for \$arch:"
|
---|
2462 | dpkg-scanpackages -a\$arch $pbos->{'version'} /dev/null > dists/$pbos->{'version'}/$projcomponent/binary-\$arch/Packages
|
---|
2463 | gzip -9 < dists/$pbos->{'version'}/$projcomponent/binary-\$arch/Packages > dists/$pbos->{'version'}/$projcomponent/binary-\$arch/Packages.gz
|
---|
2464 | bzip2 -9 < dists/$pbos->{'version'}/$projcomponent/binary-\$arch/Packages > dists/$pbos->{'version'}/$projcomponent/binary-\$arch/Packages.bz2
|
---|
2465 | done
|
---|
2466 | echo "Creating Contents metadata"
|
---|
2467 | apt-ftparchive contents $pbos->{'version'} | gzip -c9 > dists/$pbos->{'version'}/Contents.gz
|
---|
2468 | echo "Creating Release metadata ($pbos->{'arch'} aka $debarch)"
|
---|
2469 | cat \$TMPD/Release > dists/$pbos->{'version'}/$projcomponent/binary-$debarch/Release
|
---|
2470 | echo "Architecture: $debarch" >> dists/$pbos->{'version'}/$projcomponent/binary-$debarch/Release
|
---|
2471 | echo "Creating Source metadata"
|
---|
2472 | dpkg-scansources $pbos->{'version'} /dev/null > dists/$pbos->{'version'}/$projcomponent/source/Sources
|
---|
2473 | gzip -9 < dists/$pbos->{'version'}/$projcomponent/source/Sources > dists/$pbos->{'version'}/$projcomponent/source/Sources.gz
|
---|
2474 | bzip2 -9 < dists/$pbos->{'version'}/$projcomponent/source/Sources > dists/$pbos->{'version'}/$projcomponent/source/Sources.bz2
|
---|
2475 | cat \$TMPD/Release > dists/$pbos->{'version'}/$projcomponent/source/Release
|
---|
2476 | echo "Architecture: Source" >> dists/$pbos->{'version'}/$projcomponent/source/Release
|
---|
2477 | echo "Creating Release metadata"
|
---|
2478 | apt-ftparchive -o APT::FTPArchive::Release::Suite=$pbos->{version} release dists/$pbos->{'version'} > dists/$pbos->{'version'}/Release
|
---|
2479 | rm -rf \$TMPD
|
---|
2480 | EOF
|
---|
2481 | } elsif ($pbos->{'type'} eq "ebuild") {
|
---|
2482 | # make a pbscript to generate links to latest version
|
---|
2483 | print PBS << "EOF";
|
---|
2484 | #!/bin/bash
|
---|
2485 | # Prepare a script to create correct links
|
---|
2486 | cd $tdir
|
---|
2487 | for p in $src; do
|
---|
2488 | echo \$p | grep -q '.ebuild'
|
---|
2489 | if [ \$\? -eq 0 ]; then
|
---|
2490 | j=`basename \$p`
|
---|
2491 | pp=`echo \$j | perl -pe 's/-[0-9\.]+-r[0-9].*\.ebuild/.ebuild/'`
|
---|
2492 | ln -sf \$j \$pp
|
---|
2493 | fi
|
---|
2494 | done
|
---|
2495 | EOF
|
---|
2496 | }
|
---|
2497 | close(PBS);
|
---|
2498 | chmod 0755,$pbscript{$v};
|
---|
2499 | } else {
|
---|
2500 | return;
|
---|
2501 | }
|
---|
2502 |
|
---|
2503 | # Useless for VE
|
---|
2504 | my $nport = pb_get_port($sshport,$pbos,$cmt) if ($cmt !~ /^VE/);
|
---|
2505 |
|
---|
2506 | # Remove a potential $ENV{'HOME'} as tdir should be relative to pb's home
|
---|
2507 | if ($cmt =~ /^VE/o) {
|
---|
2508 | $tdir =~ s|\$ENV.+\}/|| or confess "for $cmt: $tdir must have \$ENV{'HOME'} in it or the remainder of the code won't work since some parts use relative pathing and others will try absolute. Use a symlink if necessary to place the real files in a different place than under your home directory";
|
---|
2509 | } else {
|
---|
2510 | # process, ok if not present.
|
---|
2511 | $tdir =~ s|\$ENV.+\}/||;
|
---|
2512 | }
|
---|
2513 |
|
---|
2514 | my $tm = "";
|
---|
2515 | if ($cmt =~ /^(V|R)M/) {
|
---|
2516 | $tm = "sleep $vtmout" if (defined $vtmout);
|
---|
2517 | }
|
---|
2518 |
|
---|
2519 | # ssh communication if not VE or CPAN
|
---|
2520 | # should use a hash instead...
|
---|
2521 | my ($shcmd,$shcmdroot,$cpcmd,$cptarget,$cp2target);
|
---|
2522 | my $tpdir;
|
---|
2523 | my $tp;
|
---|
2524 | my $context = "$ENV{'PBTMP'}";
|
---|
2525 | my %tag;
|
---|
2526 | my $dkaccount = "";
|
---|
2527 | my $cmd1 = "";
|
---|
2528 | my $cidfile = "$context/ctn-$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}.cid";
|
---|
2529 | my $homedir = "";
|
---|
2530 | if ($cmt =~ /^VE/) {
|
---|
2531 | $tp = pb_path_expand($vepath->{$ENV{'PBPROJ'}});
|
---|
2532 | $vetype = pb_ve_get_type($vetype);
|
---|
2533 | if ($vetype eq "docker") {
|
---|
2534 | $tpdir = "/";
|
---|
2535 | } else {
|
---|
2536 | $tpdir = pb_path_expand("$tp/$pbos->{'name'}/$pbos->{'version'}/$pbos->{'arch'}");
|
---|
2537 | }
|
---|
2538 | my $arch = pb_get_arch();
|
---|
2539 | if ($vetype eq "chroot") {
|
---|
2540 | $shcmdroot = "sudo /usr/sbin/chroot $tpdir ";
|
---|
2541 | $shcmd = "$shcmdroot /bin/su - $mac -c ";
|
---|
2542 | } elsif ($vetype eq "schroot") {
|
---|
2543 | $shcmd = "schroot $tp -u $mac -- ";
|
---|
2544 | } elsif ($vetype eq "docker") {
|
---|
2545 | # docker manages the storage so rely on it
|
---|
2546 | $shcmdroot = "";
|
---|
2547 | my $docrepo = pb_ve_docker_repo();
|
---|
2548 | $cmd1 = pb_check_req("docker",0);
|
---|
2549 | #my ($dockeropt) = pb_conf_get_if("dockeropt");
|
---|
2550 | # pbimage is used for docker and is setup to the right name depending on the step:
|
---|
2551 | # TODO: This is not coded yet
|
---|
2552 | my $pbimage = undef;
|
---|
2553 | # step 0 : nothing at creation -> tag n-v-a (made in VE.pm)
|
---|
2554 | # step 1 : n-v-a + setup -> tag n-v-a-pb
|
---|
2555 | # step 2 : n-v-a-pb + build -> tag n-v-a-pb-pbproj
|
---|
2556 | # step 3 : n-v-a-pb-pbproj at use
|
---|
2557 | if ((not defined $pbimage) || ($pbimage eq "")) {
|
---|
2558 | # If no image name given, create a naming convention
|
---|
2559 | $tag{1} = "$docrepo$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}";
|
---|
2560 | $tag{2} = "$tag{1}-pb";
|
---|
2561 | $tag{3} = "$tag{2}-$ENV{'PBPROJ'}";
|
---|
2562 | } else {
|
---|
2563 | # If we were given an image name, just use it
|
---|
2564 | $tag{1} = $pbimage;
|
---|
2565 | $tag{2} = $pbimage;
|
---|
2566 | $tag{3} = $pbimage;
|
---|
2567 | }
|
---|
2568 | # If we do not create the image, then use the one we should have
|
---|
2569 | # TODO: we don't do anything with found !!
|
---|
2570 | my $found = pb_ve_docker_get_image($tag{$pbstep});
|
---|
2571 | confess "No image $tag{$pbstep} available in Docker, please create one first\n" if (not defined $found);
|
---|
2572 | # Now we use that image to do what is needed
|
---|
2573 | # use a dockerfile to ease the creation of next images
|
---|
2574 | # the context is in an empty temp dir
|
---|
2575 | my $tmpd = "$context/Dockerfile";
|
---|
2576 | open(DOCKER, "> $tmpd") || die "Unable to create the docker file $tmpd";
|
---|
2577 | print DOCKER "FROM $tag{$pbstep}\n";
|
---|
2578 | print DOCKER "MAINTAINER project-builder.org aka pb\n";
|
---|
2579 | if ($pbstep == 1) {
|
---|
2580 | # setup done as root
|
---|
2581 | $dkaccount = "root";
|
---|
2582 | } else {
|
---|
2583 | # Rest done as pb
|
---|
2584 | $dkaccount = "pb";
|
---|
2585 | }
|
---|
2586 | print DOCKER "USER $dkaccount\n";
|
---|
2587 | if ($pbstep <= 2) {
|
---|
2588 | $shcmd = "$cmd1 build -t $tag{$pbstep+1} $context";
|
---|
2589 | } else {
|
---|
2590 | # As we are in run phase use docker run. cmd will be completed below
|
---|
2591 | $shcmd = "$cmd1 run --cidfile=\"$cidfile\" --user $dkaccount --name $pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}-$cmt";
|
---|
2592 | }
|
---|
2593 | #$shcmd = "$cmd1 build $dockeropt->{$ENV{'PBPROJ'}} -t $tag{$pbstep+1} $context";
|
---|
2594 | }
|
---|
2595 | $shcmd = "setarch i386 $shcmd" if (($pbos->{'arch'} =~ /i[3456]86/) && ($arch eq 'x86_64'));
|
---|
2596 | if (($cmt =~ /VE/) && ($vetype ne "docker")) {
|
---|
2597 | # We need to get the home dir of the target account to deliver in the right place
|
---|
2598 | open(PASS,"$tpdir/etc/passwd") || die "Unable to open $tpdir/etc/passwd: $!";
|
---|
2599 | while (<PASS>) {
|
---|
2600 | my ($c1,$c2,$c3,$c4,$c5,$c6,$c7) = split(/:/);
|
---|
2601 | $homedir = $c6 if ($c1 =~ /^$mac$/);
|
---|
2602 | pb_log(3,"Homedir: $homedir - account: $c6\n");
|
---|
2603 | }
|
---|
2604 | close(PASS);
|
---|
2605 | $cpcmd = "sudo /bin/cp -r ";
|
---|
2606 | }
|
---|
2607 | if (($cmt =~ /VE/) && ($vetype eq "docker")) {
|
---|
2608 | if ($pbstep >= 2) {
|
---|
2609 | # We need to get the home dir of the target account to deliver in the right place
|
---|
2610 | $homedir = `$cmd1 run --cidfile="$cidfile" $tag{$pbstep} grep -E '^$dkaccount:' /etc/passwd | cut -d: -f6`;
|
---|
2611 | chomp($homedir);
|
---|
2612 | open(CID,"$cidfile") || confess "Unable to open $cidfile";
|
---|
2613 | my $cid = <CID>;
|
---|
2614 | close(CID);
|
---|
2615 | pb_system("$cmd1 rm $cid","","quiet");
|
---|
2616 | unlink("$cidfile");
|
---|
2617 | } else {
|
---|
2618 | $homedir = "/";
|
---|
2619 | }
|
---|
2620 | $cpcmd = "cp -r ";
|
---|
2621 | }
|
---|
2622 | $cptarget = "$tpdir/$homedir/$tdir";
|
---|
2623 | if ($cmt =~ /VE(build|prep)/) {
|
---|
2624 | $cp2target = "$tpdir/$homedir/$bdir";
|
---|
2625 | }
|
---|
2626 | pb_log(2,"On VE using $cptarget as target dir to copy to\n");
|
---|
2627 | } elsif ($cmt eq "CPAN") {
|
---|
2628 | my $ftpput = pb_check_req("ncftpput",1);
|
---|
2629 | my $ftpget = pb_check_req("wget",1);
|
---|
2630 | my ($cpanuser,$cpanpasswd) = pb_conf_get_if("cpanuser","cpanpasswd");
|
---|
2631 | return if ((not defined $cpanuser) || (not defined $cpanuser->{$ENV{'PBPROJ'}}));
|
---|
2632 | return if ((not defined $cpanpasswd) || (not defined $cpanpasswd->{$ENV{'PBPROJ'}}));
|
---|
2633 | my ($cpansubdir) = pb_conf_get_if("cpansubdir");
|
---|
2634 | $shcmd = "$ftpget --post-data \'HIDDENNAME=".$cpanuser->{$ENV{'PBPROJ'}};
|
---|
2635 | $shcmd .= "&user=".$cpanuser->{$ENV{'PBPROJ'}};
|
---|
2636 | $shcmd .= "&password=".$cpanpasswd->{$ENV{'PBPROJ'}};
|
---|
2637 | $shcmd .= "&SUBMIT_pause99_add_uri_upload=\"Upload the checked files\"";
|
---|
2638 | $shcmd .= "&pause99_add_uri_subdirtext=".$cpansubdir->{$ENV{'PBPROJ'}} if ((defined $cpansubdir) && (defined $cpansubdir->{$ENV{'PBPROJ'}}));
|
---|
2639 | foreach my $s (split(/ /,$src)) {
|
---|
2640 | $shcmd .= "&pause99_add_uri_upload=".basename($s);
|
---|
2641 | }
|
---|
2642 | $shcmd .= "'";
|
---|
2643 | $cpcmd = "$ftpput $host $dir";
|
---|
2644 | $cptarget = "CPAN";
|
---|
2645 | } else {
|
---|
2646 | my $keyopt = "";
|
---|
2647 | if ($cmt !~ /Packages/) {
|
---|
2648 | my $keyfile = pb_ssh_get(0,$remhost,$nport);
|
---|
2649 | $keyopt = "-i $keyfile" if (defined $keyfile);
|
---|
2650 | }
|
---|
2651 | my $sshcmd = pb_check_req("ssh",1);
|
---|
2652 | my $scpcmd = pb_check_req("scp",1);
|
---|
2653 | $shcmd = "$sshcmd $keyopt -q -o NoHostAuthenticationForLocalhost=yes -p $nport $mac";
|
---|
2654 | $cpcmd = "$scpcmd $keyopt -p -o NoHostAuthenticationForLocalhost=yes -P $nport";
|
---|
2655 | $cptarget = "$mac:$tdir";
|
---|
2656 | if ($cmt =~ /^(V|R)M(build|prep)/) {
|
---|
2657 | $cp2target = "$mac:$bdir";
|
---|
2658 | }
|
---|
2659 | }
|
---|
2660 |
|
---|
2661 | my $logres = "";
|
---|
2662 | # Do not touch when just announcing
|
---|
2663 | if (($cmt ne "Announce") && ($cmt ne "CPAN")) {
|
---|
2664 | # For docker everything is done in the Dockerfile for steps < 3
|
---|
2665 | if (($cmt =~ /^VE/) && ($vetype eq "docker") && ($pbstep < 3)) {
|
---|
2666 | print DOCKER "RUN mkdir -p $cptarget\n";
|
---|
2667 | print DOCKER "RUN cd $cptarget ; for i in $basesrc; do if [ -f \$i ]; then rm -f \$i; fi; done\n";
|
---|
2668 | } elsif (($cmt =~ /^VE/) && ($vetype eq "docker") && ($pbstep >= 3)) {
|
---|
2669 | # Nothing to do here
|
---|
2670 | } else {
|
---|
2671 | pb_system("$shcmd \"mkdir -p $tdir ; cd $tdir ; echo \'for i in $basesrc; do if [ -f \$i ]; then rm -f \$i; fi; done ; $cmd\' | bash -e\"","Preparing $tdir on $cptarget");
|
---|
2672 | }
|
---|
2673 | } else {
|
---|
2674 | $logres = "> ";
|
---|
2675 | }
|
---|
2676 |
|
---|
2677 | if ($cmt =~ /SandBox/) {
|
---|
2678 | # Install from sandbox mean using the result of the just passed sbx2build command
|
---|
2679 | # Get content saved in cms2build
|
---|
2680 | my ($pkg) = pb_conf_read("$ENV{'PBDESTDIR'}/$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}.pb","pbpkg");
|
---|
2681 | my $pbextdir = pb_get_extdir();
|
---|
2682 | confess "Unable to get package list" if (not defined $pkg);
|
---|
2683 |
|
---|
2684 | # We consider 2 specific packages
|
---|
2685 | my $vertag1 = $pkg->{"ProjectBuilder"};
|
---|
2686 | my $vertag2 = $pkg->{"project-builder"};
|
---|
2687 | # get the version of the current package - maybe different
|
---|
2688 | pb_log(2,"Vertag1: $vertag1\n");
|
---|
2689 | pb_log(2,"Vertag2: $vertag2\n");
|
---|
2690 | my ($pbver1,$tmp1) = split(/-/,$vertag1);
|
---|
2691 | my ($pbver2,$tmp2) = split(/-/,$vertag2);
|
---|
2692 | if ($cmt =~ /VE/) {
|
---|
2693 | # Copy inside the VE
|
---|
2694 | if ($vetype ne "docker") {
|
---|
2695 | copy("$ENV{'PBDESTDIR'}/ProjectBuilder-$pbver1$pbextdir.tar.gz","$tpdir/tmp");
|
---|
2696 | copy("$ENV{'PBDESTDIR'}/project-builder-$pbver2$pbextdir.tar.gz","$tpdir/tmp");
|
---|
2697 | } else {
|
---|
2698 | copy("$ENV{'PBDESTDIR'}/ProjectBuilder-$pbver1$pbextdir.tar.gz","$context");
|
---|
2699 | copy("$ENV{'PBDESTDIR'}/project-builder-$pbver2$pbextdir.tar.gz","$context");
|
---|
2700 | print DOCKER "COPY ProjectBuilder-$pbver1$pbextdir.tar.gz /tmp/\n";
|
---|
2701 | print DOCKER "COPY project-builder-$pbver2$pbextdir.tar.gz /tmp/\n";
|
---|
2702 | }
|
---|
2703 | } else {
|
---|
2704 | pb_system("$cpcmd $ENV{'PBDESTDIR'}/ProjectBuilder-$pbver1$pbextdir.tar.gz $ENV{'PBDESTDIR'}/project-builder-$pbver2$pbextdir.tar.gz $mac:/tmp","Copying local project files to VM/RM");
|
---|
2705 | }
|
---|
2706 | }
|
---|
2707 |
|
---|
2708 | pb_log(0,"Sources handled ($cmt): $src\n");
|
---|
2709 | # For docker everything is done in the Dockerfile
|
---|
2710 | if (($cmt =~ /^VE/) && ($vetype eq "docker") && ($pbstep < 3)) {
|
---|
2711 | foreach my $f (split(/ +/,$src)) {
|
---|
2712 | copy("$f","$context");
|
---|
2713 | print DOCKER "COPY ".basename($f)." $cptarget/\n";
|
---|
2714 | }
|
---|
2715 | print DOCKER "RUN cd $cptarget ; $cmd\n" if ((defined $cmd) && ($cmd ne ""));
|
---|
2716 | } elsif (($cmt =~ /^VE/) && ($vetype eq "docker") && ($pbstep >= 3)) {
|
---|
2717 | # Nothing to do
|
---|
2718 | } else {
|
---|
2719 | pb_system("cd $ENV{'PBBUILDDIR'} ; $cpcmd $src $cptarget 2> /dev/null","$cmt delivery in $cptarget");
|
---|
2720 | }
|
---|
2721 |
|
---|
2722 | # For VE we need to change the owner manually
|
---|
2723 | if ($cmt =~ /^VE/) {
|
---|
2724 | if ($vetype ne "docker") {
|
---|
2725 | my $sudomode = pb_distro_get_param($pbos,pb_conf_get("ossudoersmode"));
|
---|
2726 | my $res = pb_system("$shcmdroot sed -i '/requiretty/d' /etc/sudoers","Removing potential requiretty in sudoers","quiet");
|
---|
2727 | pb_system("$shcmdroot sed '/requiretty/d' /etc/sudoers > /tmp/sudoers.new ; mv /tmp/sudoers.new $tpdir/tmp/sudoers.new ; $shcmdroot mv /tmp/sudoers.new /etc/sudoers ; $shcmdroot chown root:root /etc/sudoers ; $shcmdroot chmod $sudomode /etc/sudoers","Removing again potential requiretty in sudoers as sed -i failed") if (($res ne 0) && (-f "$tpdir/etc/sudoers"));
|
---|
2728 | pb_system("$shcmd \"sudo chown -R $mac $tdir\"","Adapt owner in $tdir to $mac");
|
---|
2729 | } else {
|
---|
2730 | print DOCKER "RUN sed -i '/requiretty/d' /etc/sudoers\n" if ($pbstep == 1);
|
---|
2731 | }
|
---|
2732 | }
|
---|
2733 |
|
---|
2734 | # Use the right script name depending on context
|
---|
2735 | my $pbscript2;
|
---|
2736 | if (($cmt =~ /^(V[EM]|RM)/) || ($cmt =~ /Packages/)){
|
---|
2737 | $pbscript2 = $pbscript{$v};
|
---|
2738 | } else {
|
---|
2739 | $pbscript2 = "pbscript";
|
---|
2740 | }
|
---|
2741 |
|
---|
2742 | # It's already ready for CPAN
|
---|
2743 | my $shcmdbase = $shcmd;
|
---|
2744 | my $dkcmd;
|
---|
2745 | if ($cmt ne "CPAN") {
|
---|
2746 | if (($cmt =~ /^VE/) && ($vetype eq "docker") && ($pbstep < 3)) {
|
---|
2747 | copy("$pbscript2","$context");
|
---|
2748 | my $s = basename($pbscript2);
|
---|
2749 | print DOCKER "COPY $s $cptarget/\n";
|
---|
2750 | # call directly with perl to avoid text file busy error messages
|
---|
2751 | print DOCKER "RUN cd $cptarget ; perl ./$s ; if [ '$pbkeep' = '0' ]; then rm -f $s; fi\n";
|
---|
2752 | print DOCKER "RUN rm -rf $cptarget/*\n";
|
---|
2753 | print DOCKER "RUN chown -R $dkaccount $cptarget\n" if ($pbstep == 1);
|
---|
2754 | close(DOCKER);
|
---|
2755 | } elsif (($cmt =~ /^VE/) && ($vetype eq "docker") && ($pbstep >= 3)) {
|
---|
2756 | $dkcmd = "/bin/bash $homedir/$tdir-2/".basename($pbscript2);
|
---|
2757 | } else {
|
---|
2758 | my $s = basename($pbscript2);
|
---|
2759 | $shcmd .= " \"echo \'cd $tdir ; if [ -x $s ]; then ./$s; fi ; if [ $pbkeep == 0 ]; then rm -f $s; fi\' | bash\"";
|
---|
2760 | }
|
---|
2761 | }
|
---|
2762 | my $cmdverb = "verbose";
|
---|
2763 | if (($cmt eq "Announce") || ($cmt eq "CPAN")) {
|
---|
2764 | $cmdverb = undef;
|
---|
2765 | } elsif (defined ($v)) {
|
---|
2766 | $cmdverb = "verbose_\[$v\] ";
|
---|
2767 | }
|
---|
2768 | # this is where we launch the execution
|
---|
2769 | if (($cmt =~ /^VE/) && ($vetype eq "docker") && ($pbstep >= 3)) {
|
---|
2770 | pb_mkdir_p("$ENV{'PBTMP'}/vebuild");
|
---|
2771 | my $tsrc = "";
|
---|
2772 | foreach my $f (split(/ +/,$src)) {
|
---|
2773 | copy("$f","$ENV{'PBTMP'}/vebuild");
|
---|
2774 | $tsrc .= " ".basename($f);
|
---|
2775 | }
|
---|
2776 | my $ts = "$ENV{'PBTMP'}/vebuild/".basename($pbscript2);
|
---|
2777 | open(DOCKER," > $ts") || confess "Unable to create $ts";
|
---|
2778 | print DOCKER "#!/bin/bash\n";
|
---|
2779 | print DOCKER "# Put content from host to guest, copying to have the correct uid/gid and rights to do so\n";
|
---|
2780 | print DOCKER "# set -x\n";
|
---|
2781 | print DOCKER "for i in $tsrc; do\n";
|
---|
2782 | print DOCKER " # Cleanup first to avoid rights issues for root owned files e.g.\n";
|
---|
2783 | print DOCKER " rm -f $homedir/$tdir/\$i\n";
|
---|
2784 | print DOCKER " cp $homedir/$tdir-2/\$i $homedir/$tdir\n";
|
---|
2785 | print DOCKER "done\n";
|
---|
2786 | print DOCKER "(cd $homedir/$tdir-2 ; ls -al)\n";
|
---|
2787 | print DOCKER pb_get_content($pbscript2);
|
---|
2788 | print DOCKER "\n# Get content back from guest in a host dir\n";
|
---|
2789 | # Host dir seen as TDIR in container
|
---|
2790 | $cp2target = "$ENV{'PBBUILDDIR'}/$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}-$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}";
|
---|
2791 | pb_mkdir_p("$cp2target");
|
---|
2792 | chmod 0777,"$cp2target";
|
---|
2793 | print DOCKER "TDIR=$homedir/$bdir-2/$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}-$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}\n";
|
---|
2794 | print DOCKER "for i in $homedir/$bdir/*; do\n";
|
---|
2795 | print DOCKER " # Cleanup first to avoid rights issues for root owned files e.g.\n";
|
---|
2796 | print DOCKER " j=\`basename \$i\`\n";
|
---|
2797 | print DOCKER " rm -f \$TDIR/\$j\n";
|
---|
2798 | print DOCKER " cp -r \$i \$TDIR\n";
|
---|
2799 | print DOCKER "done\n";
|
---|
2800 | print DOCKER "find \$TDIR -type f -print0 | xargs -0 chmod 644\n";
|
---|
2801 | print DOCKER "find \$TDIR -type d -print0 | xargs -0 chmod 777 2> /dev/null\n";
|
---|
2802 | #print DOCKER "ls -al \$TDIR\n";
|
---|
2803 | print DOCKER "exit 0\n";
|
---|
2804 | close(DOCKER);
|
---|
2805 | $shcmd .= " -v $ENV{'PBBUILDDIR'}:$homedir/$bdir-2 -v $ENV{'PBTMP'}/vebuild:$homedir/$tdir-2 -w $homedir/$tdir-2 $tag{$pbstep} $dkcmd";
|
---|
2806 | }
|
---|
2807 | my $ret = pb_system("$shcmd","Executing pbscript on $cptarget if needed",$cmdverb);
|
---|
2808 |
|
---|
2809 | # If target is deb family then sign Release file for Packages
|
---|
2810 | my $pbos2 = pb_distro_get_context($v);
|
---|
2811 | if (($cmt =~ /Packages/) && ($pbos2->{'type'} eq "deb")) {
|
---|
2812 | my $debarch = pb_get_debarch($pbos2);
|
---|
2813 | my $projcomponent = pb_get_debpc($pbos2);
|
---|
2814 | for my $f ("dists/$pbos2->{'version'}/$projcomponent/binary-$debarch/Release","dists/$pbos2->{'version'}/$projcomponent/source/Release","dists/$pbos2->{'version'}/Release") {
|
---|
2815 | my $debdir = dirname($cptarget);
|
---|
2816 | pb_system("cd $ENV{'PBBUILDDIR'} ; $cpcmd $debdir/$f . 2> /dev/null","Getting deb $f file from $debdir");
|
---|
2817 | pb_sign_file("$ENV{'PBBUILDDIR'}/Release");
|
---|
2818 | pb_system("$cpcmd $ENV{'PBBUILDDIR'}/Release.gpg $debdir/$f.gpg 2> /dev/null","Putting deb $f.gpg file to $debdir");
|
---|
2819 | }
|
---|
2820 | }
|
---|
2821 |
|
---|
2822 | if ($cmt =~ /^(V[EM]|RM)build/) {
|
---|
2823 | # Get back info on pkg produced, compute their name and get them from the VM/RM
|
---|
2824 | pb_system("$cpcmd $cp2target/pbgen-$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}-$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'} $ENV{'PBBUILDDIR'}/pbgen-$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}.$$ 2> /dev/null","Get package names in $cp2target");
|
---|
2825 | if (not -f "$ENV{'PBBUILDDIR'}/pbgen-$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}.$$") {
|
---|
2826 | pb_log(0,"ERROR with VM/RM $v on getting $ENV{'PBBUILDDIR'}/pbgen-$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}.$$\n");
|
---|
2827 | } else {
|
---|
2828 | open(KEEP,"$ENV{'PBBUILDDIR'}/pbgen-$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}.$$") || die "Unable to read $ENV{'PBBUILDDIR'}/pbgen-$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}.$$";
|
---|
2829 | my $src = <KEEP>;
|
---|
2830 | chomp($src);
|
---|
2831 | close(KEEP);
|
---|
2832 | unlink("$ENV{'PBBUILDDIR'}/pbgen-$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}.$$");
|
---|
2833 |
|
---|
2834 | $src =~ s/^ *//;
|
---|
2835 | pb_mkdir_p("$ENV{'PBBUILDDIR'}/$pbos->{'name'}/$pbos->{'version'}/$pbos->{'arch'}");
|
---|
2836 | # Change pgben to make the next send2target happy
|
---|
2837 | my $made = "";
|
---|
2838 |
|
---|
2839 | # For VM/RM we don't want shell expansion to hapen locally but remotely
|
---|
2840 | my $delim = '\'';
|
---|
2841 | if ($cmt =~ /^VEbuild/) {
|
---|
2842 | # For VE we need to support shell expansion locally
|
---|
2843 | $delim = "";
|
---|
2844 | }
|
---|
2845 |
|
---|
2846 | open(KEEP,"> $ENV{'PBBUILDDIR'}/pbgen-$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}-$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}") || die "Unable to write $ENV{'PBBUILDDIR'}/pbgen-$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}-$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}";
|
---|
2847 | foreach my $p (split(/ +/,$src)) {
|
---|
2848 | my $j = basename($p);
|
---|
2849 | pb_system("$cpcmd $cp2target/$delim$p$delim $ENV{'PBBUILDDIR'}/$pbos->{'name'}/$pbos->{'version'}/$pbos->{'arch'} 2> /dev/null","Recovery of package $j in $ENV{'PBBUILDDIR'}/$pbos->{'name'}/$pbos->{'version'}/$pbos->{'arch'}");
|
---|
2850 | $made="$made $pbos->{'name'}/$pbos->{'version'}/$pbos->{'arch'}/$j"; # if (($pbos->{'type'} ne "rpm") || ($j !~ /.src.rpm$/));
|
---|
2851 | }
|
---|
2852 |
|
---|
2853 | print KEEP "$made\n";
|
---|
2854 | close(KEEP);
|
---|
2855 | if (($cmt =~ /^VE/) && ($vetype eq "docker") && ($pbstep >= 3)) {
|
---|
2856 | pb_rm_rf("$cp2target");
|
---|
2857 | pb_system("$cmd1 rm $pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}-$cmt","","quiet");
|
---|
2858 | } else {
|
---|
2859 | pb_system("$shcmdbase \"rm -rf $tdir $bdir\"","$cmt cleanup");
|
---|
2860 | }
|
---|
2861 |
|
---|
2862 | # Sign packages locally
|
---|
2863 | pb_sign_pkgs($pbos,$made);
|
---|
2864 |
|
---|
2865 | # We want to send them to the ssh account so overwrite what has been done before
|
---|
2866 | pb_log(2,"Before sending pkgs, vexist: $vexist, vpid: $vpid\n");
|
---|
2867 | pb_send2target("Packages","$ENV{'PBDESTDIR'}/pbscript.$$",$pbos->{'name'}."-".$pbos->{'version'}."-".$pbos->{'arch'},$vexist,$vpid);
|
---|
2868 | pb_rm_rf("$ENV{'PBBUILDDIR'}/$pbos->{'name'}/$pbos->{'version'}/$pbos->{'arch'}");
|
---|
2869 | }
|
---|
2870 | }
|
---|
2871 | unlink("$pbscript{$v}") if ((($cmt =~ /^(V[ME]|RM)/) || ($cmt =~ /Packages/)) && ($pbkeep eq 0) && ($cmt !~ /Script/));
|
---|
2872 |
|
---|
2873 | pb_log(2,"Before halt, vexist: $vexist, vpid: $vpid\n");
|
---|
2874 | if ((! $vexist) && ($cmt =~ /^VM/)) {
|
---|
2875 | # If in setupvm then takes a snapshot just before halting
|
---|
2876 | if ($snapme != 0) {
|
---|
2877 | my ($vmmonport,$vmtype) = pb_conf_get("vmmonport","vmtype");
|
---|
2878 | # For monitoring control
|
---|
2879 | if ((($vmtype->{$ENV{'PBPROJ'}}) eq "kvm") || (($vmtype->{$ENV{'PBPROJ'}}) eq "qemu")) {
|
---|
2880 | eval
|
---|
2881 | {
|
---|
2882 | require Net::Telnet;
|
---|
2883 | Net::Telnet->import();
|
---|
2884 | };
|
---|
2885 | if ($@) {
|
---|
2886 | # Net::Telnet not found
|
---|
2887 | pb_log(1,"ADVISE: Install Net::Telnet to benefit from monitoring control and snapshot feature.\nWARNING: No snapshot created");
|
---|
2888 | } else {
|
---|
2889 | my $t = new Net::Telnet (Timeout => 120, Host => "localhost", Port => $vmmonport->{$ENV{'PBPROJ'}}) || die "Unable to dialog on the monitor";
|
---|
2890 | # move to monitor mode
|
---|
2891 | my @lines = $t->cmd("c");
|
---|
2892 | # Create a snapshot named pb
|
---|
2893 | @lines = $t->cmd("savevm pb");
|
---|
2894 | # Write the new status in the VM
|
---|
2895 | @lines = $t->cmd("commit all");
|
---|
2896 | # End
|
---|
2897 | @lines = $t->cmd("quit");
|
---|
2898 | }
|
---|
2899 | }
|
---|
2900 | }
|
---|
2901 | my $hoption = "-p";
|
---|
2902 | my $hpath = pb_distro_get_param($pbos,pb_conf_get("ospathcmd-halt"));
|
---|
2903 | # Solaris doesn't support -p of halt
|
---|
2904 | if ($pbos->{'type'} eq "pkg") {
|
---|
2905 | $hoption = "" ;
|
---|
2906 | }
|
---|
2907 | pb_system("$shcmdbase \"sudo $hpath $hoption \"; $tm ; echo \'if [ -d /proc/$vpid ]; then kill -9 $vpid; fi \' | bash ; sleep 10","VM $v halt (pid $vpid)");
|
---|
2908 | }
|
---|
2909 | if (($cmt =~ /^VE/) && ($snapme != 0)) {
|
---|
2910 | if ($vetype eq "docker") {
|
---|
2911 | foreach my $f (split(/ +/,$src)) {
|
---|
2912 | unlink("$context/$f");
|
---|
2913 | }
|
---|
2914 | } else {
|
---|
2915 | pb_ve_snap($pbos,$tp);
|
---|
2916 | }
|
---|
2917 | }
|
---|
2918 | }
|
---|
2919 |
|
---|
2920 | sub pb_script2v {
|
---|
2921 |
|
---|
2922 | my $pbscript=shift;
|
---|
2923 | my $vtype=shift;
|
---|
2924 | my $pbstep=shift; # Which step are we in (0: create, 1: setup pb, 2: prep pbproj, 3: use to build)
|
---|
2925 | my $pbforce=shift; # Force stop of VM. Default not.
|
---|
2926 | my $snapme=shift; # Do we have to create a snapshot. Default not.
|
---|
2927 | my $pbimage=shift; # Which image to use to start the VM/VE
|
---|
2928 |
|
---|
2929 | $pbforce = 0 if (not defined $pbforce);
|
---|
2930 | $snapme = 0 if (not defined $snapme);
|
---|
2931 | $pbstep = 3 if (not defined $pbstep);
|
---|
2932 |
|
---|
2933 | pb_log(2,"DEBUG: pb_script2v($vtype,$pbstep,$pbforce,$snapme)\n");
|
---|
2934 | pb_log(2,"DEBUG: pb_script2v(pbscript: ".Dumper($pbscript)."\n") if (defined $pbscript);
|
---|
2935 | pb_parallel_launchv($pbscript,$vtype,uc($vtype)."Script",$pbstep,$pbimage,$pbforce,$snapme,$pbsnap);
|
---|
2936 | }
|
---|
2937 |
|
---|
2938 | sub pb_parallel_launchv {
|
---|
2939 |
|
---|
2940 | my $pbscript=shift;
|
---|
2941 | my $vtype = shift;
|
---|
2942 | my $action = shift; # It an action is defined then use send2target
|
---|
2943 | my $pbstep=shift; # Which step are we in (0: create, 1: setup, 2: build 3: use)
|
---|
2944 | my $pbimage=shift; # Which image to use to start the VM/VE
|
---|
2945 | my $pbforce=shift; # Force stop of VM. Default not.
|
---|
2946 | my $snapme = shift; # By default do not snap a VM/VE/RM
|
---|
2947 | my $usesnap = shift; # By default study the usage of the snapshot feature of VM/VE/RM
|
---|
2948 | my $vm;
|
---|
2949 | my $all;
|
---|
2950 |
|
---|
2951 | $pbforce = 0 if (not defined $pbforce);
|
---|
2952 | $snapme = 0 if (not defined $snapme);
|
---|
2953 | $usesnap = 1 if (not defined $usesnap);
|
---|
2954 |
|
---|
2955 | # Adapt // mode to memory size
|
---|
2956 | $pbparallel = pb_set_parallel($vtype);
|
---|
2957 |
|
---|
2958 | pb_log(2,"DEBUG: pb_parallel_launch(vtype,step,force,snapme,usesnap: $vtype,$pbstep,$pbforce,$snapme,$usesnap)\n");
|
---|
2959 | pb_log(2,"DEBUG: pb_parallel_launch(pbscript: ".Dumper($pbscript)."\n");
|
---|
2960 |
|
---|
2961 | my ($vexist,$vpid) = (undef,undef);
|
---|
2962 | my $pm;
|
---|
2963 | my $all_ok = 1;
|
---|
2964 | if (defined $pbparallel) {
|
---|
2965 | $pm = new Parallel::ForkManager($pbparallel);
|
---|
2966 |
|
---|
2967 | # Set which port the VM/RM will use to communicate
|
---|
2968 | $pm->run_on_start(\&pb_set_port);
|
---|
2969 | $pm->run_on_finish(sub { my ($pid, $code, $id, $signal, $dump) = @_;
|
---|
2970 | unless ($code == 0 && $signal == 0 && $dump == 0) {
|
---|
2971 | $all_ok = 0;
|
---|
2972 | pb_log(0,"ERROR: pid $pid failed\n");
|
---|
2973 | }
|
---|
2974 | });
|
---|
2975 | }
|
---|
2976 |
|
---|
2977 | # Determine on which V to work
|
---|
2978 | if (defined $ENV{'PBV'}) {
|
---|
2979 | @$vm = split(/,/,$ENV{'PBV'});
|
---|
2980 | } else {
|
---|
2981 | ($vm,$all) = pb_get2v($vtype);
|
---|
2982 | }
|
---|
2983 |
|
---|
2984 | my $counter = 0;
|
---|
2985 | foreach my $v (@$vm) {
|
---|
2986 | $counter++;
|
---|
2987 | # Modulo 2 * pbparallel (to avoid synchronization problems)
|
---|
2988 | $counter = 1 if ((defined $pbparallel) && ($counter > 2 * $pbparallel));
|
---|
2989 | $pm->start($counter) and next if (defined $pbparallel);
|
---|
2990 |
|
---|
2991 | # If launched via script2v then get the script passed as the one for the that VM
|
---|
2992 | if (not defined $pbscript->{$v}) {
|
---|
2993 | $pbscript->{$v} = $pbscript->{'default'};
|
---|
2994 | }
|
---|
2995 | #
|
---|
2996 | # Launch a single operation
|
---|
2997 | ($vexist,$vpid) = pb_launchv($pbscript->{$v},$vtype,$v,$action,$pbstep,$pbforce,$snapme,$pbsnap,$pbimage);
|
---|
2998 |
|
---|
2999 | # Skip that VM/RM if something went wrong
|
---|
3000 | if (($vtype =~ /(v|r)m/) && ($vpid == 0) && ($vexist == 0)) {
|
---|
3001 | $pm->finish if (defined $pbparallel);
|
---|
3002 | next;
|
---|
3003 | }
|
---|
3004 |
|
---|
3005 |
|
---|
3006 | $pm->finish if (defined $pbparallel);
|
---|
3007 | }
|
---|
3008 | $pm->wait_all_children if (defined $pbparallel);
|
---|
3009 | die "Aborting, one or more of the children failed." if ((not $all_ok) && ($Global::pb_stop_on_error));
|
---|
3010 | }
|
---|
3011 |
|
---|
3012 | sub pb_launchv {
|
---|
3013 |
|
---|
3014 | my $pbscript1=shift; # Just the script name to execute, not the hash
|
---|
3015 | my $vtype = shift;
|
---|
3016 | my $v = shift; # Only 1 VM/VE/RM treated here
|
---|
3017 | my $action = shift; # It an action is defined then use send2target
|
---|
3018 | my $pbstep=shift; # Which step are we in (0: create, 1: setup, 2: build 3: use)
|
---|
3019 | my $pbforce=shift; # Force stop of VM. Default not.
|
---|
3020 | my $snapme = shift; # By default do not snap a VM/VE/RM
|
---|
3021 | my $usesnap = shift; # By default study the usage of the snapshot feature of VM/VE/RM
|
---|
3022 | my $pbimage=shift; # Which image to use to start the VM/VE
|
---|
3023 | my $vexist = undef;
|
---|
3024 | my $vpid = undef;
|
---|
3025 |
|
---|
3026 | $pbforce = 0 if (not defined $pbforce);
|
---|
3027 | $snapme = 0 if (not defined $snapme);
|
---|
3028 | $usesnap = 1 if (not defined $usesnap);
|
---|
3029 |
|
---|
3030 | # Check that v exists
|
---|
3031 | die "No VM/VE/RM defined, unable to launch" if (not defined $v);
|
---|
3032 |
|
---|
3033 | # If creation or snapshot creation mode, no snapshot usable
|
---|
3034 | if (($pbstep == 0) || ($snapme == 1)) {
|
---|
3035 | $usesnap = 0;
|
---|
3036 | }
|
---|
3037 |
|
---|
3038 | pb_log(2,"DEBUG: pb_launchv(vtype,v,step,force,snapme,usesnap: $vtype,$v,$pbstep,$pbforce,$snapme,$usesnap)\n");
|
---|
3039 | pb_log(2,"DEBUG: pb_launchv(pbimage: $pbimage)\n") if (defined $pbimage);
|
---|
3040 | pb_log(2,"DEBUG: pb_launchv(pbscript $pbscript1)\n") if (defined $pbscript1);
|
---|
3041 | pb_log(2,"DEBUG: pb_launchv(action: $action)\n") if (defined $action);
|
---|
3042 | # Keep only the first VM in case many were given
|
---|
3043 | if ($v =~ /,/) {
|
---|
3044 | pb_log(0,"WARNING: pruning to just the first of several VM/VE/RMs listed ($v)\n");
|
---|
3045 | $v =~ s/,.*//;
|
---|
3046 | }
|
---|
3047 |
|
---|
3048 | my $pbos = pb_distro_get_context($v);
|
---|
3049 |
|
---|
3050 | pb_apply_conf_proxy($pbos);
|
---|
3051 |
|
---|
3052 | # Launch the VMs/VEs
|
---|
3053 | if ($vtype eq "vm") {
|
---|
3054 | confess "-i image parameter needed" if (((not defined $pbimage) || ($pbimage eq "")) && ($pbstep == 0));
|
---|
3055 |
|
---|
3056 | my ($ptr,$ptr2,$vmpath,$vmport,$vms) = pb_conf_get("vmtype","vmcmd","vmpath","vmport","vmsize");
|
---|
3057 | my ($vmopt,$vmmm,$vmtmout,$vmsnap,$vmbuildtm,$vmmonport) = pb_conf_get_if("vmopt","vmmem","vmtmout","vmsnap","vmbuildtm","vmmonport");
|
---|
3058 | my $vmsize = pb_distro_get_param($pbos,$vms);
|
---|
3059 |
|
---|
3060 | my $vmtype = $ptr->{$ENV{'PBPROJ'}};
|
---|
3061 | my $vmcmd = $ptr2->{$ENV{'PBPROJ'}};
|
---|
3062 |
|
---|
3063 | if (defined $opts{'g'}) {
|
---|
3064 | if (($vmtype eq "kvm") || ($vmtype eq "qemu")) {
|
---|
3065 | $ENV{'PBVMOPT'} = "--nographic";
|
---|
3066 | }
|
---|
3067 | }
|
---|
3068 | if (not defined $ENV{'PBVMOPT'}) {
|
---|
3069 | $ENV{'PBVMOPT'} = "";
|
---|
3070 | }
|
---|
3071 | # Save the current status for later restoration
|
---|
3072 | $ENV{'PBOLDVMOPT'} = $ENV{'PBVMOPT'};
|
---|
3073 | # Set a default timeout of 2 minutes
|
---|
3074 | if (not defined $ENV{'PBVMTMOUT'}) {
|
---|
3075 | $ENV{'PBVMTMOUT'} = "120";
|
---|
3076 | }
|
---|
3077 | if (defined $vmopt->{$v}) {
|
---|
3078 | $ENV{'PBVMOPT'} .= " $vmopt->{$v}" if ($ENV{'PBVMOPT'} !~ / $vmopt->{$v}/);
|
---|
3079 | } elsif (defined $vmopt->{$ENV{'PBPROJ'}}) {
|
---|
3080 | $ENV{'PBVMOPT'} .= " $vmopt->{$ENV{'PBPROJ'}}" if ($ENV{'PBVMOPT'} !~ / $vmopt->{$ENV{'PBPROJ'}}/);
|
---|
3081 | }
|
---|
3082 |
|
---|
3083 | # How much memory to allocate for VMs
|
---|
3084 | if (defined $vmmm) {
|
---|
3085 | my $vmmem = pb_distro_get_param($pbos,$vmmm);
|
---|
3086 | if (defined $vmmem) {
|
---|
3087 | $ENV{'PBVMOPT'} .= " -m $vmmem";
|
---|
3088 | }
|
---|
3089 | }
|
---|
3090 |
|
---|
3091 | # Are we allowed to use snapshot feature
|
---|
3092 | if ($usesnap == 1) {
|
---|
3093 | if ((defined $vmsnap->{$v}) && ($vmsnap->{$v} =~ /true/i)) {
|
---|
3094 | $ENV{'PBVMOPT'} .= " -snapshot";
|
---|
3095 | } elsif ((defined $vmsnap->{$ENV{'PBPROJ'}}) && ($vmsnap->{$ENV{'PBPROJ'}} =~ /true/i)) {
|
---|
3096 | $ENV{'PBVMOPT'} .= " -snapshot";
|
---|
3097 | } elsif ($pbsnap eq 1) {
|
---|
3098 | $ENV{'PBVMOPT'} .= " -snapshot";
|
---|
3099 | }
|
---|
3100 | }
|
---|
3101 | if ($snapme != 0) {
|
---|
3102 | if (($vmtype eq "kvm") || ($vmtype eq "qemu")) {
|
---|
3103 | # Configure the monitoring to automate the creation of the 'pb' snapshot
|
---|
3104 | $ENV{'PBVMOPT'} .= " -serial mon:telnet::$vmmonport->{$ENV{'PBPROJ'}},server,nowait" if ((defined $vmmonport) && (defined $vmmonport->{$ENV{'PBPROJ'}}));
|
---|
3105 | # In that case no snapshot call needed
|
---|
3106 | $ENV{'PBVMOPT'} =~ s/ -snapshot//;
|
---|
3107 | }
|
---|
3108 | }
|
---|
3109 | if (defined $vmtmout->{$v}) {
|
---|
3110 | $ENV{'PBVMTMOUT'} = $vmtmout->{$v};
|
---|
3111 | } elsif (defined $vmtmout->{$ENV{'PBPROJ'}}) {
|
---|
3112 | $ENV{'PBVMTMOUT'} = $vmtmout->{$ENV{'PBPROJ'}};
|
---|
3113 | }
|
---|
3114 | my $nport = pb_get_port($vmport,$pbos,$vtype);
|
---|
3115 |
|
---|
3116 | my $cmd;
|
---|
3117 | my $vmm; # has to be used for pb_check_ps
|
---|
3118 | if (($vmtype eq "qemu") || ($vmtype eq "kvm")) {
|
---|
3119 | $vmm = "$vmpath->{$ENV{'PBPROJ'}}/$v.qemu";
|
---|
3120 | if (($pbstep == 0) || (defined $pbimage)) {
|
---|
3121 | $ENV{'PBVMOPT'} .= " -cdrom $pbimage -boot d";
|
---|
3122 | }
|
---|
3123 | # Always redirect the network and always try to use a 'pb' snapshot
|
---|
3124 | #$cmd = "$vmcmd $ENV{'PBVMOPT'} -net nic,model=virtio -net user,hostfwd=tcp::$nport:22 -loadvm pb $vmm"
|
---|
3125 | #$cmd = "$vmcmd $ENV{'PBVMOPT'} -redir tcp:$nport:10.0.2.15:22 $vmm"
|
---|
3126 | $cmd = "$vmcmd $ENV{'PBVMOPT'} -net nic -net user,hostfwd=tcp:127.0.0.1:$nport-:22 $vmm"
|
---|
3127 | } elsif ($vmtype eq "xen") {
|
---|
3128 | } elsif ($vmtype eq "vmware") {
|
---|
3129 | } else {
|
---|
3130 | die "VM of type $vmtype not supported. Report to the dev team";
|
---|
3131 | }
|
---|
3132 | # Restore the ENV VAR Value
|
---|
3133 | $ENV{'PBVMOPT'} = $ENV{'PBOLDVMOPT'};
|
---|
3134 |
|
---|
3135 | my ($tmpcmd,$void) = split(/ +/,$cmd);
|
---|
3136 | my $vmmport = undef;
|
---|
3137 | ($vexist,$vmmport) = pb_check_ps($tmpcmd,$vmm);
|
---|
3138 | $vpid = 0;
|
---|
3139 |
|
---|
3140 | if (! $vexist) {
|
---|
3141 | if ($pbstep == 0) {
|
---|
3142 | die("Found an existing Virtual machine $vmm. Won't overwrite") if (-r $vmm);
|
---|
3143 | if (($vmtype eq "qemu") || ($vmtype eq "xen") || ($vmtype eq "kvm")) {
|
---|
3144 | my $command = pb_check_req("qemu-img",0);
|
---|
3145 | pb_system("$command create -f qcow2 $vmm $vmsize","Creating the QEMU VM");
|
---|
3146 | } elsif ($vmtype eq "vmware") {
|
---|
3147 | } else {
|
---|
3148 | }
|
---|
3149 | }
|
---|
3150 | if (! -f "$vmm") {
|
---|
3151 | pb_log(0,"Unable to find VM $vmm\n");
|
---|
3152 | } else {
|
---|
3153 | # Is the SSH port free? if not kill the existing process using it after a build timeout period
|
---|
3154 | my ($vmssh,$void) = pb_check_ps($tmpcmd,$vmm);
|
---|
3155 | if ($vmssh) {
|
---|
3156 | my $buildtm = $ENV{'PBVMTMOUT'};
|
---|
3157 | if (defined $vmbuildtm->{$v}) {
|
---|
3158 | $buildtm = $vmbuildtm->{$v};
|
---|
3159 | } elsif (defined $vmbuildtm->{$ENV{'PBPROJ'}}) {
|
---|
3160 | $buildtm = $vmbuildtm->{$ENV{'PBPROJ'}};
|
---|
3161 | }
|
---|
3162 |
|
---|
3163 | sleep $buildtm;
|
---|
3164 | pb_log(0,"WARNING: Killing the process ($vmssh) using port $void (previous failed VM ?)\n");
|
---|
3165 | kill 15,$vmssh;
|
---|
3166 | # Let it time to exit
|
---|
3167 | sleep 5;
|
---|
3168 | }
|
---|
3169 | pb_system("$cmd &","Launching the VM $vmm");
|
---|
3170 | # Using system allows to kill it externaly if needed,sosupport that in the call
|
---|
3171 | pb_system("sleep $ENV{'PBVMTMOUT'}","Waiting $ENV{'PBVMTMOUT'} s for VM $v to come up","mayfail");
|
---|
3172 | ($vpid,$void) = pb_check_ps($tmpcmd,$vmm);
|
---|
3173 | pb_log(0,"VM $vmm launched (pid $vpid)\n");
|
---|
3174 | }
|
---|
3175 | } else {
|
---|
3176 | pb_log(0,"Found an existing VM $vmm (pid $vexist - Port $vmmport)\n");
|
---|
3177 | # Set the correct port here based on what is done
|
---|
3178 | $pbport = $vmmport;
|
---|
3179 | $vpid = $vexist;
|
---|
3180 | }
|
---|
3181 | pb_log(2,"DEBUG: pb_launchv returns ($vexist,$vpid)\n");
|
---|
3182 |
|
---|
3183 | # After the VM is launched, then setup the SSH access if not in creation phase
|
---|
3184 | if ($pbstep != 0) {
|
---|
3185 | if (defined $pbport) {
|
---|
3186 | pb_ssh_setup($vtype,$pbport);
|
---|
3187 | } else {
|
---|
3188 | pb_ssh_setup($vtype,$nport);
|
---|
3189 | }
|
---|
3190 | }
|
---|
3191 |
|
---|
3192 | # Skip that VM if something went wrong
|
---|
3193 | return($vexist,$vpid) if (($vpid == 0) && ($vexist == 0));
|
---|
3194 |
|
---|
3195 | # If force stopping the VM then reset vexist
|
---|
3196 | if ($pbforce == 1) {
|
---|
3197 | $vpid = $vexist;
|
---|
3198 | $vexist = 0;
|
---|
3199 | }
|
---|
3200 | } elsif ($vtype eq "ve") {
|
---|
3201 | # Force the creation of the VE and no snapshot usable
|
---|
3202 | $vetype = pb_ve_get_type($vetype);
|
---|
3203 | pb_ve_launch($v,$pbstep,$usesnap,$vetype,$pbimage);
|
---|
3204 | $vexist = 0;
|
---|
3205 | $vpid = 0;
|
---|
3206 | } else {
|
---|
3207 | # Get RM context
|
---|
3208 | my ($ptr,$rmpath,$rmport) = pb_conf_get("rmtype","rmpath","rmport");
|
---|
3209 |
|
---|
3210 | # Nothing more to do for RM. No real launch
|
---|
3211 | # For the moment we support the RM is already running
|
---|
3212 | # For ProLiant may be able to power them on if needed later on as an example.
|
---|
3213 | $vexist = 0;
|
---|
3214 | $vpid = 0;
|
---|
3215 |
|
---|
3216 | # After the RM is launched, then setup the SSH access
|
---|
3217 | my $nport = pb_get_port($rmport,$pbos,$vtype);
|
---|
3218 | pb_ssh_setup($vtype,$nport);
|
---|
3219 | }
|
---|
3220 |
|
---|
3221 | # Gather all required files to send them to the VM/VE/RM
|
---|
3222 | # and launch the build through pbscript
|
---|
3223 | if (defined $action) {
|
---|
3224 | pb_log(2,"DEBUG: Before send2target, vexist: $vexist, vpid: $vpid, vtype: $vtype, v: $v, action: $action\n");
|
---|
3225 | pb_send2target($action,$pbscript1,$v,$vexist,$vpid,$snapme,$pbstep);
|
---|
3226 | }
|
---|
3227 | return($vexist,$vpid);
|
---|
3228 | }
|
---|
3229 |
|
---|
3230 | # Return string for date synchro
|
---|
3231 | sub pb_date2v {
|
---|
3232 |
|
---|
3233 | my $vtype = shift;
|
---|
3234 | my $pbos = shift;
|
---|
3235 | my $nontp = shift || 0;
|
---|
3236 |
|
---|
3237 | pb_log(1,"Entering pb_date2v\n");
|
---|
3238 | # VE gets time from parent OS.
|
---|
3239 | return "/bin/true" if ($vtype) =~ /^ve/o;
|
---|
3240 |
|
---|
3241 | my ($ntp) = pb_conf_get_if($vtype."ntp");
|
---|
3242 | my $vntp = undef;
|
---|
3243 | my $ntpline = undef;
|
---|
3244 | $vntp = $ntp->{$ENV{'PBPROJ'}} if (defined $ntp);
|
---|
3245 |
|
---|
3246 | if ((defined $vntp) && ($vntp !~ /^\s*$/)) {
|
---|
3247 | pb_log(2,"ntp server is $vntp\n");
|
---|
3248 | # ntp command depends on pbos
|
---|
3249 | my $vntpcmd = pb_distro_get_param($pbos,pb_conf_get("oscmdntp"));
|
---|
3250 | $ntpline = "sudo $vntpcmd $vntp";
|
---|
3251 | }
|
---|
3252 | # Force new date to be in the future compared to the date
|
---|
3253 | # of the host by adding 1 minute
|
---|
3254 | my @date=pb_get_date();
|
---|
3255 | $date[1]++;
|
---|
3256 | my $upddate = strftime("%m%d%H%M%Y", @date);
|
---|
3257 | my $dateline = "sudo /bin/date $upddate";
|
---|
3258 | if ((defined $ntpline) && ($nontp == 0)) {
|
---|
3259 | pb_log(1,"pb_date2v returns $ntpline\n");
|
---|
3260 | return($ntpline);
|
---|
3261 | } else {
|
---|
3262 | pb_log(1,"pb_date2v returns $dateline\n");
|
---|
3263 | return($dateline);
|
---|
3264 | }
|
---|
3265 | }
|
---|
3266 |
|
---|
3267 | sub pb_build2v {
|
---|
3268 |
|
---|
3269 | my $vtype = shift;
|
---|
3270 | my $action = shift; #build, test or prep
|
---|
3271 | my $pbforce=shift; # Force stop of VM. Default not.
|
---|
3272 |
|
---|
3273 | $action = "build" if (not defined $action);
|
---|
3274 | $pbforce = 0 if (not defined $pbforce);
|
---|
3275 |
|
---|
3276 | my $pbstep;
|
---|
3277 |
|
---|
3278 | $vetype = pb_ve_get_type($vetype);
|
---|
3279 | if ($action eq "prep") {
|
---|
3280 | $pbstep = 2;
|
---|
3281 | } else {
|
---|
3282 | $pbstep = 3;
|
---|
3283 | # Need to be traversed by others if using containers
|
---|
3284 | if (($vtype eq "ve") && ($vetype eq "docker")) {
|
---|
3285 | chmod 0751,"$ENV{'PBBUILDDIR'}";
|
---|
3286 | }
|
---|
3287 | }
|
---|
3288 |
|
---|
3289 | pb_log(1,"Entering in pb_build2v ($action in $vtype) \n");
|
---|
3290 |
|
---|
3291 | # We prepare scripts for all V of this vtype to allow sharing that part
|
---|
3292 | my ($v,$all) = pb_get2v($vtype);
|
---|
3293 |
|
---|
3294 | # Send tar files when we do a global generation
|
---|
3295 | pb_build2ssh() if (($all == 1) && ($action eq "build"));
|
---|
3296 |
|
---|
3297 | # Adapt // mode to memory size
|
---|
3298 | $pbparallel = pb_set_parallel($vtype);
|
---|
3299 |
|
---|
3300 | my ($vexist,$vpid) = (undef,undef);
|
---|
3301 | my $pm;
|
---|
3302 | my $all_ok = 1;
|
---|
3303 | if (defined $pbparallel) {
|
---|
3304 | $pm = new Parallel::ForkManager($pbparallel);
|
---|
3305 |
|
---|
3306 | # Set which port the VM/RM will use to communicate
|
---|
3307 | $pm->run_on_start(\&pb_set_port);
|
---|
3308 | $pm->run_on_finish(sub { my ($pid, $code, $id, $signal, $dump) = @_;
|
---|
3309 | unless ($code == 0 && $signal == 0 && $dump == 0) {
|
---|
3310 | $all_ok = 0;
|
---|
3311 | pb_log(0,"ERROR: pid $pid failed\n");
|
---|
3312 | }
|
---|
3313 | });
|
---|
3314 | }
|
---|
3315 |
|
---|
3316 | my $counter = 0;
|
---|
3317 | foreach my $v (@$v) {
|
---|
3318 | $counter++;
|
---|
3319 | $pbscript{$v} = "$ENV{'PBDESTDIR'}/pb$action-$v";
|
---|
3320 | # Modulo 2 * pbparallel (to avoid synchronization problems)
|
---|
3321 | $counter = 1 if ((defined $pbparallel) && ($counter > 2 * $pbparallel));
|
---|
3322 | $pm->start($counter) and next if (defined $pbparallel);
|
---|
3323 | #
|
---|
3324 | # Prepare the script to be executed on the VM/VE/RM
|
---|
3325 | open(SCRIPT,"> $pbscript{$v}") || die "Unable to create $pbscript{$v}";
|
---|
3326 | print SCRIPT "#!/bin/bash\n";
|
---|
3327 |
|
---|
3328 | # Transmit the verbosity level to the virtual env/mach.
|
---|
3329 | my $verbose = "";
|
---|
3330 | my $i = 0; # minimal debug level
|
---|
3331 | while ($i lt $pbdebug) {
|
---|
3332 | $verbose .= "-v ";
|
---|
3333 | $i++;
|
---|
3334 | }
|
---|
3335 | print SCRIPT "set -e\n" if ($Global::pb_stop_on_error);
|
---|
3336 | # Activate script verbosity if at least 2 for pbdebug
|
---|
3337 | print SCRIPT "set -x\n" if ($i gt 1);
|
---|
3338 | # Quiet if asked to be so on the original system
|
---|
3339 | $verbose = "-q" if ($pbdebug eq -1);
|
---|
3340 |
|
---|
3341 | print SCRIPT "echo ... Execution needed\n";
|
---|
3342 | print SCRIPT "echo ==== Start of script for $vtype $v =====\n";
|
---|
3343 | print SCRIPT "# This is in directory pbdelivery\n";
|
---|
3344 | print SCRIPT "# Setup the variables required for building\n";
|
---|
3345 | print SCRIPT "export PBPROJ=$ENV{'PBPROJ'}\n";
|
---|
3346 |
|
---|
3347 | if (($action eq "build") || ($action eq "prep")) {
|
---|
3348 | print SCRIPT "# Preparation for pb\n";
|
---|
3349 | print SCRIPT "rm -f \$HOME/.pbrc\n";
|
---|
3350 | print SCRIPT "cp .pbrc \$HOME\n";
|
---|
3351 | # TODO: Why ?
|
---|
3352 | print SCRIPT "cd ..\n";
|
---|
3353 | }
|
---|
3354 |
|
---|
3355 | # VE needs a good /proc, tolerate one being potentially left around after a failure
|
---|
3356 | if (($vtype eq "ve") && ($vetype ne "docker")) {
|
---|
3357 | print SCRIPT "[ -d /proc/1 ] || sudo /bin/mount -t proc /proc /proc\n";
|
---|
3358 | }
|
---|
3359 |
|
---|
3360 | # Get distro context
|
---|
3361 | my $pbos = pb_distro_get_context($v);
|
---|
3362 |
|
---|
3363 | my $ntpline = pb_date2v($vtype,$pbos);
|
---|
3364 | print SCRIPT "# Time sync\n";
|
---|
3365 | print SCRIPT "echo setting up date with $ntpline\n";
|
---|
3366 | print SCRIPT "$ntpline\n";
|
---|
3367 | print SCRIPT "if [ \$\? -ne 0 ]; then\n";
|
---|
3368 | # Force if bad status a trick using date
|
---|
3369 | $ntpline = pb_date2v($vtype,$pbos, 1);
|
---|
3370 | print SCRIPT " $ntpline\n";
|
---|
3371 | print SCRIPT "fi\n";
|
---|
3372 | # Use potential local proxy declaration in case we need it to download repo, pkgs, ...
|
---|
3373 | if (defined $ENV{'http_proxy'}) {
|
---|
3374 | print SCRIPT "export http_proxy=\"$ENV{'http_proxy'}\"\n";
|
---|
3375 | }
|
---|
3376 |
|
---|
3377 | if (defined $ENV{'ftp_proxy'}) {
|
---|
3378 | print SCRIPT "export ftp_proxy=\"$ENV{'ftp_proxy'}\"\n";
|
---|
3379 | }
|
---|
3380 |
|
---|
3381 | # Get list of packages to build/test and get some ENV vars as well
|
---|
3382 | my $ptr = pb_get_pkg();
|
---|
3383 | @pkgs = @$ptr;
|
---|
3384 | my $p = join(' ',@pkgs) if (@pkgs);
|
---|
3385 | print SCRIPT "export PBPROJVER=$ENV{'PBPROJVER'}\n";
|
---|
3386 | print SCRIPT "export PBPROJTAG=$ENV{'PBPROJTAG'}\n";
|
---|
3387 | print SCRIPT "export PBPACKAGER=\"$ENV{'PBPACKAGER'}\"\n";
|
---|
3388 |
|
---|
3389 | # We may need to do some other tasks before building. Read a script here to finish setup
|
---|
3390 | if (-x "$ENV{'PBDESTDIR'}/pb$vtype".".pre") {
|
---|
3391 | print SCRIPT "# Special pre-instructions to be launched\n";
|
---|
3392 | print SCRIPT pb_get_content("$ENV{'PBDESTDIR'}/pb$vtype".".pre");
|
---|
3393 | }
|
---|
3394 |
|
---|
3395 | if (-x "$ENV{'PBDESTDIR'}/pb$vtype"."$action.pre") {
|
---|
3396 | print SCRIPT "# Special pre-$action instructions to be launched\n";
|
---|
3397 | print SCRIPT pb_get_content("$ENV{'PBDESTDIR'}/pb$vtype"."$action.pre");
|
---|
3398 | }
|
---|
3399 |
|
---|
3400 | print SCRIPT "# $action\n";
|
---|
3401 | print SCRIPT "echo $action"."ing packages on $vtype...\n";
|
---|
3402 |
|
---|
3403 | if (($action eq "test") && (! -x "$ENV{'PBDESTDIR'}/pbtest")) {
|
---|
3404 | confess "No test script ($ENV{'PBDESTDIR'}/pbtest) found when in test mode. Aborting ...";
|
---|
3405 | }
|
---|
3406 | print SCRIPT "pb -h\n";
|
---|
3407 | my $act = "$action"."2pkg";
|
---|
3408 | $act = "build2prep" if ($action eq "prep");
|
---|
3409 | print SCRIPT "pb $verbose -p $ENV{'PBPROJ'} $act $p\n";
|
---|
3410 |
|
---|
3411 | if (($vtype eq "ve") && ($vetype ne "docker")) {
|
---|
3412 | print SCRIPT "sudo /bin/umount /proc\n";
|
---|
3413 | }
|
---|
3414 |
|
---|
3415 | # We may need to do some other tasks after building. Read a script here to exit properly
|
---|
3416 | if (-x "$ENV{'PBDESTDIR'}/pb$vtype"."$action.post") {
|
---|
3417 | print SCRIPT "# Special post-$action instructions to be launched\n";
|
---|
3418 | print SCRIPT pb_get_content("$ENV{'PBDESTDIR'}/pb$vtype"."$action.post");
|
---|
3419 | }
|
---|
3420 |
|
---|
3421 | if (-x "$ENV{'PBDESTDIR'}/pb$vtype".".post") {
|
---|
3422 | print SCRIPT "# Special post-instructions to be launched\n";
|
---|
3423 | print SCRIPT pb_get_content("$ENV{'PBDESTDIR'}/pb$vtype".".post");
|
---|
3424 | }
|
---|
3425 |
|
---|
3426 | print SCRIPT "echo ==== End of script $$ for $vtype $v =====\n";
|
---|
3427 | close(SCRIPT);
|
---|
3428 | chmod 0755,"$pbscript{$v}";
|
---|
3429 |
|
---|
3430 | $pm->finish if (defined $pbparallel);
|
---|
3431 | }
|
---|
3432 | $pm->wait_all_children if (defined $pbparallel);
|
---|
3433 | die "Aborting, one or more of the children failed." if ((not $all_ok) && ($Global::pb_stop_on_error));
|
---|
3434 |
|
---|
3435 | # Launch the VM/VE/RM
|
---|
3436 | # docker image to use if needed
|
---|
3437 | # Gather all required files to send them to the VM/VE
|
---|
3438 | # and launch the build through pbscript
|
---|
3439 | pb_parallel_launchv(\%pbscript,$vtype,uc($vtype).$action,$pbstep,$pbimage,$pbforce);
|
---|
3440 | }
|
---|
3441 |
|
---|
3442 |
|
---|
3443 | sub pb_clean {
|
---|
3444 |
|
---|
3445 | my $sleep=10;
|
---|
3446 | die "Unable to get env var PBDESTDIR" if (not defined $ENV{'PBDESTDIR'});
|
---|
3447 | die "Unable to get env var PBBUILDDIR" if (not defined $ENV{'PBBUILDDIR'});
|
---|
3448 | pb_log(0,"We will now wait $sleep s before removing both directories\n$ENV{'PBDESTDIR'} and $ENV{'PBBUILDDIR'}.\nPlease break me if this is wrong\n");
|
---|
3449 | sleep $sleep;
|
---|
3450 | pb_rm_rf($ENV{'PBDESTDIR'});
|
---|
3451 | pb_rm_rf($ENV{'PBBUILDDIR'});
|
---|
3452 | }
|
---|
3453 |
|
---|
3454 | sub pb_newver {
|
---|
3455 |
|
---|
3456 | die "-V Version parameter needed" if ((not defined $newver) || ($newver eq ""));
|
---|
3457 |
|
---|
3458 | # Need this call for PBDIR
|
---|
3459 | my ($scheme2,$uri) = pb_cms_init($pbinit);
|
---|
3460 |
|
---|
3461 | my ($pbconf,$pburl) = pb_conf_get("pbconfurl","pburl");
|
---|
3462 | $uri = $pbconf->{$ENV{'PBPROJ'}};
|
---|
3463 | my ($scheme, $account, $host, $port, $path) = pb_get_uri($uri);
|
---|
3464 |
|
---|
3465 | # Checking CMS repositories status
|
---|
3466 | ($scheme2, $account, $host, $port, $path) = pb_get_uri($pburl->{$ENV{'PBPROJ'}});
|
---|
3467 |
|
---|
3468 | if ($scheme !~ /^svn/) {
|
---|
3469 | die "Only SVN is supported at the moment";
|
---|
3470 | }
|
---|
3471 |
|
---|
3472 | my $res = pb_vcs_isdiff($scheme,$ENV{'PBROOTDIR'});
|
---|
3473 | die "ERROR: No differences accepted in CMS for $ENV{'PBROOTDIR'} before creating a new version" if ($res != 0);
|
---|
3474 |
|
---|
3475 | $res = pb_vcs_isdiff($scheme2,$ENV{'PBDIR'});
|
---|
3476 | die "ERROR: No differences accepted in CMS for $ENV{'PBDIR'} before creating a new version" if ($res != 0);
|
---|
3477 |
|
---|
3478 | # Tree identical between PBCONFDIR and PBROOTDIR. The delta is what
|
---|
3479 | # we want to get for the root of the new URL
|
---|
3480 |
|
---|
3481 | my $oldver = $ENV{'PBROOTDIR'};
|
---|
3482 | $oldver =~ s|^$ENV{'PBCONFDIR'}||;
|
---|
3483 |
|
---|
3484 | pb_log(2, "PBCONFDIR: $ENV{'PBCONFDIR'}\nPBROOTDIR: $ENV{'PBROOTDIR'}\n");
|
---|
3485 |
|
---|
3486 | my $newurl = "$uri/$newver";
|
---|
3487 | # Should probably use projver in the old file
|
---|
3488 | my $oldvertxt= basename($oldver);
|
---|
3489 | my $newvertxt = basename($newver);
|
---|
3490 |
|
---|
3491 | # Duplicate and extract project-builder part
|
---|
3492 | pb_log(2,"Copying $uri/$oldver to $newurl\n");
|
---|
3493 | pb_vcs_copy($scheme,"$uri/$oldver",$newurl);
|
---|
3494 | pb_log(2,"Checkout $newurl to $ENV{'PBCONFDIR'}/$newver\n");
|
---|
3495 | pb_vcs_up($scheme,"$ENV{'PBCONFDIR'}");
|
---|
3496 |
|
---|
3497 | # Duplicate and extract project
|
---|
3498 | my $newurl2 = "$pburl->{$ENV{'PBPROJ'}}/$newver";
|
---|
3499 |
|
---|
3500 | pb_log(2,"Copying $pburl->{$ENV{'PBPROJ'}}/$oldver to $newurl2\n");
|
---|
3501 | pb_vcs_copy($scheme2,"$pburl->{$ENV{'PBPROJ'}}/$oldver",$newurl2);
|
---|
3502 |
|
---|
3503 | my $tmp = $ENV{'PBDIR'};
|
---|
3504 | $tmp =~ s|$oldver$||;
|
---|
3505 | pb_log(2,"Checkout $newurl2 to $tmp/$newver\n");
|
---|
3506 | pb_vcs_up($scheme2,"$tmp");
|
---|
3507 |
|
---|
3508 | # Update the .pb file
|
---|
3509 | open(FILE,"$ENV{'PBCONFDIR'}/$newver/$ENV{'PBPROJ'}.pb") || die "Unable to open $ENV{'PBCONFDIR'}/$newver/$ENV{'PBPROJ'}.pb";
|
---|
3510 | open(OUT,"> $ENV{'PBCONFDIR'}/$newver/$ENV{'PBPROJ'}.pb.new") || die "Unable to write to $ENV{'PBCONFDIR'}/$newver/$ENV{'PBPROJ'}.pb.new";
|
---|
3511 | while(<FILE>) {
|
---|
3512 | if (/^projver\s+$ENV{'PBPROJ'}\s*=\s*$oldvertxt$/) {
|
---|
3513 | s/^projver\s+$ENV{'PBPROJ'}\s*=\s*$oldvertxt$/projver $ENV{'PBPROJ'} = $newvertxt/;
|
---|
3514 | pb_log(0,"Changing projver from $oldvertxt to $newvertxt in $ENV{'PBCONFDIR'}/$newver/$ENV{'PBPROJ'}.pb\n");
|
---|
3515 | }
|
---|
3516 | if (/^testver/) {
|
---|
3517 | s/^testver/#testver/;
|
---|
3518 | pb_log(0,"Commenting testver in $ENV{'PBCONFDIR'}/$newver/$ENV{'PBPROJ'}.pb\n") if (/^testver/);
|
---|
3519 | }
|
---|
3520 | if (/^delivery/) {
|
---|
3521 | my $txt = $_;
|
---|
3522 | chomp($txt);
|
---|
3523 | pb_log(0,"Please check delivery ($txt) in $ENV{'PBCONFDIR'}/$newver/$ENV{'PBPROJ'}.pb\n");
|
---|
3524 | }
|
---|
3525 | print OUT $_;
|
---|
3526 | }
|
---|
3527 | close(FILE);
|
---|
3528 | close(OUT);
|
---|
3529 | rename("$ENV{'PBCONFDIR'}/$newver/$ENV{'PBPROJ'}.pb.new","$ENV{'PBCONFDIR'}/$newver/$ENV{'PBPROJ'}.pb");
|
---|
3530 |
|
---|
3531 | # Checking pbcl files
|
---|
3532 | foreach my $f (<$ENV{'PBROOTDIR'}/*/pbcl>) {
|
---|
3533 | # Compute new pbcl file
|
---|
3534 | my $f2 = $f;
|
---|
3535 | $f2 =~ s|$ENV{'PBROOTDIR'}|$ENV{'PBCONFDIR'}/$newver/|;
|
---|
3536 | open(PBCL,$f) || die "Unable to open $f";
|
---|
3537 | my $foundnew = 0;
|
---|
3538 | while (<PBCL>) {
|
---|
3539 | $foundnew = 1 if (/^$newvertxt \(/);
|
---|
3540 | }
|
---|
3541 | close(PBCL);
|
---|
3542 | open(OUT,"> $f2") || die "Unable to write to $f2: $!";
|
---|
3543 | open(PBCL,$f) || die "Unable to open $f";
|
---|
3544 | # We skip as long as we do not reach the new version in the log (case of devel)
|
---|
3545 | my $skip = 1;
|
---|
3546 | # We let the 4 first lines
|
---|
3547 | my $cnt = 0;
|
---|
3548 | while (<PBCL>) {
|
---|
3549 | $skip = 0 if (/^$newvertxt \(/);
|
---|
3550 | if ((/^$oldvertxt \(/) && ($foundnew == 0)) {
|
---|
3551 | print OUT "$newvertxt ($pbdate)\n";
|
---|
3552 | print OUT "- TBD\n";
|
---|
3553 | print OUT "\n";
|
---|
3554 | pb_log(0,"WARNING: version $newvertxt not found in $f so added to $f2...\n") if ($foundnew == 0);
|
---|
3555 | }
|
---|
3556 | if ($cnt <= 4) {
|
---|
3557 | print OUT "$_" if (not /^$oldvertxt \(/);
|
---|
3558 | } else {
|
---|
3559 | print OUT "$_" if ($skip == 0);
|
---|
3560 | }
|
---|
3561 | $cnt++;
|
---|
3562 | }
|
---|
3563 | close(OUT);
|
---|
3564 | close(PBCL);
|
---|
3565 | }
|
---|
3566 |
|
---|
3567 | pb_log(2,"Checkin $ENV{'PBCONFDIR'}/$newver\n");
|
---|
3568 | pb_vcs_checkin($scheme,"$ENV{'PBCONFDIR'}/$newver","updated to $ENV{'PBCONFDIR'}");
|
---|
3569 | }
|
---|
3570 |
|
---|
3571 | #
|
---|
3572 | # Return the list of VMs/VEs/RMs we are working on
|
---|
3573 | # $all is a flag to know if we return all of them
|
---|
3574 | # or only some (if all we publish also tar files in addition to pkgs
|
---|
3575 | #
|
---|
3576 | sub pb_get2v {
|
---|
3577 |
|
---|
3578 | my $vtype = shift;
|
---|
3579 | my @v;
|
---|
3580 | my $all = 0;
|
---|
3581 | my $pbv = 'PBV';
|
---|
3582 | my $vlist = $vtype."list";
|
---|
3583 |
|
---|
3584 | # Get VM/VE list
|
---|
3585 | if ((not defined $ENV{$pbv}) || ($ENV{$pbv} =~ /^all$/)) {
|
---|
3586 | my ($ptr) = pb_conf_get($vlist);
|
---|
3587 | $ENV{$pbv} = $ptr->{$ENV{'PBPROJ'}};
|
---|
3588 | $all = 1;
|
---|
3589 | }
|
---|
3590 | pb_log(2,"$vtype: $ENV{$pbv}\n");
|
---|
3591 | @v = split(/,/,$ENV{$pbv});
|
---|
3592 | return(\@v,$all);
|
---|
3593 | }
|
---|
3594 |
|
---|
3595 | # This function creates a giant script to configure a particular VM/VE/RM, it then copies the
|
---|
3596 | # script to the target.
|
---|
3597 |
|
---|
3598 | # Function to create a potentialy missing pb account on the VM/VE/RM, and adds it to sudo
|
---|
3599 | # Needs to use root account to connect to the VM/VE/RM
|
---|
3600 | # pb will take your local public SSH key to access
|
---|
3601 | # the pb account in the VM/VE/RM later on if needed
|
---|
3602 | sub pb_setup2v {
|
---|
3603 |
|
---|
3604 | my $vtype = shift;
|
---|
3605 | my $sbx = shift;
|
---|
3606 | my $pbstep = 1;
|
---|
3607 |
|
---|
3608 | my ($vm,$all) = pb_get2v($vtype);
|
---|
3609 |
|
---|
3610 | # Adapt // mode to memory size
|
---|
3611 | $pbparallel = pb_set_parallel($vtype);
|
---|
3612 |
|
---|
3613 | my $pm;
|
---|
3614 | my $all_ok = 1;
|
---|
3615 | if (defined $pbparallel) {
|
---|
3616 | $pm = new Parallel::ForkManager($pbparallel);
|
---|
3617 |
|
---|
3618 | # Set which port the VM/RM will use to communicate
|
---|
3619 | $pm->run_on_start(\&pb_set_port);
|
---|
3620 | $pm->run_on_finish(sub { my ($pid, $code, $id, $signal, $dump) = @_;
|
---|
3621 | $all_ok = 0 unless (($code == 0) && ($signal == 0) && ($dump == 0));
|
---|
3622 | });
|
---|
3623 | }
|
---|
3624 |
|
---|
3625 | my $counter = 0;
|
---|
3626 | foreach my $v (@$vm) {
|
---|
3627 | # Script generated - should be before the fork - after the %pbscript hash isn't updated correctly
|
---|
3628 | $pbscript{$v} = "$ENV{'PBDESTDIR'}/setupv-$v";
|
---|
3629 |
|
---|
3630 | $counter++;
|
---|
3631 | # Modulo 2 * pbparallel (to avoid synchronization problems)
|
---|
3632 | $counter = 1 if ((defined $pbparallel) && ($counter > 2 * $pbparallel));
|
---|
3633 | $pm->start($counter) and next if (defined $pbparallel);
|
---|
3634 |
|
---|
3635 | # Get distro context
|
---|
3636 | my $pbos = pb_distro_get_context($v);
|
---|
3637 |
|
---|
3638 | # Deal with date sync.
|
---|
3639 | my $ntpline = pb_date2v($vtype,$pbos);
|
---|
3640 |
|
---|
3641 | # Name of the account to deal with for VM/VE/RM
|
---|
3642 | # Do not use the one passed potentially with -a
|
---|
3643 | my ($pbac) = pb_conf_get($vtype."login");
|
---|
3644 | my $key = "";
|
---|
3645 |
|
---|
3646 | # Prepare the script to be executed on the VM/VE/RM
|
---|
3647 | # in $ENV{'PBDESTDIR'}/setupv
|
---|
3648 |
|
---|
3649 | pb_log(2,"Preparing setup script for $v in $pbscript{$v}\n");
|
---|
3650 | open(SCRIPT,"> $pbscript{$v}") || die "Unable to create $pbscript{$v}";
|
---|
3651 |
|
---|
3652 | print SCRIPT << 'EOF';
|
---|
3653 | #!/usr/bin/perl -w
|
---|
3654 |
|
---|
3655 | use strict;
|
---|
3656 | use File::Copy;
|
---|
3657 |
|
---|
3658 | # We should not need in this script more functions than what is provided
|
---|
3659 | # by Base, Conf and Distribution to avoid problems at exec time.
|
---|
3660 | # They are appended at the end.
|
---|
3661 |
|
---|
3662 | # Define mandatory global vars
|
---|
3663 | EOF
|
---|
3664 | my $ppref = "our";
|
---|
3665 | $ppref = "my" if (($pbos->{'name'} =~ /^redhat/) && ($pbos->{'version'} =~ /^6/));
|
---|
3666 | print SCRIPT << "EOF";
|
---|
3667 | $ppref \$pbdebug = $pbdebug;
|
---|
3668 | $ppref \$pbLOG;
|
---|
3669 | $ppref \$pbsynmsg = "pbscript";
|
---|
3670 | $ppref \$pbdisplaytype = "text";
|
---|
3671 | $ppref \$pblocale = "";
|
---|
3672 | EOF
|
---|
3673 | print SCRIPT << 'EOF';
|
---|
3674 | pb_log_init($pbdebug, $pbLOG);
|
---|
3675 | EOF
|
---|
3676 | print SCRIPT << "EOF";
|
---|
3677 | \$Global::pb_stop_on_error = $Global::pb_stop_on_error;
|
---|
3678 | pb_temp_init($pbkeep);
|
---|
3679 | pb_conf_init("$ENV{'PBPROJ'}");
|
---|
3680 |
|
---|
3681 | EOF
|
---|
3682 |
|
---|
3683 | if ($vtype eq "ve") {
|
---|
3684 | $vetype = pb_ve_get_type($vetype);
|
---|
3685 | if ($vetype ne "docker") {
|
---|
3686 | print SCRIPT << "EOF";
|
---|
3687 | # For VE we first need to mount some FS
|
---|
3688 | pb_system("mount -t proc /proc /proc") unless (-d "/proc/$$");
|
---|
3689 | EOF
|
---|
3690 | }
|
---|
3691 | print SCRIPT << "EOF";
|
---|
3692 | # For VE we need a good null dev
|
---|
3693 | # Except for RedHat 6.2 where it's good and doesnt like being recreated
|
---|
3694 | EOF
|
---|
3695 | print SCRIPT "pb_system('rm -f /dev/null; mknod /dev/null c 1 3; chmod 777 /dev/null');\n" unless (($pbos->{'name'} =~ /^redhat/) && ($pbos->{'version'} =~ /^6/));
|
---|
3696 | } elsif ($vtype =~ /(v|r)m/) {
|
---|
3697 | # Nothing to do now.
|
---|
3698 | my $useless = 0;
|
---|
3699 | } else {
|
---|
3700 | die "Unknown virtual type $vtype";
|
---|
3701 | }
|
---|
3702 |
|
---|
3703 | print SCRIPT << 'EOF';
|
---|
3704 |
|
---|
3705 | # Adds $pbac->{$ENV{'PBPROJ'}} as an account if needed
|
---|
3706 | #
|
---|
3707 | my $file="/etc/passwd";
|
---|
3708 | open(PBFILE,$file) || die "Unable to open $file";
|
---|
3709 | my $found = 0;
|
---|
3710 | while (<PBFILE>) {
|
---|
3711 | EOF
|
---|
3712 | print SCRIPT << "EOF";
|
---|
3713 | \$found = 1 if (/^$pbac->{$ENV{'PBPROJ'}}:/);
|
---|
3714 | EOF
|
---|
3715 |
|
---|
3716 | # TODO: use an external parameter
|
---|
3717 | my $home = "/home";
|
---|
3718 | # Solaris doesn't like that we use /home
|
---|
3719 | $home = "/export/home" if ($pbos->{'type'} eq "pkg");
|
---|
3720 |
|
---|
3721 | print SCRIPT << "EOF";
|
---|
3722 | }
|
---|
3723 | close(PBFILE);
|
---|
3724 |
|
---|
3725 | if ( \$found == 0 ) {
|
---|
3726 | if ( ! -d "$home" ) {
|
---|
3727 | pb_mkdir_p("$home");
|
---|
3728 | }
|
---|
3729 | EOF
|
---|
3730 | # TODO: Level of portability of these cmds ? Critical now for RM
|
---|
3731 | # TODO: Check existence before adding to avoid errors
|
---|
3732 | print SCRIPT << "EOF";
|
---|
3733 | pb_system("/usr/sbin/groupadd $pbac->{$ENV{'PBPROJ'}}","Adding group $pbac->{$ENV{'PBPROJ'}}");
|
---|
3734 | pb_system("/usr/sbin/useradd -g $pbac->{$ENV{'PBPROJ'}} -m -d $home/$pbac->{$ENV{'PBPROJ'}} $pbac->{$ENV{'PBPROJ'}}","Adding user $pbac->{$ENV{'PBPROJ'}} (group $pbac->{$ENV{'PBPROJ'}} - home $home/$pbac->{$ENV{'PBPROJ'}})");
|
---|
3735 | }
|
---|
3736 | EOF
|
---|
3737 |
|
---|
3738 | # Copy the content of our local conf file to the VM/VE/RM
|
---|
3739 | my $content = pb_get_content(pb_distro_conffile());
|
---|
3740 | print SCRIPT << "EOF";
|
---|
3741 | #
|
---|
3742 | # Create a temporary local conf file for distribution support
|
---|
3743 | # This is created here before its use later. Its place is hardcoded, so no choice for the path
|
---|
3744 | #
|
---|
3745 | my \$tempconf = pb_distro_conffile();
|
---|
3746 | pb_mkdir_p(dirname(\$tempconf));
|
---|
3747 | open(CONF,"> \$tempconf") || die "Unable to create \$tempconf";
|
---|
3748 | print CONF q{$content};
|
---|
3749 | close(CONF);
|
---|
3750 | EOF
|
---|
3751 |
|
---|
3752 | if ($vtype =~ /(v|r)m/) {
|
---|
3753 | print SCRIPT << "EOF";
|
---|
3754 | # allow ssh entry to build
|
---|
3755 | #
|
---|
3756 | mkdir "$home/$pbac->{$ENV{'PBPROJ'}}/.ssh",0700;
|
---|
3757 | # Allow those accessing root to access the build account
|
---|
3758 | copy("\$ENV{'HOME'}/.ssh/authorized_keys","$home/$pbac->{$ENV{'PBPROJ'}}/.ssh/authorized_keys");
|
---|
3759 | chmod 0600,".ssh/authorized_keys";
|
---|
3760 | pb_system("chown -R $pbac->{$ENV{'PBPROJ'}}:$pbac->{$ENV{'PBPROJ'}} $home/$pbac->{$ENV{'PBPROJ'}}","Finish setting up the account env for $pbac->{$ENV{'PBPROJ'}}");
|
---|
3761 |
|
---|
3762 | EOF
|
---|
3763 | }
|
---|
3764 | print SCRIPT << 'EOF';
|
---|
3765 | # No passwd for build account only keys
|
---|
3766 | $file="/etc/shadow";
|
---|
3767 | if (-f $file) {
|
---|
3768 | open(PBFILE,$file) || die "Unable to open $file";
|
---|
3769 | open(PBOUT,"> $file.new") || die "Unable to open $file.new";
|
---|
3770 | while (<PBFILE>) {
|
---|
3771 | EOF
|
---|
3772 | print SCRIPT << "EOF";
|
---|
3773 | s/^$pbac->{$ENV{'PBPROJ'}}:\!\!:/$pbac->{$ENV{'PBPROJ'}}:*:/;
|
---|
3774 | s/^$pbac->{$ENV{'PBPROJ'}}:\!:/$pbac->{$ENV{'PBPROJ'}}:*:/; #SLES 9 e.g.
|
---|
3775 | s/^$pbac->{$ENV{'PBPROJ'}}:\\*LK\\*:/$pbac->{$ENV{'PBPROJ'}}:NP:/; #Solaris e.g.
|
---|
3776 | EOF
|
---|
3777 | print SCRIPT << 'EOF';
|
---|
3778 | print PBOUT $_;
|
---|
3779 | }
|
---|
3780 | close(PBFILE);
|
---|
3781 | close(PBOUT);
|
---|
3782 | rename("$file.new",$file);
|
---|
3783 | chmod 0640,$file;
|
---|
3784 | }
|
---|
3785 |
|
---|
3786 | # TODO: systemd support
|
---|
3787 | # Keep the VM in text mode
|
---|
3788 | $file="/etc/inittab";
|
---|
3789 | if (-f $file) {
|
---|
3790 | open(PBFILE,$file) || die "Unable to open $file";
|
---|
3791 | open(PBOUT,"> $file.new") || die "Unable to open $file.new";
|
---|
3792 | while (<PBFILE>) {
|
---|
3793 | s/^(..):5:initdefault:$/$1:3:initdefault:/;
|
---|
3794 | print PBOUT $_;
|
---|
3795 | }
|
---|
3796 | close(PBFILE);
|
---|
3797 | close(PBOUT);
|
---|
3798 | rename("$file.new",$file);
|
---|
3799 | chmod 0640,$file;
|
---|
3800 | }
|
---|
3801 |
|
---|
3802 | # pb has to be added to portage group on gentoo
|
---|
3803 |
|
---|
3804 | # We need to have that pb_distro_get_context function
|
---|
3805 | # Get it from Project-Builder::Distribution
|
---|
3806 | # And we now need the conf file required for this to work created above
|
---|
3807 |
|
---|
3808 | my $pbos = pb_distro_get_context();
|
---|
3809 | print "distro tuple: ".Dumper($pbos)."\n";
|
---|
3810 |
|
---|
3811 | # Adapt sudoers
|
---|
3812 | # sudo is not default on Solaris and needs to be installed first
|
---|
3813 | # from http://www.sunfreeware.com/programlistsparc10.html#sudo
|
---|
3814 | if ($pbos->{'type'} eq "pkg") {
|
---|
3815 | $file="/usr/local/etc/sudoers";
|
---|
3816 | } else {
|
---|
3817 | $file="/etc/sudoers";
|
---|
3818 | }
|
---|
3819 | open(PBFILE,$file) || die "Unable to open $file";
|
---|
3820 | open(PBOUT,"> $file.new") || die "Unable to open $file.new";
|
---|
3821 | while (<PBFILE>) {
|
---|
3822 | EOF
|
---|
3823 | # Skip what will be generated
|
---|
3824 | print SCRIPT << "EOF";
|
---|
3825 | next if (/^$pbac->{$ENV{'PBPROJ'}}\\s+/);
|
---|
3826 | next if (/^Defaults:$pbac->{$ENV{'PBPROJ'}}\\s+/);
|
---|
3827 | next if (/^Defaults:root\\s+\!requiretty/);
|
---|
3828 | next if (/^Defaults:root\\s+env_keep/);
|
---|
3829 | EOF
|
---|
3830 | print SCRIPT << 'EOF';
|
---|
3831 | s/Defaults\s+requiretty/# disable Defaults: requiretty/;
|
---|
3832 | print PBOUT $_;
|
---|
3833 | }
|
---|
3834 | close(PBFILE);
|
---|
3835 | EOF
|
---|
3836 | print SCRIPT << "EOF";
|
---|
3837 | # Some distro force requiretty at compile time, so disable here
|
---|
3838 | print PBOUT "Defaults:$pbac->{$ENV{'PBPROJ'}} !requiretty\\n";
|
---|
3839 | print PBOUT "Defaults:root !requiretty\\n";
|
---|
3840 | EOF
|
---|
3841 | # RH 6.2 sudo doesn't support env_keep
|
---|
3842 | if (($pbos->{'name'} ne "redhat") && ($pbos->{'version'} ne "6.2")) {
|
---|
3843 | print SCRIPT << "EOF";
|
---|
3844 | # Keep proxy configuration while using sudo
|
---|
3845 | print PBOUT "Defaults:$pbac->{$ENV{'PBPROJ'}} env_keep += \\\"http_proxy ftp_proxy\\\"\\n";
|
---|
3846 | print PBOUT "Defaults:root env_keep += \\\"http_proxy ftp_proxy\\\"\\n";
|
---|
3847 | EOF
|
---|
3848 | }
|
---|
3849 | # Try to restrict security to what is really needed
|
---|
3850 | if ($vtype =~ /^vm/) {
|
---|
3851 | my $hpath = pb_distro_get_param($pbos,pb_conf_get("ospathcmd-halt"));
|
---|
3852 | my @sudocmds = pb_get_sudocmds($pbos,$ntpline,"sudo $hpath","sudo /bin/date");
|
---|
3853 | print SCRIPT << "EOF";
|
---|
3854 | # This is needed in order to be able on VM to halt the machine from the $pbac->{$ENV{'PBPROJ'}} account at least
|
---|
3855 | # Build account $pbac->{$ENV{'PBPROJ'}} in VM also needs to setup date and install deps.
|
---|
3856 | # Nothing else should be needed
|
---|
3857 | EOF
|
---|
3858 | foreach my $c (@sudocmds) {
|
---|
3859 | print SCRIPT "print PBOUT \"$pbac->{$ENV{'PBPROJ'}} ALL = NOPASSWD: $c\\n\";\n";
|
---|
3860 | }
|
---|
3861 | } elsif ($vtype =~ /^rm/) {
|
---|
3862 | my @sudocmds = pb_get_sudocmds($pbos,$ntpline);
|
---|
3863 | print SCRIPT << "EOF";
|
---|
3864 | # Build account $pbac->{$ENV{'PBPROJ'}} in RM only needs to setup date and install deps if needed each time
|
---|
3865 | EOF
|
---|
3866 | foreach my $c (@sudocmds) {
|
---|
3867 | print SCRIPT "print PBOUT \"$pbac->{$ENV{'PBPROJ'}} ALL = NOPASSWD: $c\\n\";\n";
|
---|
3868 | }
|
---|
3869 | } else {
|
---|
3870 | print SCRIPT << "EOF";
|
---|
3871 | # Build account $pbac->{$ENV{'PBPROJ'}} for VE needs to do a lot in the host (and chroot), so allow without restriction for now
|
---|
3872 | print PBOUT "$pbac->{$ENV{'PBPROJ'}} ALL=(ALL) NOPASSWD:ALL\n";
|
---|
3873 | EOF
|
---|
3874 | }
|
---|
3875 | my $sudomode = pb_distro_get_param($pbos,pb_conf_get("ossudoersmode"));
|
---|
3876 | print SCRIPT << 'EOF';
|
---|
3877 | close(PBOUT);
|
---|
3878 | rename("$file.new",$file);
|
---|
3879 | EOF
|
---|
3880 | print SCRIPT << "EOF";
|
---|
3881 | chmod 0$sudomode,\$file;
|
---|
3882 |
|
---|
3883 | EOF
|
---|
3884 |
|
---|
3885 | if ($vtype =~ /(v|r)m/) {
|
---|
3886 | # Sync date
|
---|
3887 | # do it after sudoers is setup
|
---|
3888 | print SCRIPT "my \$res = pb_system(\"$ntpline\",\"Updating time with $ntpline\",\"mayfailverbose\");\n";
|
---|
3889 | print SCRIPT "if (\$res != 0) {\n";
|
---|
3890 | # try again with another method
|
---|
3891 | $ntpline = pb_date2v($vtype,$pbos,1);
|
---|
3892 | print SCRIPT " pb_system(\"$ntpline\",\"Updating time with $ntpline\",\"mayfailverbose\");\n";
|
---|
3893 | print SCRIPT "}\n";
|
---|
3894 | }
|
---|
3895 |
|
---|
3896 | print SCRIPT << 'EOF';
|
---|
3897 |
|
---|
3898 | # We may need a proxy configuration. Get it from the local env
|
---|
3899 | pb_apply_conf_proxy($pbos);
|
---|
3900 |
|
---|
3901 | # First install all required packages
|
---|
3902 | pb_system("dnf clean all","Cleaning dnf env","mayfail") if (($pbos->{'name'} eq "fedora") && ($pbos->{'version'} >= 22));
|
---|
3903 | pb_system("yum clean all","Cleaning yum env","mayfail") if ((($pbos->{'name'} eq "fedora") && ($pbos->{'version'} < 22)) || ($pbos->{'name'} eq "asianux") || ($pbos->{'name'} eq "rhel"));
|
---|
3904 | my ($ospkgdep) = pb_conf_get_if("ospkgdep");
|
---|
3905 |
|
---|
3906 | my $pkgdep = pb_distro_get_param($pbos,$ospkgdep);
|
---|
3907 | pb_distro_installdeps(undef,$pbos,pb_distro_only_deps_needed($pbos,join(' ',split(/,/,$pkgdep))));
|
---|
3908 |
|
---|
3909 | EOF
|
---|
3910 | my $itype = pb_distro_get_param($pbos,pb_conf_get("pbinstalltype"));
|
---|
3911 | # Install from sandbox mean a file base install
|
---|
3912 | $itype = "file" if (defined $sbx);
|
---|
3913 | if ($itype =~ /^file/) {
|
---|
3914 | my $cmdget;
|
---|
3915 | if (defined $sbx) {
|
---|
3916 | # Install from sandbox mean using the result of the just passed sbx2build command
|
---|
3917 | # Get content saved in cms2build
|
---|
3918 | my ($pkg) = pb_conf_read("$ENV{'PBDESTDIR'}/$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}.pb","pbpkg");
|
---|
3919 | my $pbextdir = pb_get_extdir();
|
---|
3920 | die "Unable to get package list" if (not defined $pkg);
|
---|
3921 |
|
---|
3922 | # We consider 2 specific packages
|
---|
3923 | my $vertag1 = $pkg->{"ProjectBuilder"};
|
---|
3924 | my $vertag2 = $pkg->{"project-builder"};
|
---|
3925 | # get the version of the current package - maybe different
|
---|
3926 | pb_log(2,"Vertag1: $vertag1\n");
|
---|
3927 | pb_log(2,"Vertag2: $vertag2\n");
|
---|
3928 | my ($pbver1,$tmp1) = split(/-/,$vertag1);
|
---|
3929 | my ($pbver2,$tmp2) = split(/-/,$vertag2);
|
---|
3930 | $cmdget = "mv /tmp/ProjectBuilder-$pbver1$pbextdir.tar.gz ProjectBuilder-latest.tar.gz ; mv /tmp/project-builder-$pbver2$pbextdir.tar.gz project-builder-latest.tar.gz";
|
---|
3931 | } else {
|
---|
3932 | # TODO: shouldn't we have a variable to specify the source file?
|
---|
3933 | # And shouldn't we be getting the one that matches the current pb version?
|
---|
3934 | # For the moment, just take the latest stable one
|
---|
3935 | my $wget = pb_check_req("wget",0);
|
---|
3936 | $cmdget = "$wget --passive-ftp ftp://ftp.project-builder.org/src/ProjectBuilder-latest.tar.gz; $wget --passive-ftp ftp://ftp.project-builder.org/src/project-builder-latest.tar.gz";
|
---|
3937 | }
|
---|
3938 | print SCRIPT << 'EOF';
|
---|
3939 | # Then install manually the missing perl modules
|
---|
3940 | my ($osperldep,$osperlver) = pb_conf_get_if("osperldep","osperlver");
|
---|
3941 |
|
---|
3942 | my $perldep = pb_distro_get_param($pbos,$osperldep);
|
---|
3943 | my $bashopt = "";
|
---|
3944 | $bashopt = "-x" if ($pbdebug ge 1);
|
---|
3945 | my $verbopt = "mayfail";
|
---|
3946 | $verbopt = "mayfailverbose" if ($pbdebug ge 1);
|
---|
3947 | foreach my $m (split(/,/,$perldep)) {
|
---|
3948 | # Skip empty deps
|
---|
3949 | next if ($m =~ /^\s*$/);
|
---|
3950 | my $dir = $m;
|
---|
3951 | $dir =~ s/-.*//;
|
---|
3952 | pb_system("echo \"rm -rf $m* ; wget http://search.cpan.org/CPAN/modules/by-module/$dir/$m-$osperlver->{$m}.tar.gz ; gzip -cd $m-$osperlver->{$m}.tar.gz | tar xf - ; cd $m* ; if [ -f Build.PL ]; then perl Build.PL; ./Build ; ./Build install ; else perl Makefile.PL; make ; make install ; fi; cd .. ; rm -rf $m*\" | bash -e $bashopt" ,"Installing perl module $m-$osperlver->{$m}",$verbopt);
|
---|
3953 | }
|
---|
3954 | EOF
|
---|
3955 | print SCRIPT << 'EOF';
|
---|
3956 | pb_system("rm -rf ProjectBuilder-* ; rm -rf project-builder-* ; rm -rf `perl -V:installvendorlib | awk -F\"'\" '{print \$2}'`/ProjectBuilder ;
|
---|
3957 | EOF
|
---|
3958 | print SCRIPT " $cmdget ; ";
|
---|
3959 | print SCRIPT << 'EOF'
|
---|
3960 | gzip -cd ProjectBuilder-latest.tar.gz | tar xf - ; cd ProjectBuilder-* ; perl Makefile.PL ; make ; make install ; cd .. ; rm -rf ProjectBuilder-* ; gzip -cd project-builder-latest.tar.gz | tar xf - ; cd project-builder-* ; perl Makefile.PL ; make ; make install ; cd .. ; rm -rf project-builder-* ;","Building Project-Builder");
|
---|
3961 | EOF
|
---|
3962 | } elsif ($itype =~ /^pkg/) {
|
---|
3963 | # pkg based install. We need to point to the project-builder.org repository
|
---|
3964 | print SCRIPT << 'EOF';
|
---|
3965 | my $pkgforpb = pb_distro_get_param($pbos,pb_conf_get_if("ospkg"));
|
---|
3966 | pb_distro_setuposrepo($pbos);
|
---|
3967 | pb_distro_installdeps(undef,$pbos,pb_distro_only_deps_needed($pbos,join(' ',split(/,/,$pkgforpb))));
|
---|
3968 | EOF
|
---|
3969 | } else {
|
---|
3970 | # Unknown install type
|
---|
3971 | die("Unknown install type $itype->{$ENV{'PBPROJ'}} for param pbinstalltype");
|
---|
3972 | }
|
---|
3973 | print SCRIPT << 'EOF';
|
---|
3974 | pb_system("pbdistrocheck",undef,"verbose");
|
---|
3975 | pb_system("pb -h",undef,"verbose");
|
---|
3976 | EOF
|
---|
3977 | if ($vtype eq "ve"){
|
---|
3978 | if ($vetype ne "docker") {
|
---|
3979 | print SCRIPT << 'EOF';
|
---|
3980 | # For VE we need to umount some FS at the end
|
---|
3981 |
|
---|
3982 | pb_system("umount /proc","Unmounting /proc","mayfail");
|
---|
3983 | EOF
|
---|
3984 | }
|
---|
3985 | print SCRIPT << 'EOF';
|
---|
3986 | # Create a basic network file if not already there
|
---|
3987 |
|
---|
3988 | my $nf="/etc/sysconfig/network";
|
---|
3989 | if ((! -f $nf) && ($pbos->{'type'} eq "rpm") && ($pbos->{'family'} ne "novell")) {
|
---|
3990 | open(NF,"> $nf") || die "Unable to create $nf";
|
---|
3991 | print NF "NETWORKING=yes\n";
|
---|
3992 | print NF "HOSTNAME=localhost\n";
|
---|
3993 | close(NF);
|
---|
3994 | }
|
---|
3995 | chmod 0755,$nf;
|
---|
3996 | EOF
|
---|
3997 | }
|
---|
3998 |
|
---|
3999 | # Adds pb_distro_get_context and all functions needed from ProjectBuilder::Distribution, Conf and Base
|
---|
4000 | foreach my $m ("ProjectBuilder/Base.pm","ProjectBuilder/Distribution.pm","ProjectBuilder/Conf.pm") {
|
---|
4001 | foreach my $d (@INC) {
|
---|
4002 | my $f = "$d/$m";
|
---|
4003 | if (-f "$f") {
|
---|
4004 | open(PBD,"$f") || die "Unable to open $f: $!";
|
---|
4005 | while (<PBD>) {
|
---|
4006 | next if (/^package/);
|
---|
4007 | next if (/^use Exporter/);
|
---|
4008 | next if (/^use ProjectBuilder::/);
|
---|
4009 | next if (/^our /);
|
---|
4010 | print SCRIPT $_;
|
---|
4011 | }
|
---|
4012 | close(PBD);
|
---|
4013 | # We just need the first one of each file wrt @INC - hopefully that's the right one.
|
---|
4014 | last;
|
---|
4015 | }
|
---|
4016 | }
|
---|
4017 | }
|
---|
4018 | # Use a fake pb_version_init version here
|
---|
4019 | print SCRIPT << "EOF";
|
---|
4020 | sub pb_version_init {
|
---|
4021 |
|
---|
4022 | return("$projectbuilderver","$projectbuilderrev");
|
---|
4023 | }
|
---|
4024 | 1;
|
---|
4025 | EOF
|
---|
4026 | close(SCRIPT);
|
---|
4027 | chmod 0755,"$pbscript{$v}";
|
---|
4028 |
|
---|
4029 | $pm->finish if (defined $pbparallel);
|
---|
4030 | }
|
---|
4031 | $pm->wait_all_children if (defined $pbparallel);
|
---|
4032 | die "Aborting, one or more of the children failed." if ((not $all_ok) && ($Global::pb_stop_on_error));
|
---|
4033 |
|
---|
4034 | # These build scripts need to be run as root and force stop of VM at end
|
---|
4035 | $ENV{'PBACCOUNT'} = "root";
|
---|
4036 |
|
---|
4037 | $sbx = "" if (not defined $sbx);
|
---|
4038 | # Launch the VM/VE/RM
|
---|
4039 | pb_log(2,"DEBUG: before parallel launch, pbscript hash is:".Dumper(%pbscript)."\n");
|
---|
4040 | pb_parallel_launchv(\%pbscript,$vtype,uc($vtype)."Script".$sbx,$pbstep,$pbimage,$pbforce,0,$pbsnap);
|
---|
4041 |
|
---|
4042 | return;
|
---|
4043 | }
|
---|
4044 |
|
---|
4045 | # Function to create a snapshot named 'pb' for VMs and a compressed tar for VEs
|
---|
4046 | sub pb_snap2v {
|
---|
4047 |
|
---|
4048 | my $vtype = shift;
|
---|
4049 | my $pbstep = 2;
|
---|
4050 |
|
---|
4051 | my ($vm,$all) = pb_get2v($vtype);
|
---|
4052 |
|
---|
4053 | # Script generated
|
---|
4054 | my $pbscript = "$ENV{'PBDESTDIR'}/snapv.$$";
|
---|
4055 |
|
---|
4056 | my ($pbac) = pb_conf_get($vtype."login");
|
---|
4057 |
|
---|
4058 | foreach my $v (@$vm) {
|
---|
4059 | if ($vtype eq "ve") {
|
---|
4060 | # Get distro context
|
---|
4061 | my $pbos = pb_distro_get_context($v);
|
---|
4062 | my ($vepath) = pb_conf_get("vepath");
|
---|
4063 |
|
---|
4064 | # Test if an existing snapshot exists and remove it if there is a VE
|
---|
4065 | if ((-f "$vepath->{$ENV{'PBPROJ'}}/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}.tar.gz") &&
|
---|
4066 | (! -d "$vepath->{$ENV{'PBPROJ'}}/$pbos->{'name'}/$pbos->{'version'}/$pbos->{'arch'}")) {
|
---|
4067 | pb_system("sudo rm -f $vepath->{$ENV{'PBPROJ'}}/$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}.tar.gz","Removing previous snapshot $pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}.tar.gz");
|
---|
4068 | }
|
---|
4069 | }
|
---|
4070 |
|
---|
4071 | # Prepare the script to be executed on the VM/VE
|
---|
4072 | open(SCRIPT,"> $pbscript") || die "Unable to create $pbscript";
|
---|
4073 | print SCRIPT << 'EOF';
|
---|
4074 | #!/bin/bash
|
---|
4075 | sleep 2
|
---|
4076 | EOF
|
---|
4077 | close(SCRIPT);
|
---|
4078 | chmod 0755,"$pbscript";
|
---|
4079 |
|
---|
4080 | # Force shutdown of VM/VE
|
---|
4081 | # Force snapshot of VM/VE
|
---|
4082 | pb_script2v($pbscript,$vtype,$pbstep,1,$v,1);
|
---|
4083 | }
|
---|
4084 | return;
|
---|
4085 | }
|
---|
4086 |
|
---|
4087 | # Function to update VMs/VEs/RMs with the latest distribution content
|
---|
4088 | sub pb_update2v {
|
---|
4089 |
|
---|
4090 | my $vtype = shift;
|
---|
4091 | my $pbstep = 2;
|
---|
4092 |
|
---|
4093 | my ($vm,$all) = pb_get2v($vtype);
|
---|
4094 |
|
---|
4095 | # Script generated
|
---|
4096 | my $pbscript = "$ENV{'PBDESTDIR'}/updatev.$$";
|
---|
4097 |
|
---|
4098 | my ($pbac) = pb_conf_get($vtype."login");
|
---|
4099 |
|
---|
4100 | foreach my $v (@$vm) {
|
---|
4101 | # Get distro context
|
---|
4102 | my $pbos = pb_distro_get_context($v);
|
---|
4103 |
|
---|
4104 | # Prepare the script to be executed on the VM/VE/RM
|
---|
4105 | # in $ENV{'PBDESTDIR'}/updatev
|
---|
4106 | open(SCRIPT,"> $pbscript") || die "Unable to create $pbscript";
|
---|
4107 |
|
---|
4108 | print SCRIPT << 'EOF';
|
---|
4109 | #!/bin/bash
|
---|
4110 | sleep 2
|
---|
4111 | EOF
|
---|
4112 | # VE needs a good /proc
|
---|
4113 | $vetype = pb_ve_get_type($vetype);
|
---|
4114 | if (($vtype eq "ve") && ($vetype ne "docker")) {
|
---|
4115 | print SCRIPT "sudo /bin/mount -t proc /proc /proc\n";
|
---|
4116 | }
|
---|
4117 | print SCRIPT "$pbos->{'update'}\n";
|
---|
4118 | if (($vtype eq "ve") && ($vetype ne "docker")) {
|
---|
4119 | print SCRIPT "sudo /bin/umount /proc\n";
|
---|
4120 | }
|
---|
4121 | close(SCRIPT);
|
---|
4122 | chmod 0755,"$pbscript";
|
---|
4123 |
|
---|
4124 | # Force shutdown of VM except
|
---|
4125 | pb_script2v($pbscript,$vtype,$pbstep,1,$v);
|
---|
4126 | }
|
---|
4127 | return;
|
---|
4128 | }
|
---|
4129 |
|
---|
4130 | sub pb_announce {
|
---|
4131 |
|
---|
4132 | my $antype = shift;
|
---|
4133 |
|
---|
4134 | # Get all required parameters
|
---|
4135 | my ($pbpackager,$pbrepo) = pb_conf_get("pbpackager","pbrepo");
|
---|
4136 | my ($pkgv, $pkgt, $testver) = pb_conf_get_if("pkgver","pkgtag","testver");
|
---|
4137 | if (((not defined $testver) || (not defined $testver->{$ENV{'PBPROJ'}}) || ($testver->{$ENV{'PBPROJ'}} !~ /true/i)) && ($antype eq "Clean")) {
|
---|
4138 | # We just clean for test versions
|
---|
4139 | pb_log(0,"Unable to clean SSH repository for non testver version\n");
|
---|
4140 | return;
|
---|
4141 | }
|
---|
4142 | my $pkg = pb_cms_get_pkg($defpkgdir,$extpkgdir);
|
---|
4143 | my @pkgs = @$pkg;
|
---|
4144 | my %pkgs;
|
---|
4145 | my $first = 0;
|
---|
4146 |
|
---|
4147 | # Get all distros concerned
|
---|
4148 | my $pbos = pb_distro_get_context();
|
---|
4149 | my $distrolist = pb_get_distros($pbos,undef);
|
---|
4150 | my %dl;
|
---|
4151 | my %theorlist;
|
---|
4152 | my %archlist;
|
---|
4153 | foreach my $d (split(/,/,$distrolist)) {
|
---|
4154 | my ($d1,$d2,$d3) = split(/-/,$d);
|
---|
4155 | $dl{$d1}++;
|
---|
4156 | }
|
---|
4157 |
|
---|
4158 | # Command to find packages on repo
|
---|
4159 | my $findstr = "find ".join(" ",keys %dl)." ";
|
---|
4160 | my $srcstr = "";
|
---|
4161 | # Generated announce files
|
---|
4162 | my @files;
|
---|
4163 |
|
---|
4164 | foreach my $pbpkg (@pkgs) {
|
---|
4165 | if ($first != 0) {
|
---|
4166 | $findstr .= "-o ";
|
---|
4167 | }
|
---|
4168 | $first++;
|
---|
4169 | if ((defined $pkgv) && (defined $pkgv->{$pbpkg})) {
|
---|
4170 | $pbver = $pkgv->{$pbpkg};
|
---|
4171 | } else {
|
---|
4172 | $pbver = $ENV{'PBPROJVER'};
|
---|
4173 | }
|
---|
4174 | if ((defined $pkgt) && (defined $pkgt->{$pbpkg})) {
|
---|
4175 | $pbtag = $pkgt->{$pbpkg};
|
---|
4176 | } else {
|
---|
4177 | $pbtag = $ENV{'PBPROJTAG'};
|
---|
4178 | }
|
---|
4179 |
|
---|
4180 | # TODO: use virtual/real names here now
|
---|
4181 | my $pbrealpkg = $pbpkg;
|
---|
4182 | my $pbrealpkgrpm = pb_cms_get_real_pkg($pbpkg,"rpm");
|
---|
4183 | my $pbrealpkgdeb = pb_cms_get_real_pkg($pbpkg,"deb");
|
---|
4184 | if ($antype eq "Clean") {
|
---|
4185 | # We only clean test versions anyway
|
---|
4186 | $pbtag = "0";
|
---|
4187 | my $nver = $pbver;
|
---|
4188 | my $ntag = "[2-9][0-9][0-9][0-9][0-9][0-9][0-9][0-9][0-9][0-9][0-9][0-9]";
|
---|
4189 | $pbver .= $ntag;
|
---|
4190 | $findstr .= "-name \'$pbrealpkgrpm-$pbver-$pbtag\.*.rpm\' -o -name \'$pbrealpkgrpm-debug-$pbver-$pbtag\.*.rpm\' -o -name \'$pbrealpkgdeb"."_$pbver-$pbtag"."_*\.deb\' -o -name \'$pbrealpkgdeb"."_$pbver-$pbtag.dsc\' -o -name \'$pbrealpkgdeb"."_$pbver-$pbtag.tar.gz\' -o -name \'$pbrealpkg-$nver"."_p$ntag\.ebuild\' -o -name \'$pbrealpkg-$pbver-$pbtag*\.pkg\.gz\' -o -name \'$pbrealpkg-$pbver-$pbtag*\.sd\' ";
|
---|
4191 | $srcstr .= "src/$pbrealpkg-$pbver.tar.gz src/$pbrealpkg-$pbver.pbconf.tar.gz ";
|
---|
4192 | } else {
|
---|
4193 | # "Announce" and "Check"
|
---|
4194 | my @date=pb_get_date();
|
---|
4195 | # the matching is only done on packages made the same day for test version. Hopefully this is enough
|
---|
4196 | my $nver;
|
---|
4197 | if ((defined $testver) && (defined $testver->{$ENV{'PBPROJ'}}) && ($testver->{$ENV{'PBPROJ'}} =~ /true/i)) {
|
---|
4198 | $pbtag = "0";
|
---|
4199 | my $ntag .= strftime("%Y%m%d*", @date);
|
---|
4200 | $nver = $pbver."_p$ntag*";
|
---|
4201 | $pbver .= "$ntag*";
|
---|
4202 | } else {
|
---|
4203 | $nver = $pbver."-r$pbtag";
|
---|
4204 | }
|
---|
4205 | $findstr .= "-name \'$pbrealpkgrpm-$pbver-$pbtag\.*.rpm\' -o -name \'$pbrealpkgdeb"."_$pbver-$pbtag"."_*\.deb\' -o -name \'$pbrealpkg-$nver\.ebuild\' -o -name \'$pbrealpkg-$pbver\.pkg\.gz\' -o -name \'$pbrealpkg-$pbver\.sd\' ";
|
---|
4206 | }
|
---|
4207 |
|
---|
4208 | if ($antype eq "Announce") {
|
---|
4209 | my $chglog;
|
---|
4210 |
|
---|
4211 | pb_cms_init($pbinit);
|
---|
4212 | # Get project info on log file and generate tmp files used later on
|
---|
4213 | $chglog = "$ENV{'PBROOTDIR'}/$pbpkg/pbcl";
|
---|
4214 | $chglog = "$ENV{'PBROOTDIR'}/pbcl" if (! -f $chglog);
|
---|
4215 | $chglog = undef if (! -f $chglog);
|
---|
4216 |
|
---|
4217 | open(OUT,"> $ENV{'PBTMP'}/$pbpkg.ann") || die "Unable to create $ENV{'PBTMP'}/$pbpkg.ann: $!";
|
---|
4218 | my $pb;
|
---|
4219 | $pb->{'realpkg'} = $pbrealpkg;
|
---|
4220 | $pb->{'ver'} = $pbver;
|
---|
4221 | $pb->{'tag'} = $pbtag;
|
---|
4222 | $pb->{'date'} = $pbdate;
|
---|
4223 | $pb->{'extdir'} = pb_get_extdir();
|
---|
4224 | $pb->{'chglog'} = $chglog;
|
---|
4225 | $pb->{'packager'} = $pbpackager;
|
---|
4226 | $pb->{'proj'} = $ENV{'PBPROJ'};
|
---|
4227 | $pb->{'repo'} = $pbrepo;
|
---|
4228 | $pb->{'pbos'}->{'type'} = "announce";
|
---|
4229 | $pb->{'pbos'}->{'suffix'} = "none";
|
---|
4230 | pb_changelog($pb,\*OUT,"yes");
|
---|
4231 | close(OUT);
|
---|
4232 | push(@files,"$ENV{'PBTMP'}/$pbpkg.ann");
|
---|
4233 | } elsif ($antype eq "Check") {
|
---|
4234 | # For the check we also build the theoritical complete list we should get
|
---|
4235 | foreach my $d (split(/,/,$distrolist)) {
|
---|
4236 | $pbos = pb_distro_get_context($d);
|
---|
4237 | if ($pbos->{'type'} eq "rpm") {
|
---|
4238 | $theorlist{"$pbos->{'name'}/$pbos->{'version'}/$pbos->{'arch'}/$pbrealpkgrpm-$pbver-$pbtag$pbos->{'suffix'}"} = 0;
|
---|
4239 | } elsif ($pbos->{'type'} eq "deb") {
|
---|
4240 | $theorlist{"$pbos->{'name'}/$pbos->{'version'}/$pbrealpkgdeb"."_$pbver-$pbtag"} = 0;
|
---|
4241 | # TODO are we always using the last arch ?
|
---|
4242 | $archlist{"$pbos->{'name'}/$pbos->{'version'}/$pbrealpkgdeb"."_$pbver-$pbtag"} = "$pbos->{'arch'}";
|
---|
4243 | } elsif ($pbos->{'type'} eq "ebuild") {
|
---|
4244 | my $prefix = "-r";
|
---|
4245 | $prefix = "_p" if ((defined $testver) && (defined $testver->{$ENV{'PBPROJ'}}) && ($testver->{$ENV{'PBPROJ'}} =~ /true/i));
|
---|
4246 | $theorlist{"$pbos->{'name'}/$pbos->{'version'}/$pbrealpkg-$pbver$prefix$pbtag.ebuild"} = 0;
|
---|
4247 | $archlist{"$pbos->{'name'}/$pbos->{'version'}/$pbrealpkg-$pbver$prefix$pbtag.ebuild"} = "$pbos->{'arch'}";
|
---|
4248 | } elsif ($pbos->{'type'} eq "pkg") {
|
---|
4249 | $theorlist{"$pbos->{'name'}/$pbos->{'version'}/$pbos->{'arch'}/$pbrealpkg-$pbver-$pbtag.pkg.gz"} = 0;
|
---|
4250 | } else {
|
---|
4251 | pb_log(1,"No theoritical list possible for type $pbos->{'type'}\n");
|
---|
4252 | }
|
---|
4253 | }
|
---|
4254 | }
|
---|
4255 | pb_log(2,"theorlist initial: ".Dumper(%theorlist)."\n");
|
---|
4256 | pb_log(2,"archlist: ".Dumper(%archlist)."\n");
|
---|
4257 | }
|
---|
4258 | if ($antype eq "Announce") {
|
---|
4259 | $findstr .= " | grep -Ev \'src.rpm\'";
|
---|
4260 | } elsif ($antype eq "Clean") {
|
---|
4261 | $findstr .= " | xargs rm -f -v $srcstr ";
|
---|
4262 | }
|
---|
4263 |
|
---|
4264 | # Prepare the command to run and execute it
|
---|
4265 | open(PBS,"> $ENV{'PBTMP'}/pbscript") || die "Unable to create $ENV{'PBTMP'}/pbscript";
|
---|
4266 | print PBS "#!/bin/bash\n";
|
---|
4267 | print PBS "set -x\n" if ($pbdebug gt 1);
|
---|
4268 | print PBS "$findstr | sort 2> /dev/null\n";
|
---|
4269 | close(PBS);
|
---|
4270 | chmod 0755,"$ENV{'PBTMP'}/pbscript";
|
---|
4271 | pb_send2target("Announce","$ENV{'PBTMP'}/pbscript");
|
---|
4272 |
|
---|
4273 | my $sl = "Project $ENV{'PBPROJ'} version $ENV{'PBPROJVER'} is now available";
|
---|
4274 | if ($antype eq "Announce") {
|
---|
4275 | # Get subject line
|
---|
4276 | pb_log(0,"Please enter the title of your announce\n");
|
---|
4277 | pb_log(0,"(By default: $sl)\n");
|
---|
4278 | my $sl2 = <STDIN>;
|
---|
4279 | $sl = $sl2 if ($sl2 !~ /^$/);
|
---|
4280 |
|
---|
4281 | # Prepare a template of announce
|
---|
4282 | open(ANN,"> $ENV{'PBTMP'}/announce.html") || die "Unable to create $ENV{'PBTMP'}/announce.html: $!";
|
---|
4283 | print ANN << "EOF";
|
---|
4284 | $sl</p>
|
---|
4285 |
|
---|
4286 | <p>The project team is happy to announce the availability of a newest version of $ENV{'PBPROJ'} $ENV{'PBPROJVER'}. Enjoy it as usual!</p>
|
---|
4287 | <p>
|
---|
4288 | Now available at <a href="$pbrepo->{$ENV{'PBPROJ'}}">$pbrepo->{$ENV{'PBPROJ'}}</a>
|
---|
4289 | </p>
|
---|
4290 | <p>
|
---|
4291 | EOF
|
---|
4292 | }
|
---|
4293 |
|
---|
4294 | open(LOG,"$ENV{'PBTMP'}/system.$$.log") || die "Unable to read $ENV{'PBTMP'}/system.$$.log: $!";
|
---|
4295 | if ($antype eq "Announce") {
|
---|
4296 | my $col = 2;
|
---|
4297 | my $i = 1;
|
---|
4298 | print ANN << 'EOF';
|
---|
4299 | <TABLE WIDTH="100%" CELLPADDING="0" CELLSPACING="0" BORDER="0">
|
---|
4300 | <TR>
|
---|
4301 | EOF
|
---|
4302 | while (<LOG>) {
|
---|
4303 | print ANN "<TD><A HREF=\"$pbrepo->{$ENV{'PBPROJ'}}/$_\">$_</A></TD>";
|
---|
4304 | $i++;
|
---|
4305 | if ($i > $col) {
|
---|
4306 | print ANN "</TR>\n<TR>";
|
---|
4307 | $i = 1;
|
---|
4308 | }
|
---|
4309 | }
|
---|
4310 | } elsif ($antype eq "Clean") {
|
---|
4311 | while (<LOG>) {
|
---|
4312 | # skip errors
|
---|
4313 | next if ($_ !~ /^removed /);
|
---|
4314 | pb_log(0,"$_");
|
---|
4315 | }
|
---|
4316 | } else {
|
---|
4317 | # In "Check" mode we need to compare the 2 lists (real and theoritical)
|
---|
4318 | while (<LOG>) {
|
---|
4319 | # Get package name and remove what is in extra for the theoritical list (arch at the end)
|
---|
4320 | chomp();
|
---|
4321 | # skip find errors
|
---|
4322 | next if (/^find:/);
|
---|
4323 | my $p = $_;
|
---|
4324 | $p =~ s/\.(i[3456]86|x86_64|noarch|src)\.rpm$//;
|
---|
4325 | $p =~ s/_(i[3456]86|amd64|all)\.deb$//;
|
---|
4326 | $p =~ s/(-0\.[0-9]{8})[0-9]{6}/$1*/ if ((defined $testver) && (defined $testver->{$ENV{'PBPROJ'}}) && ($testver->{$ENV{'PBPROJ'}} =~ /true/i));
|
---|
4327 | $p =~ s/(-r|_p[0-9]+)\.ebuild/$1*/ if ((defined $testver) && (defined $testver->{$ENV{'PBPROJ'}}) && ($testver->{$ENV{'PBPROJ'}} =~ /true/i));
|
---|
4328 | $theorlist{$p} = -2 if (not defined $theorlist{$p});
|
---|
4329 | $theorlist{$p} = $theorlist{$p} + 1;
|
---|
4330 | }
|
---|
4331 | pb_log(2,"theorlist after check: ".Dumper(%theorlist)."\n");
|
---|
4332 | }
|
---|
4333 | close(LOG);
|
---|
4334 |
|
---|
4335 | # Nothing more for the cleanssh case
|
---|
4336 | return if ($antype eq "Clean");
|
---|
4337 |
|
---|
4338 | if ($antype eq "Check") {
|
---|
4339 | my ($chkex) = pb_conf_get_if("checkexclude");
|
---|
4340 | my $vmbuildlist = "";
|
---|
4341 | my $vebuildlist = "";
|
---|
4342 | my $rmbuildlist = "";
|
---|
4343 | my @pt = pb_conf_get_if("vmlist","velist","rmlist");
|
---|
4344 | foreach my $t (sort keys %theorlist) {
|
---|
4345 | if (defined $theorlist{$t} and $theorlist{$t} >= 1) {
|
---|
4346 | pb_log(1,"Packages found for $t\n");
|
---|
4347 | } elsif (defined $theorlist{$t} and $theorlist{$t} < 0) {
|
---|
4348 | pb_log(0,"Extra Package found for $t\n");
|
---|
4349 | } else {
|
---|
4350 | pb_log(2,"Analyzing $t\n");
|
---|
4351 | my ($os,$ver,$arch,$package) = split(/\//,$t);
|
---|
4352 | # Some distro have no arch subdir
|
---|
4353 | if (not defined $package) {
|
---|
4354 | $package = $arch;
|
---|
4355 | # TODO: If both arch have failed, we just make the last one
|
---|
4356 | $arch = $archlist{$t} if (defined $archlist{$t});
|
---|
4357 | }
|
---|
4358 | my $pbos = pb_distro_get_context("$os-$ver-$arch");
|
---|
4359 | my $pkgn = $package;
|
---|
4360 | if ($pbos->{'type'} ne "deb") {
|
---|
4361 | # package name is more easily found from the end for non deb
|
---|
4362 | # as '-' is the separator, but it can also be used in names
|
---|
4363 | $pkgn = reverse($package);
|
---|
4364 | # search the second '-' and isolate the now last part which is the full name
|
---|
4365 | $pkgn =~ s/([^-]+)-([^-]+)-([\S])+$/$3/;
|
---|
4366 | } else {
|
---|
4367 | $pkgn =~ s/([^_]+)_([\S])+$/$2/;
|
---|
4368 | }
|
---|
4369 | my $found = 0;
|
---|
4370 | # Handle the exclusion of OSes
|
---|
4371 | my $excl = "";
|
---|
4372 | $excl .= $chkex->{$pkgn} if (defined $chkex->{$pkgn});
|
---|
4373 | $excl .= $chkex->{"all"} if (defined $chkex->{"all"});
|
---|
4374 |
|
---|
4375 | pb_log(2,"Searching in excl $excl\n");
|
---|
4376 | foreach my $ex (split(/,/,$excl)) {
|
---|
4377 | $found = 1 if ("$os-$ver-$arch" =~ /^$ex/);
|
---|
4378 | }
|
---|
4379 | # Skip as excluded
|
---|
4380 | next if ($found == 1);
|
---|
4381 | pb_log(0,"Package NOT found for $t\n");
|
---|
4382 | # Avoid duplicates in list
|
---|
4383 | next if ($vmbuildlist =~ /$os-$ver-$arch/);
|
---|
4384 | next if ($vebuildlist =~ /$os-$ver-$arch/);
|
---|
4385 | next if ($rmbuildlist =~ /$os-$ver-$arch/);
|
---|
4386 | # check with which method we need to build
|
---|
4387 | if ((defined $pt[0]->{$ENV{'PBPROJ'}}) and ($pt[0]->{$ENV{'PBPROJ'}} =~ /$os-$ver-$arch/)) {
|
---|
4388 | $vmbuildlist = "$os-$ver-$arch" if ($vmbuildlist eq "");
|
---|
4389 | $vmbuildlist .= ",$os-$ver-$arch" if ($vmbuildlist !~ /$os-$ver-$arch/);
|
---|
4390 | next;
|
---|
4391 | }
|
---|
4392 | if ((defined $pt[1]->{$ENV{'PBPROJ'}}) and ($pt[1]->{$ENV{'PBPROJ'}} =~ /$os-$ver-$arch/)) {
|
---|
4393 | $vebuildlist = "$os-$ver-$arch" if ($vebuildlist eq "");
|
---|
4394 | $vebuildlist .= ",$os-$ver-$arch" if ($vebuildlist !~ /$os-$ver-$arch/);
|
---|
4395 | next;
|
---|
4396 | }
|
---|
4397 | if ((defined $pt[2]->{$ENV{'PBPROJ'}}) and ($pt[2]->{$ENV{'PBPROJ'}} =~ /$os-$ver-$arch/)) {
|
---|
4398 | $rmbuildlist = "$os-$ver-$arch" if ($rmbuildlist eq "");
|
---|
4399 | $rmbuildlist .= ",$os-$ver-$arch" if ($rmbuildlist !~ /$os-$ver-$arch/);
|
---|
4400 | }
|
---|
4401 | }
|
---|
4402 | }
|
---|
4403 | # If we want to rebuild automatically, let's do it
|
---|
4404 | if (defined $opts{'rebuild'}) {
|
---|
4405 | # SandBox or CMS
|
---|
4406 | pb_log(0,"Rebuilding from SandBox\n");
|
---|
4407 | pb_log(0,"for VMs: $vmbuildlist\n") if ($vmbuildlist ne "");
|
---|
4408 | pb_log(0,"for VEs: $vebuildlist\n") if ($vebuildlist ne "");
|
---|
4409 | pb_log(0,"for RMs: $rmbuildlist\n") if ($rmbuildlist ne "");
|
---|
4410 | pb_cms2build("SandBox");
|
---|
4411 | # Which mode
|
---|
4412 | $ENV{'PBV'} = $vmbuildlist;
|
---|
4413 | pb_build2v("vm","build") if ($vmbuildlist ne "");
|
---|
4414 | $ENV{'PBV'} = $vebuildlist;
|
---|
4415 | pb_build2v("ve","build") if ($vebuildlist ne "");
|
---|
4416 | $ENV{'PBV'} = $rmbuildlist;
|
---|
4417 | pb_build2v("rm","build") if ($rmbuildlist ne "");
|
---|
4418 | }
|
---|
4419 | # For the check part this is now finished
|
---|
4420 | return;
|
---|
4421 | }
|
---|
4422 |
|
---|
4423 | print ANN << "EOF";
|
---|
4424 | </TR>
|
---|
4425 | </TABLE>
|
---|
4426 | </p>
|
---|
4427 |
|
---|
4428 | <p>As usual source packages are also available in the same directory.</p>
|
---|
4429 |
|
---|
4430 | <p>
|
---|
4431 | Changes are :
|
---|
4432 | </p>
|
---|
4433 | <p>
|
---|
4434 | EOF
|
---|
4435 | # Get each package changelog content
|
---|
4436 | foreach my $f (sort(@files)) {
|
---|
4437 | open(IN,"$f") || die "Unable to read $f:$!";
|
---|
4438 | while (<IN>) {
|
---|
4439 | print ANN $_;
|
---|
4440 | }
|
---|
4441 | close(IN);
|
---|
4442 | print ANN "</p><p>\n";
|
---|
4443 | }
|
---|
4444 | print ANN "</p>\n";
|
---|
4445 | close(ANN);
|
---|
4446 |
|
---|
4447 | # Allow for modification
|
---|
4448 | my $editor = "vi";
|
---|
4449 | $editor = $ENV{'EDITOR'} if (defined $ENV{'EDITOR'});
|
---|
4450 | pb_system("$editor $ENV{'PBTMP'}/announce.html","Allowing modification of the announce","noredir");
|
---|
4451 |
|
---|
4452 | # Store it in DB for external usage (Web pages generation)
|
---|
4453 | my $dbh = pb_connect_db();
|
---|
4454 |
|
---|
4455 | # To read whole file
|
---|
4456 | local $/;
|
---|
4457 | open(ANN,"$ENV{'PBTMP'}/announce.html") || die "Unable to read $ENV{'PBTMP'}/announce.html: $!";
|
---|
4458 | my $announce = <ANN>;
|
---|
4459 | close(ANN);
|
---|
4460 |
|
---|
4461 | pb_log(2,"INSERT INTO announces VALUES (NULL, $pbdate, $announce)");
|
---|
4462 | my $sth = $dbh->prepare(qq{INSERT INTO announces VALUES (NULL,?,?)})
|
---|
4463 | || die "Unable to insert into db";
|
---|
4464 | $sth->execute($pbdate, $announce);
|
---|
4465 | $sth->finish();
|
---|
4466 | $dbh->disconnect;
|
---|
4467 |
|
---|
4468 | # Then deliver it on the Web
|
---|
4469 | # $TOOLHOME/livwww www
|
---|
4470 |
|
---|
4471 | # Mail it to project's ML
|
---|
4472 | open(ML,"| w3m -dump -T text/html > $ENV{'PBTMP'}/announce.txt") || die "Unable to create $ENV{'PBTMP'}/announce.txt: $!";
|
---|
4473 | print ML << 'EOF';
|
---|
4474 | <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/x html1/DTD/xhtml1-strict.dtd">
|
---|
4475 |
|
---|
4476 | <html xmlns="http://www.w3.org/1999/xhtml" dir="ltr" xml:lang="en" lang="en">
|
---|
4477 | <head>
|
---|
4478 | </head>
|
---|
4479 | <body>
|
---|
4480 | <p>
|
---|
4481 | EOF
|
---|
4482 | open(ANN,"$ENV{'PBTMP'}/announce.html") || die "Unable to read $ENV{'PBTMP'}/announce.html: $!";
|
---|
4483 | while(<ANN>) {
|
---|
4484 | print ML $_;
|
---|
4485 | }
|
---|
4486 | print ML << 'EOF';
|
---|
4487 | </body>
|
---|
4488 | </html>
|
---|
4489 | EOF
|
---|
4490 | close(ML);
|
---|
4491 |
|
---|
4492 | # To read whole file
|
---|
4493 | local $/;
|
---|
4494 | open(ANN,"$ENV{'PBTMP'}/announce.txt") || die "Unable to read $ENV{'PBTMP'}/announce.txt: $!";
|
---|
4495 | my $msg = <ANN>;
|
---|
4496 | close(ANN);
|
---|
4497 |
|
---|
4498 | # Preparation of headers
|
---|
4499 | eval
|
---|
4500 | {
|
---|
4501 | require Mail::Sendmail;
|
---|
4502 | Mail::Sendmail->import();
|
---|
4503 | };
|
---|
4504 | if ($@) {
|
---|
4505 | # Mail::Sendmail not found not sending mail !
|
---|
4506 | pb_log(0,"No Mail::Sendmail module found so not sending any mail !\n");
|
---|
4507 | } else {
|
---|
4508 | my ($pbml,$pbsmtp) = pb_conf_get("pbml","pbsmtp");
|
---|
4509 | my %mail = (
|
---|
4510 | To => $pbml->{$ENV{'PBPROJ'}},
|
---|
4511 | From => $pbpackager->{$ENV{'PBPROJ'}},
|
---|
4512 | Smtp => $pbsmtp->{$ENV{'PBPROJ'}},
|
---|
4513 | Body => $msg,
|
---|
4514 | Subject => "[ANNOUNCE] $sl",
|
---|
4515 | );
|
---|
4516 |
|
---|
4517 | # Send mail
|
---|
4518 | if (! sendmail(%mail)) {
|
---|
4519 | if ((defined $Mail::Sendmail::error) and (defined $Mail::Sendmail::log)) {
|
---|
4520 | die "Unable to send mail ($Mail::Sendmail::error): $Mail::Sendmail::log";
|
---|
4521 | }
|
---|
4522 | }
|
---|
4523 | }
|
---|
4524 | }
|
---|
4525 |
|
---|
4526 | #
|
---|
4527 | # Creates a set of HTML file containing the news for the project
|
---|
4528 | # based on what has been generated by the pb_announce function
|
---|
4529 | #
|
---|
4530 | sub pb_web_news2html {
|
---|
4531 |
|
---|
4532 | my $dest = shift || $ENV{'PBTMP'};
|
---|
4533 |
|
---|
4534 | # Get all required parameters
|
---|
4535 | my ($pkgv, $pkgt) = pb_conf_get_if("pkgver","pkgtag");
|
---|
4536 |
|
---|
4537 |
|
---|
4538 | my $dbh = pb_connect_db();
|
---|
4539 | # For date handling
|
---|
4540 | $ENV{LANGUAGE}="C";
|
---|
4541 | my $firstjan = strftime("%Y-%m-%d", 0, 0, 0, 1, 0, localtime->year(), 0, 0, -1);
|
---|
4542 | my $oldfirst = strftime("%Y-%m-%d", 0, 0, 0, 1, 0, localtime->year()-1, 0, 0, -1);
|
---|
4543 | pb_log(2,"firstjan: $firstjan, oldfirst: $oldfirst, pbdate:$pbdate\n");
|
---|
4544 | my $all = $dbh->selectall_arrayref("SELECT id,date,announce FROM announces ORDER BY date DESC");
|
---|
4545 | my %news;
|
---|
4546 | $news{"cy"} = ""; # current year's news
|
---|
4547 | $news{"ly"} = ""; # last year news
|
---|
4548 | $news{"py"} = ""; # previous years news
|
---|
4549 | $news{"fp"} = ""; # first page news
|
---|
4550 | my $cpt = 4; # how many news for first page
|
---|
4551 | # Extract info from DB
|
---|
4552 | foreach my $row (@$all) {
|
---|
4553 | my ($id, $date, $announce) = @$row;
|
---|
4554 | $news{"cy"} = $news{"cy"}."<p><B>$date</B> $announce\n" if ((($date cmp $pbdate) le 0) && (($firstjan cmp $date) le 0));
|
---|
4555 | $news{"ly"} = $news{"ly"}."<p><B>$date</B> $announce\n" if ((($date cmp $firstjan) le 0) && (($oldfirst cmp $date) le 0));
|
---|
4556 | $news{"py"} = $news{"py"}."<p><B>$date</B> $announce\n" if (($date cmp $oldfirst) le 0);
|
---|
4557 | $news{"fp"} = $news{"fp"}."<p><B>$date</B> $announce\n" if ($cpt > 0);
|
---|
4558 | $cpt--;
|
---|
4559 | }
|
---|
4560 | pb_log(1,"news{fp}: ".$news{"fp"}."\n");
|
---|
4561 | $dbh->disconnect;
|
---|
4562 |
|
---|
4563 | # Generate the HTML content
|
---|
4564 | foreach my $pref (keys %news) {
|
---|
4565 | open(NEWS,"> $dest/pb_web_$pref"."news.html") || die "Unable to create $dest/pb_web_$pref"."news.html: $!";
|
---|
4566 | print NEWS "$news{$pref}";
|
---|
4567 | close(NEWS);
|
---|
4568 | }
|
---|
4569 | }
|
---|
4570 |
|
---|
4571 | sub pb_connect_db {
|
---|
4572 |
|
---|
4573 | # Store in DB for external usage (Web pages generation)
|
---|
4574 | my $db = "$ENV{'PBCONFDIR'}/announces3.sql";
|
---|
4575 |
|
---|
4576 | my $precmd = "";
|
---|
4577 | if (! -f $db) {
|
---|
4578 | $precmd = "CREATE TABLE announces (id INTEGER PRIMARY KEY AUTOINCREMENT, date DATE, announce VARCHAR[65535])";
|
---|
4579 | # we should add the $db to the VCS as well here
|
---|
4580 | }
|
---|
4581 |
|
---|
4582 | my $dbh = DBI->connect("dbi:SQLite:dbname=$db","","",
|
---|
4583 | { RaiseError => 1, AutoCommit => 1 })
|
---|
4584 | || die "Unable to connect to $db";
|
---|
4585 |
|
---|
4586 | if ($precmd ne "") {
|
---|
4587 | my $sth = $dbh->prepare(qq{$precmd})
|
---|
4588 | || die "Unable to create table into $db";
|
---|
4589 | $sth->execute();
|
---|
4590 | }
|
---|
4591 |
|
---|
4592 | return($dbh);
|
---|
4593 | }
|
---|
4594 |
|
---|
4595 |
|
---|
4596 | # Return the SSH key file to use
|
---|
4597 | # Potentially create it if needed
|
---|
4598 |
|
---|
4599 | sub pb_ssh_get {
|
---|
4600 |
|
---|
4601 | my $create = shift || 0; # Do not create keys by default. In that case prefer native keys, not pb's ones
|
---|
4602 | my $target = shift || "localhost";
|
---|
4603 | my $nport = shift || "22";
|
---|
4604 |
|
---|
4605 | my ($pbagent) = pb_conf_get_if("pbusesshagent");
|
---|
4606 | # use ssh-agent if asked so.
|
---|
4607 | return(undef) if (($create == 0) && (defined $pbagent->{$ENV{'PBPROJ'}}) && ($pbagent->{$ENV{'PBPROJ'}} =~ /true/io));
|
---|
4608 |
|
---|
4609 | # Check the SSH environment
|
---|
4610 | my $keyfile = undef;
|
---|
4611 |
|
---|
4612 | # Check whether ecdsa is accepted by the remote host
|
---|
4613 | my $ecdsa = 0;
|
---|
4614 | my $dsa = 0;
|
---|
4615 | open(SCAN,"ssh-keyscan -p $nport -t ecdsa $target 2>/dev/null |") || pb_log(0,"Unable to check ECDSA support for $target");
|
---|
4616 | while(<SCAN>) {
|
---|
4617 | $ecdsa = 1 if ($_ =~ /ecdsa-/);
|
---|
4618 | pb_log(2,"Found ECDSA support on $target ($nport)\n");
|
---|
4619 | }
|
---|
4620 | close(SCAN);
|
---|
4621 |
|
---|
4622 | open(SCAN,"ssh-keyscan -p $nport -t dsa $target 2>/dev/null |") || pb_log(0,"Unable to check DSA support for $target");
|
---|
4623 | while(<SCAN>) {
|
---|
4624 | $dsa = 1 if ($_ =~ /ssh-dss/);
|
---|
4625 | pb_log(2,"Found DSA support on $target ($nport)\n");
|
---|
4626 | }
|
---|
4627 | close(SCAN);
|
---|
4628 |
|
---|
4629 | # We have specific keys by default - create them if needed
|
---|
4630 | if (!(-e "$ENV{'HOME'}/.ssh/pb_ecdsa") && ($create eq 1)) {
|
---|
4631 | my $res = pb_system("ssh-keygen -q -b 1024 -N '' -f $ENV{'HOME'}/.ssh/pb_ecdsa -t ecdsa","Generating ECDSA SSH keys for pb");
|
---|
4632 | if (($res != 0) && !(-e "$ENV{'HOME'}/.ssh/pb_dsa") && ($create eq 1)) {
|
---|
4633 | # ecdsa may not be supported if old ssh
|
---|
4634 | pb_system("ssh-keygen -q -b 1024 -N '' -f $ENV{'HOME'}/.ssh/pb_dsa -t dsa","Generating DSA SSH keys for pb");
|
---|
4635 | }
|
---|
4636 | }
|
---|
4637 |
|
---|
4638 | #$keyfile = "$ENV{'HOME'}/.ssh/id_rsa" if (-s "$ENV{'HOME'}/.ssh/id_rsa");
|
---|
4639 | # We still favoud DSA and fall back to ECDSA if not available
|
---|
4640 | if ($dsa != 0) {
|
---|
4641 | # And we favour a specific key over a generic one except if in delivery to ftp server
|
---|
4642 | $keyfile = "$ENV{'HOME'}/.ssh/id_dsa" if ((-s "$ENV{'HOME'}/.ssh/id_dsa") && ($create eq 0));
|
---|
4643 | $keyfile = "$ENV{'HOME'}/.ssh/pb_dsa" if (-s "$ENV{'HOME'}/.ssh/pb_dsa");
|
---|
4644 | } else {
|
---|
4645 | # And we favour a specific key over a generic one except if in delivery to ftp server
|
---|
4646 | $keyfile = "$ENV{'HOME'}/.ssh/id_ecdsa" if ((-s "$ENV{'HOME'}/.ssh/id_ecdsa") && ($ecdsa != 0) && ($create eq 0));
|
---|
4647 | $keyfile = "$ENV{'HOME'}/.ssh/pb_ecdsa" if ((-s "$ENV{'HOME'}/.ssh/pb_ecdsa") && ($ecdsa != 0));
|
---|
4648 | }
|
---|
4649 | if (defined $keyfile) {
|
---|
4650 | pb_log(2,"Using ssh key file $keyfile\n");
|
---|
4651 | } else {
|
---|
4652 | pb_log(0, "Unable to find your public ssh key under $ENV{'HOME'}/.ssh\n");
|
---|
4653 | }
|
---|
4654 | return($keyfile);
|
---|
4655 | }
|
---|
4656 |
|
---|
4657 |
|
---|
4658 | # Returns the pid of a running VM command using a specific VM file, and the port used if the VM exists
|
---|
4659 | sub pb_check_ps {
|
---|
4660 | my $vmcmd = shift;
|
---|
4661 | my $vmm = shift;
|
---|
4662 | my $vexist = 0; # FALSE by default
|
---|
4663 | my $vmport = undef; # NONE by default
|
---|
4664 |
|
---|
4665 | open(PS, "ps auxhww|") || die "Unable to call ps";
|
---|
4666 | while (<PS>) {
|
---|
4667 | next if (! /$vmcmd/);
|
---|
4668 | next if (! /$vmm/);
|
---|
4669 | my $void1;
|
---|
4670 | ($void1, $vexist, $vmport) = split(/ +/,$_,3);
|
---|
4671 | pb_log(2,"pb_check_ps $vmport\n");
|
---|
4672 | $vmport =~ s/.*=tcp:127\.0\.0\.1:([0-9]+)-:22.*/$1/;
|
---|
4673 | chomp($vmport);
|
---|
4674 | pb_log(2,"pb_check_ps found a VM $vexist using port $vmport\n");
|
---|
4675 | last;
|
---|
4676 | }
|
---|
4677 | return($vexist,$vmport);
|
---|
4678 | }
|
---|
4679 |
|
---|
4680 |
|
---|
4681 | sub pb_extract_build_files {
|
---|
4682 |
|
---|
4683 | my $src=shift;
|
---|
4684 | my $dir=shift;
|
---|
4685 | my $ddir=shift;
|
---|
4686 | my $mandatory=shift || "spec";
|
---|
4687 |
|
---|
4688 | my $flag = "mayfail" if (($mandatory eq "patch") || ($mandatory eq "src"));
|
---|
4689 | my $res;
|
---|
4690 |
|
---|
4691 | if ($src =~ /tar\.gz$/) {
|
---|
4692 | $res = pb_system("tar xfpz $src $dir","Extracting $mandatory files from $src",$flag);
|
---|
4693 | } elsif ($src =~ /tar\.bz2$/) {
|
---|
4694 | $res = pb_system("tar xfpj $src $dir","Extracting $mandatory files from $src",$flag);
|
---|
4695 | } else {
|
---|
4696 | die "Unknown compression algorithm for $src";
|
---|
4697 | }
|
---|
4698 | # If not mandatory return now
|
---|
4699 | return() if (($res != 0) and (($mandatory eq "patch") || ($mandatory eq "src")));
|
---|
4700 | return(pb_move_extracted_files($dir,$ddir));
|
---|
4701 | }
|
---|
4702 |
|
---|
4703 | sub pb_move_extracted_files {
|
---|
4704 |
|
---|
4705 | my $dir = shift;
|
---|
4706 | my $ddir = shift;
|
---|
4707 | my @files;
|
---|
4708 |
|
---|
4709 | # In a recursive function , we need a local var as DIR handle
|
---|
4710 | my $bdir;
|
---|
4711 |
|
---|
4712 | opendir($bdir,"$dir") || confess "Unable to open directory $dir: $!";
|
---|
4713 | foreach my $f (readdir($bdir)) {
|
---|
4714 | next if ($f =~ /^\./);
|
---|
4715 | # Skip potential patch dir
|
---|
4716 | next if ($f =~ /^pbpatch/);
|
---|
4717 | # Skip potential source dir
|
---|
4718 | next if ($f =~ /^pbsrc/);
|
---|
4719 | # Skip potential backup files
|
---|
4720 | next if ($f =~ /~$/);
|
---|
4721 | if ((-d "$dir/$f") && (! -l "$dir/$f")) {
|
---|
4722 | # Case of the bug debian subdir
|
---|
4723 | pb_mkdir_p("$ddir/$f");
|
---|
4724 | push @files,pb_move_extracted_files("$dir/$f","$ddir/$f");
|
---|
4725 | } else {
|
---|
4726 | move("$dir/$f","$ddir") || die "Unable to move $dir/$f to $ddir";
|
---|
4727 | pb_log(2,"mv $dir/$f $ddir\n");
|
---|
4728 | push @files,"$ddir/$f";
|
---|
4729 | }
|
---|
4730 | }
|
---|
4731 | closedir($bdir);
|
---|
4732 | # Not enough but still a first cleanup
|
---|
4733 | pb_rm_rf("$dir");
|
---|
4734 | return(@files);
|
---|
4735 | }
|
---|
4736 |
|
---|
4737 | sub pb_list_bfiles {
|
---|
4738 |
|
---|
4739 | my $dir = shift;
|
---|
4740 | my $pbpkg = shift;
|
---|
4741 | my $bfiles = shift;
|
---|
4742 | my $pkgfiles = shift;
|
---|
4743 | my $supfiles = shift;
|
---|
4744 | # subdir to keep if recursive mode, empty by default
|
---|
4745 | my $subdir = shift || "";
|
---|
4746 | # In a recursive function , we need a local var as DIR handle
|
---|
4747 | my $bdir;
|
---|
4748 |
|
---|
4749 | pb_log(2,"DEBUG: entering pb_list_bfiles in $dir: ".Dumper($bfiles)."\n");
|
---|
4750 | opendir($bdir,"$dir") || die "Unable to open dir $dir: $!";
|
---|
4751 | foreach my $f (readdir($bdir)) {
|
---|
4752 | pb_log(3,"DEBUG: pb_list_bfiles found $f\n");
|
---|
4753 | next if ($f =~ /^\./);
|
---|
4754 | if (-d "$dir/$f") {
|
---|
4755 | # Recurse for directories (Debian 3.0 format e.g.)
|
---|
4756 | pb_log(2,"DEBUG: pb_list_bfiles recurse in $dir/$f\n");
|
---|
4757 | pb_list_bfiles("$dir/$f",$pbpkg,$bfiles,$pkgfiles,$supfiles,$f);
|
---|
4758 | next;
|
---|
4759 | }
|
---|
4760 |
|
---|
4761 | my $key = $f;
|
---|
4762 | # if recursive then store also the subdir
|
---|
4763 | $key = "$subdir/$f" if ($subdir ne "");
|
---|
4764 | $bfiles->{$key} = "$dir/$f";
|
---|
4765 | $bfiles->{$key} =~ s~$ENV{'PBROOTDIR'}~~;
|
---|
4766 | if (defined $supfiles->{$pbpkg}) {
|
---|
4767 | $pkgfiles->{$key} = "$dir/$f" if ($f =~ /$supfiles->{$pbpkg}/);
|
---|
4768 | }
|
---|
4769 | }
|
---|
4770 | closedir($bdir);
|
---|
4771 | pb_log(2,"DEBUG: exiting pb_list_bfiles: ".Dumper($bfiles)."\n");
|
---|
4772 | }
|
---|
4773 |
|
---|
4774 | sub pb_add_comma {
|
---|
4775 |
|
---|
4776 | my $str = shift;
|
---|
4777 | my $addstr = shift;
|
---|
4778 |
|
---|
4779 | $str .= "," if (defined $str);
|
---|
4780 | $str .= $addstr;
|
---|
4781 | return($str);
|
---|
4782 | }
|
---|
4783 |
|
---|
4784 | sub pb_list_sfiles {
|
---|
4785 |
|
---|
4786 | my $sdir = shift;
|
---|
4787 | my $ptr = shift;
|
---|
4788 | my $pbos = shift;
|
---|
4789 | my $extdir = shift;
|
---|
4790 |
|
---|
4791 | pb_log(2,"DEBUG: entering pb_list_sfiles: ".Dumper($ptr)."\n");
|
---|
4792 | my $key = "$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}";
|
---|
4793 |
|
---|
4794 | # Prepare local sources for this distro - They are always applied first - May be a problem one day
|
---|
4795 | # This function works for both patches and additional sources
|
---|
4796 | foreach my $p (sort(<$sdir/*>)) {
|
---|
4797 | $ptr->{$key} = pb_add_comma($ptr->{$key},"file://$p") if (($p =~ /\.all$/) || ($p =~ /\.$pbos->{'os'}$/) || ($p =~ /\.$pbos->{'type'}$/) || ($p =~ /\.$pbos->{'family'}$/) || ($p =~ /\.$pbos->{'name'}$/) || ($p =~ /\.$pbos->{'name'}-$pbos->{'version'}$/) ||($p =~ /\.$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}$/));
|
---|
4798 | }
|
---|
4799 |
|
---|
4800 | # Prepare also remote sources to be included - Applied after the local ones
|
---|
4801 | foreach my $p ("all","$pbos->{'os'}","$pbos->{'type'}","$pbos->{'family'}","$pbos->{'name'}","$pbos->{'name'}-$pbos->{'version'}","$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}") {
|
---|
4802 | my $f = "$extdir.".".$p";
|
---|
4803 | next if (not -f $f);
|
---|
4804 | if (not open(PATCH,$f)) {
|
---|
4805 | pb_display("Unable to open existing external source file content $f\n");
|
---|
4806 | next;
|
---|
4807 | }
|
---|
4808 | while (<PATCH>) {
|
---|
4809 | chomp();
|
---|
4810 | $ptr->{$key} = pb_add_comma($ptr->{$key},"$_");
|
---|
4811 | }
|
---|
4812 | close(PATCH);
|
---|
4813 | }
|
---|
4814 | pb_log(2,"DEBUG: exiting pb_list_sfiles: ".Dumper($ptr)."\n");
|
---|
4815 | return($ptr);
|
---|
4816 | }
|
---|
4817 |
|
---|
4818 | #
|
---|
4819 | # Return the list of packages we are working on in a non CMS action
|
---|
4820 | #
|
---|
4821 | sub pb_get_pkg {
|
---|
4822 |
|
---|
4823 | my @pkgs = ();
|
---|
4824 |
|
---|
4825 | # In script actions that file may not exist.
|
---|
4826 | return(\@pkgs) if ((not defined $ENV{'PBDESTDIR'}) || (not defined $ENV{'PBPROJVER'}) || (not defined $ENV{'PBPROJTAG'}) || (! -f "$ENV{'PBDESTDIR'}/$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}.pb"));
|
---|
4827 |
|
---|
4828 | my ($var) = pb_conf_read("$ENV{'PBDESTDIR'}/$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}.pb","pbpkg");
|
---|
4829 | @pkgs = keys %$var if (defined $var);
|
---|
4830 |
|
---|
4831 | pb_log(0,"Packages: ".join(',',@pkgs)."\n") if (@pkgs);
|
---|
4832 | return(\@pkgs);
|
---|
4833 | }
|
---|
4834 |
|
---|
4835 | # Manages VM/RM SSH port communication
|
---|
4836 | sub pb_get_port {
|
---|
4837 |
|
---|
4838 | my $port = shift;
|
---|
4839 | my $pbos = shift;
|
---|
4840 | my $cmt = shift;
|
---|
4841 | my $nport;
|
---|
4842 |
|
---|
4843 | die "No port passed in parameter. Report to dev team\n" if (not defined $port);
|
---|
4844 | # key is project on VM, but machine tuple for RM
|
---|
4845 | if ($cmt =~ /^RM/i) {
|
---|
4846 | $nport = $port->{"$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}"};
|
---|
4847 | } else {
|
---|
4848 | $nport = $port->{$ENV{'PBPROJ'}};
|
---|
4849 | }
|
---|
4850 | pb_log(2,"pb_get_port with $nport\n");
|
---|
4851 | if ($cmt =~ /^VM/i) {
|
---|
4852 | # Maybe a port was given as parameter so overwrite
|
---|
4853 | $nport = "$pbport" if (defined $pbport);
|
---|
4854 | # Maybe in // mode so use the env var set up as an offset to the base port
|
---|
4855 | $nport = $nport + $ENV{'PBVMPORT'} -1 if ((defined $pbparallel) && (defined $ENV{'PBVMPORT'}));
|
---|
4856 | }
|
---|
4857 | pb_log(2,"pb_get_port returns $nport\n");
|
---|
4858 | return($nport);
|
---|
4859 | }
|
---|
4860 |
|
---|
4861 | sub pb_set_port {
|
---|
4862 |
|
---|
4863 | my ($pid,$ident) = @_;
|
---|
4864 | pb_log(2,"pb_set_port for VM ($pid), id $ident\n");
|
---|
4865 | $ENV{'PBVMPORT'} = $ident;
|
---|
4866 | pb_log(2,"pb_set_port sets PBVMPORT in env to $ENV{'PBVMPORT'}\n");
|
---|
4867 | }
|
---|
4868 |
|
---|
4869 | sub pb_set_parallel {
|
---|
4870 |
|
---|
4871 | my $vtype = shift;
|
---|
4872 |
|
---|
4873 | pb_log(2,"pb_set_parallel vtype: $vtype\n");
|
---|
4874 | # Take care of memory size if VM, parallel mode and more than 1 action
|
---|
4875 | if ((defined $pbparallel) && ($pbparallel ne 1) && ($vtype eq "vm")) {
|
---|
4876 | eval
|
---|
4877 | {
|
---|
4878 | require Linux::SysInfo;
|
---|
4879 | Linux::SysInfo->import();
|
---|
4880 | };
|
---|
4881 | if ($@) {
|
---|
4882 | # Linux::SysInfo not found
|
---|
4883 | pb_log(1,"ADVISE: Install Linux::SysInfo to benefit from automatic parallelism optimization.\nOr optimize manually pbparallel in your pb.conf file\nUsing $pbparallel processes max at a time for the moment\nWARNING: This may consume too much memory for your system\n");
|
---|
4884 | } else {
|
---|
4885 | # Using the memory size
|
---|
4886 | my $si = Linux::SysInfo::sysinfo();
|
---|
4887 | if (not defined $si) {
|
---|
4888 | pb_log(1,"ADVISE: Install Linux::SysInfo to benefit from automatic parallelism optimization.\nOr optimize manually pbparallel in your pb.conf file\nUsing $pbparallel processes max at a time for the moment\nWARNING: This may consume too much memory for your system\n");
|
---|
4889 | } else {
|
---|
4890 | # Keep the number of VM whose memory can be allocated
|
---|
4891 | my $ram = $si->{"totalram"}-$si->{"sharedram"}-$si->{"bufferram"};
|
---|
4892 | my $ram2;
|
---|
4893 | my ($vmmem) = pb_conf_get_if("vmmem");
|
---|
4894 |
|
---|
4895 | my $v = "default";
|
---|
4896 | if ((defined $vmmem) and (defined $vmmem->{$v})) {
|
---|
4897 | $ram2 = $vmmem->{$v};
|
---|
4898 | } else {
|
---|
4899 | # Default for KVM/QEMU
|
---|
4900 | $ram2 = 128;
|
---|
4901 | }
|
---|
4902 | $pbparallel = sprintf("%d",$ram/$ram2);
|
---|
4903 | }
|
---|
4904 | pb_log(1,"Using $pbparallel processes at a time\n");
|
---|
4905 | }
|
---|
4906 | }
|
---|
4907 | pb_log(2,"pb_set_parallel returns: $pbparallel\n") if (defined $pbparallel);
|
---|
4908 | return($pbparallel);
|
---|
4909 | }
|
---|
4910 |
|
---|
4911 | sub pb_get_sudocmds {
|
---|
4912 |
|
---|
4913 | my $pbos = shift;
|
---|
4914 | my @cmds = @_;
|
---|
4915 | my %sudocmds;
|
---|
4916 |
|
---|
4917 | pb_log(2,"pb_get_sudocmds entering with lines:".Dumper(@cmds)."\n");
|
---|
4918 | foreach my $c (split(/;/,$pbos->{'update'}),split(/;/,$pbos->{'install'}),@cmds) {
|
---|
4919 | pb_log(2,"pb_get_sudocmds analyses $c\n");
|
---|
4920 | next if ($c !~ /^\s*sudo/);
|
---|
4921 | # remove sudo and leading spaces
|
---|
4922 | $c =~ s/^\s*sudo\s+//;
|
---|
4923 | # keep only the command, not the params
|
---|
4924 | $c =~ s/([^\s]+)\s.*$/$1/;
|
---|
4925 | $sudocmds{$c} = "";
|
---|
4926 | }
|
---|
4927 | pb_log(2,"pb_get_sudocmds returns ".Dumper(keys %sudocmds)."\n");
|
---|
4928 | return(keys %sudocmds);
|
---|
4929 | }
|
---|
4930 |
|
---|
4931 | sub pb_sign_setenv {
|
---|
4932 |
|
---|
4933 | my ($passfile, $passphrase, $passpath) = pb_conf_get_if("pbpassfile","pbpassphrase","pbpasspath");
|
---|
4934 | $ENV{'PBPASSPHRASE'} = $passphrase->{$ENV{'PBPROJ'}} if ((not defined $ENV{'PBPASSPHRASE'}) && (defined $passphrase->{$ENV{'PBPROJ'}}));
|
---|
4935 | $ENV{'PBPASSFILE'} = $passfile->{$ENV{'PBPROJ'}} if ((not defined $ENV{'PBPASSFILE'})&& (defined $passfile->{$ENV{'PBPROJ'}})) ;
|
---|
4936 | $ENV{'PBPASSPATH'} = $passpath->{$ENV{'PBPROJ'}} if ((not defined $ENV{'PBPASSPATH'})&& (defined $passpath->{$ENV{'PBPROJ'}})) ;
|
---|
4937 |
|
---|
4938 | eval
|
---|
4939 | {
|
---|
4940 | require RPM4::Sign;
|
---|
4941 | RPM4::Sign->import();
|
---|
4942 | };
|
---|
4943 | if ($@) {
|
---|
4944 | # RPM4::Sign not found
|
---|
4945 | pb_log(1,"WARNING: Install RPM4::Sign to benefit from automatic package signing.\n");
|
---|
4946 | return(undef);
|
---|
4947 | } else {
|
---|
4948 | return(undef) if (((not defined $ENV{'PBPASSPHRASE'}) and (not defined $ENV{'PBPASSFILE'})) || (not defined $ENV{'PBPACKAGER'}) || (not defined $ENV{'PBPASSPATH'}));
|
---|
4949 | my $sign = RPM4::Sign->new(
|
---|
4950 | passphrase => $ENV{'PBPASSPHRASE'},
|
---|
4951 | name => $ENV{'PBPACKAGER'},
|
---|
4952 | path => $ENV{'PBPASSPATH'},
|
---|
4953 | password_file => $ENV{'PBPASSFILE'},
|
---|
4954 | );
|
---|
4955 | return(undef) if (not defined $sign);
|
---|
4956 | return(undef) if (not defined $sign->{name});
|
---|
4957 | return(undef) if (not defined $sign->{path});
|
---|
4958 | return(undef) if ((not defined $sign->{passphrase}) && (not defined $sign->{password_file}));
|
---|
4959 | return(undef) if (not defined $sign->{keyid});
|
---|
4960 |
|
---|
4961 | return($sign);
|
---|
4962 | }
|
---|
4963 | }
|
---|
4964 |
|
---|
4965 | # Cf: http://wiki.debian.org/SecureApt
|
---|
4966 |
|
---|
4967 | sub pb_sign_file {
|
---|
4968 |
|
---|
4969 | my $file = shift;
|
---|
4970 |
|
---|
4971 | eval
|
---|
4972 | {
|
---|
4973 | require Crypt::OpenPGP;
|
---|
4974 | Crypt::OpenPGP->import();
|
---|
4975 | };
|
---|
4976 | if ($@) {
|
---|
4977 | # Crypt::OpenPGP not found
|
---|
4978 | pb_log(1,"WARNING: Install Crypt::OpenPGP to benefit from debian Release file signing.\n");
|
---|
4979 | } else {
|
---|
4980 | my $sign = pb_sign_setenv();
|
---|
4981 | my $pgp = Crypt::OpenPGP->new;
|
---|
4982 | my $pgpsign = $pgp->sign(
|
---|
4983 | Filename => $file,
|
---|
4984 | KeyID => $sign->{keyid},
|
---|
4985 | Passphrase => $sign->{passphrase},
|
---|
4986 | Detach => 1,
|
---|
4987 | Armour => 1,
|
---|
4988 | );
|
---|
4989 | open(DEST, "> $file.gpg") || confess "Unable to write to $file.gpg";
|
---|
4990 | print DEST $pgpsign if (defined $pgpsign);
|
---|
4991 | close(DEST);
|
---|
4992 | }
|
---|
4993 | }
|
---|
4994 |
|
---|
4995 | sub pb_sign_pkgs {
|
---|
4996 |
|
---|
4997 | my $pbos = shift;
|
---|
4998 | my $made = shift;
|
---|
4999 |
|
---|
5000 | pb_log(2,"entering pb_sign_pkg: $made ".Dumper($pbos)."\n");
|
---|
5001 | # Remove extra spaces
|
---|
5002 | $made =~ s/\s+/ /g;
|
---|
5003 | $made =~ s/^\s//g;
|
---|
5004 | $made =~ s/\s$//g;
|
---|
5005 |
|
---|
5006 | if ($pbos->{'type'} eq "rpm") {
|
---|
5007 | my $sign = pb_sign_setenv();
|
---|
5008 | if (defined $sign) {
|
---|
5009 | pb_log(0,"Signing RPM packages...\n");
|
---|
5010 | pb_log(2,"pb_sign_pkg: pkgs:".Dumper(split(/ /,$made))."\n");
|
---|
5011 | $sign->rpmssign(split(/ /,$made));
|
---|
5012 | }
|
---|
5013 | } elsif ($pbos->{'type'} eq "deb") {
|
---|
5014 | my $changes = "";
|
---|
5015 | foreach my $c (split(/ /,$made)) {
|
---|
5016 | $changes .= " $ENV{'PBBUILDDIR'}/$c" if (($c =~ /\.changes$/) && (-f "$ENV{PBBUILDDIR}/$c"));
|
---|
5017 | }
|
---|
5018 | my $debsigncmd = pb_check_req("debsign",1);
|
---|
5019 | pb_system("$debsigncmd -m\'$ENV{'PBPACKAGER'}\' $changes","Signing DEB packages","mayfail") if (($changes ne "") && (defined $debsigncmd));
|
---|
5020 | } else {
|
---|
5021 | pb_log(0,"I don't know yet how to sign packages for type $pbos->{'type'}.\nPlease give feedback to dev team\n");
|
---|
5022 | }
|
---|
5023 | pb_log(2,"exiting pb_sign_pkg\n");
|
---|
5024 | }
|
---|
5025 |
|
---|
5026 | # return list of all distributions supported, comma separated
|
---|
5027 | sub pb_get_distros {
|
---|
5028 |
|
---|
5029 | my $pbos = shift;
|
---|
5030 | my $pbtarget = shift;
|
---|
5031 |
|
---|
5032 | my @dists = ("$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}");
|
---|
5033 |
|
---|
5034 | # Get list of distributions for which we need to generate build files if no target
|
---|
5035 | if (not defined $pbtarget) {
|
---|
5036 | my @pt = pb_conf_get_if("vmlist","velist","rmlist");
|
---|
5037 | foreach my $pt (@pt) {
|
---|
5038 | push(@dists, split(/,/, $pt->{$ENV{PBPROJ}})) if defined $pt->{$ENV{PBPROJ}};
|
---|
5039 | }
|
---|
5040 | # remove any whitespace
|
---|
5041 | grep(s/\s+//go, @dists);
|
---|
5042 | }
|
---|
5043 | return(join(",",@dists));
|
---|
5044 | }
|
---|
5045 |
|
---|
5046 | sub pb_get_extdir {
|
---|
5047 |
|
---|
5048 | # the pbrc file should contain it and whatever the key, we take it
|
---|
5049 | my ($ed) = pb_conf_read("$ENV{'PBDESTDIR'}/pbrc","pbextdir");
|
---|
5050 | pb_log(2,"ed: ".Dumper($ed)."\n");
|
---|
5051 | my $pbextdir = "";
|
---|
5052 | foreach my $k (keys %$ed) {
|
---|
5053 | $pbextdir = $ed->{$k};
|
---|
5054 | # In case we have an empty field, empty it completely
|
---|
5055 | pb_log(2,"pbextdir: ***$pbextdir***\n");
|
---|
5056 | $pbextdir =~ s/^\s*$//;
|
---|
5057 | }
|
---|
5058 | pb_log(2,"pbextdir: ***$pbextdir***\n");
|
---|
5059 | return($pbextdir);
|
---|
5060 | }
|
---|
5061 |
|
---|
5062 | # Unused now
|
---|
5063 | sub pb_keep_step {
|
---|
5064 |
|
---|
5065 | my $pbos = shift;
|
---|
5066 | my $pbstep = shift;
|
---|
5067 |
|
---|
5068 | $pbstep = 3 if (not defined $pbstep);
|
---|
5069 |
|
---|
5070 | my %h;
|
---|
5071 | my $h = \%h;
|
---|
5072 | $h = pb_conf_cache("$ENV{'PBDESTDIR'}/$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}.pb",$h);
|
---|
5073 | $h->{'pbstep'}->{"$pbos->{'name'}-$pbos->{'version'}-$pbos->{'arch'}"} = $pbstep;
|
---|
5074 | pb_conf_write("$ENV{'PBDESTDIR'}/$ENV{'PBPROJVER'}-$ENV{'PBPROJTAG'}.pb",$h);
|
---|
5075 | }
|
---|
5076 |
|
---|
5077 | sub create_solaris_prototype {
|
---|
5078 |
|
---|
5079 | my $uidgid = "bin bin";
|
---|
5080 | my $pkgdestdir = $ENV{'PBSOLDESTDIR'};
|
---|
5081 |
|
---|
5082 | return if ($_ =~ /^$pkgdestdir$/);
|
---|
5083 | if (-d $_) {
|
---|
5084 | my $n = $File::Find::name;
|
---|
5085 | $n =~ s~$pkgdestdir/~~;
|
---|
5086 | print PROTO "d none $n 0755 $uidgid\n";
|
---|
5087 | } elsif (-x $_) {
|
---|
5088 | my $n = $File::Find::name;
|
---|
5089 | $n =~ s~$pkgdestdir/~~;
|
---|
5090 | print PROTO "f none $n 0755 $uidgid\n";
|
---|
5091 | } elsif (-f $_) {
|
---|
5092 | my $n = $File::Find::name;
|
---|
5093 | $n =~ s~$pkgdestdir/~~;
|
---|
5094 | print PROTO "f none $n 0644 $uidgid\n";
|
---|
5095 | }
|
---|
5096 | }
|
---|
5097 |
|
---|
5098 | sub pb_ssh_setup {
|
---|
5099 |
|
---|
5100 | my $vtype = shift;
|
---|
5101 | my $nport = shift;
|
---|
5102 |
|
---|
5103 | my $keyfile;
|
---|
5104 | my $vmhost;
|
---|
5105 |
|
---|
5106 | pb_log(1,"Entering pb_ssh_setup\n");
|
---|
5107 | ($vmhost) = pb_conf_get($vtype."host");
|
---|
5108 | # Prepare the key to be used and transfered remotely
|
---|
5109 | $keyfile = pb_ssh_get(1,$vmhost->{$ENV{'PBPROJ'}},$nport);
|
---|
5110 | return if (not defined $keyfile);
|
---|
5111 |
|
---|
5112 | # We call true to avoid problems if SELinux is not activated, but chcon is present and returns in that case 1
|
---|
5113 | pb_system("cat $keyfile.pub | ssh -q -o UserKnownHostsFile=/dev/null -p $nport -i $keyfile root\@$vmhost->{$ENV{'PBPROJ'}} \"mkdir -p .ssh ; chmod 700 .ssh ; cat >> .ssh/authorized_keys ; chmod 600 .ssh/authorized_keys ; if [ -x /usr/bin/chcon ]; then /usr/bin/chcon -Rt home_ssh_t .ssh 2> /dev/null; /bin/true; fi\"","Copying local keys to $vtype. This may require the root password");
|
---|
5114 | # once this is done, we can do what we need on the VM/RM remotely
|
---|
5115 |
|
---|
5116 | # in particular we can remove duplicate in .ssh/authorized_keys of our key if needed
|
---|
5117 | # Store the pub key part in a variable
|
---|
5118 | open(FILE,"$keyfile.pub") || die "Unable to open $keyfile.pub";
|
---|
5119 | my ($zero0,$zero1,$zero2) = split(/ /,<FILE>);
|
---|
5120 | close(FILE);
|
---|
5121 | my $key = "\Q$zero1";
|
---|
5122 | my $perls = "$ENV{'PBTMP'}/perls.$$";
|
---|
5123 |
|
---|
5124 | open(FILE,"> $perls") || die "Unable to open $perls";
|
---|
5125 | print FILE << 'EOF';
|
---|
5126 | my $file1="$ENV{'HOME'}/.ssh/authorized_keys";
|
---|
5127 | open(PBFILE,$file1) || die "Unable to open $file1";
|
---|
5128 | open(PBOUT,"> $file1.new") || die "Unable to open $file1.new";
|
---|
5129 | my $count = 0;
|
---|
5130 | while (<PBFILE>) {
|
---|
5131 | EOF
|
---|
5132 | print FILE << "EOF";
|
---|
5133 | if (/ $key /) {
|
---|
5134 | \$count++;
|
---|
5135 | }
|
---|
5136 | print PBOUT \$_ if ((\$count <= 1) || (\$_ !~ / $key /));
|
---|
5137 | }
|
---|
5138 | close(PBFILE);
|
---|
5139 | close(PBOUT);
|
---|
5140 | rename("\$file1.new",\$file1);
|
---|
5141 | chmod 0600,\$file1;
|
---|
5142 | EOF
|
---|
5143 | close(FILE);
|
---|
5144 | pb_system("cat $perls | ssh -q -o UserKnownHostsFile=/dev/null -p $nport -i $keyfile root\@$vmhost->{$ENV{'PBPROJ'}} perl","","quiet");
|
---|
5145 | return;
|
---|
5146 | }
|
---|
5147 |
|
---|
5148 | # Returns debian architecture
|
---|
5149 | #
|
---|
5150 | sub pb_get_debarch {
|
---|
5151 |
|
---|
5152 | my $pbos = shift;
|
---|
5153 |
|
---|
5154 | my $debarch = $pbos->{'arch'};
|
---|
5155 | $debarch = "amd64" if ($pbos->{'arch'} eq "x86_64");
|
---|
5156 |
|
---|
5157 | return($debarch);
|
---|
5158 | }
|
---|
5159 |
|
---|
5160 | sub pb_get_debpc {
|
---|
5161 |
|
---|
5162 | my ($projcomponent_map) = pb_conf_get_if("projcomponent");
|
---|
5163 | pb_log(2,"projcomponent = ".Dumper($projcomponent_map)."\n");
|
---|
5164 | my $projcomponent = $projcomponent_map->{$ENV{PBPROJ}};
|
---|
5165 | $projcomponent ||= 'contrib';
|
---|
5166 |
|
---|
5167 | return($projcomponent);
|
---|
5168 | }
|
---|
5169 |
|
---|
5170 | 1;
|
---|