source: ProjectBuilder/devel/pb-modules/lib/ProjectBuilder/Base.pm@ 1060

Last change on this file since 1060 was 1060, checked in by Bruno Cornec, 14 years ago

r3906@wsip-70-165-197-185: bruno | 2010-06-20 08:04:11 +0200

  • Force printing on stdout in pb_log if 0 level
File size: 11.4 KB
Line 
1#!/usr/bin/perl -w
2#
3# Base subroutines brought by the the Project-Builder project
4# which can be easily used by whatever perl project
5#
6# Copyright B. Cornec 2007-2008
7# Provided under the GPL v2
8#
9# $Id$
10#
11
12package ProjectBuilder::Base;
13
14use strict;
15use lib qw (lib);
16use File::Path;
17use File::Temp qw(tempdir);
18use Data::Dumper;
19use Time::localtime qw(localtime);
20use Pod::Usage;
21use English;
22use POSIX qw(locale_h);
23
24# Inherit from the "Exporter" module which handles exporting functions.
25
26use Exporter;
27
28# Export, by default, all the functions into the namespace of
29# any code which uses this module.
30
31our $pbdebug = 0; # Global debug level
32our $pbLOG = \*STDOUT; # File descriptor of the log file
33our $pbsynmsg = "Error"; # Global error message
34our $pbdisplaytype = "text";
35 # default display mode for messages
36our $pblocale = "C";
37
38our @ISA = qw(Exporter);
39our @EXPORT = qw(pb_mkdir_p pb_system pb_rm_rf pb_get_date pb_log pb_log_init pb_get_uri pb_get_content pb_set_content pb_display_file pb_syntax_init pb_syntax pb_temp_init pb_get_arch pb_check_requirements $pbdebug $pbLOG $pbdisplaytype $pblocale);
40
41=pod
42
43=head1 NAME
44
45ProjectBuilder::Base, part of the project-builder.org - module dealing with generic functions suitable for perl project development
46
47=head1 DESCRIPTION
48
49This modules provides generic functions suitable for perl project development
50
51=head1 SYNOPSIS
52
53 use ProjectBuilder::Base;
54
55 #
56 # Create a directory and its parents
57 #
58 pb_mkdir_p("/tmp/foo/bar");
59
60 #
61 # Remove recursively a directory and its children
62 #
63 pb_rm_rf("/tmp/foo");
64
65 #
66 # Encapsulate the system call for better output and return value test
67 #
68 pb_system("ls -l", "Printing directory content");
69
70 #
71 # Analysis a URI and return its components in a table
72 #
73 my ($scheme, $account, $host, $port, $path) = pb_get_uri("svn+ssh://ac@my.server.org/path/to/dir");
74
75 #
76 # Gives the current date in a table
77 #
78 @date = pb_get_date();
79
80 #
81 # Manages logs of the program
82 #
83 pb_log_init(2,\*STDOUT);
84 pb_log(1,"Message to print\n");
85
86 #
87 # Manages content of a file
88 #
89 pb_display_file("/etc/passwd");
90 my $cnt = pb_get_content("/etc/passwd");
91
92=head1 USAGE
93
94=over 4
95
96=item B<pb_mkdir_p>
97
98Internal mkdir -p function. Forces mode to 755. Supports multiple parameters.
99
100Based on File::Path mkpath.
101
102=cut
103
104sub pb_mkdir_p {
105my @dir = @_;
106my $ret = mkpath(@dir, 0, 0755);
107return($ret);
108}
109
110=item B<pb_rm_rf>
111
112Internal rm -rf function. Supports multiple parameters.
113
114Based on File::Path rmtree.
115
116=cut
117
118sub pb_rm_rf {
119my @dir = @_;
120my $ret = rmtree(@dir, 0, 0);
121return($ret);
122}
123
124=item B<pb_system>
125
126Encapsulate the "system" call for better output and return value test
127Needs a $ENV{'PBTMP'} variable which is created by calling the pb_mktemp_init function
128Needs pb_log support, so pb_log_init should have benn called before.
129
130The first parameter is the shell command to call.
131The second parameter is the message to print on screen. If none is given, then the command is printed.
132The third parameter print the result of the command after correct execution if value is verbose. If value is noredir, it avoids redirecting outputs (e.g. for vi).
133This function returns the result the return value of the system command.
134
135If no error reported, it prints OK on the screen, just after the message. Else it prints the errors generated.
136
137=cut
138
139sub pb_system {
140
141my $cmd=shift;
142my $cmt=shift || $cmd;
143my $verbose=shift || undef;
144my $redir = "";
145
146pb_log(0,"$cmt... ") if ((! defined $verbose) || ($verbose ne "quiet"));
147pb_log(1,"Executing $cmd\n");
148unlink("$ENV{'PBTMP'}/system.log") if (-f "$ENV{'PBTMP'}/system.log");
149$redir = "2>> $ENV{'PBTMP'}/system.log 1>> $ENV{'PBTMP'}/system.log" if ((! defined $verbose) || ($verbose ne "noredir"));
150system("$cmd $redir");
151my $res = $?;
152# Exit now if the command may fail
153if ((defined $verbose) and ($verbose eq "mayfail")) {
154 pb_log(0,"N/A\n") if ($res != 0);
155 pb_log(0,"OK\n") if ($res == 0);
156 return($res)
157 }
158if ($res == -1) {
159 pb_log(0,"failed to execute ($cmd): $!\n") if ((! defined $verbose) || ($verbose ne "quiet"));
160 pb_display_file("$ENV{'PBTMP'}/system.log") if ((-f "$ENV{'PBTMP'}/system.log") and ((! defined $verbose) || ($verbose ne "quiet")));
161} elsif ($res & 127) {
162 pb_log(0, "child ($cmd) died with signal ".($? & 127).", ".($? & 128) ? 'with' : 'without'." coredump\n") if ((! defined $verbose) || ($verbose ne "quiet"));
163 pb_display_file("$ENV{'PBTMP'}/system.log") if ((-f "$ENV{'PBTMP'}/system.log") and ((! defined $verbose) || ($verbose ne "quiet")));
164} elsif ($res == 0) {
165 pb_log(0,"OK\n") if ((! defined $verbose) || ($verbose ne "quiet"));
166 pb_display_file("$ENV{'PBTMP'}/system.log") if ((defined $verbose) and (-f "$ENV{'PBTMP'}/system.log") and ($verbose ne "quiet"));
167} else {
168 pb_log(0, "child ($cmd) exited with value ".($? >> 8)."\n") if ((! defined $verbose) || ($verbose ne "quiet"));
169 pb_display_file("$ENV{'PBTMP'}/system.log") if ((-f "$ENV{'PBTMP'}/system.log") and ((! defined $verbose) || ($verbose ne "quiet")));
170}
171return($res);
172}
173
174=item B<pb_get_uri>
175
176This function returns a list of 6 parameters indicating the protocol, account, password, server, port, and path contained in the URI passed in parameter.
177
178A URI has the format protocol://[ac@]host[:port][path[?query][#fragment]].
179
180Cf man URI.
181
182=cut
183
184sub pb_get_uri {
185
186my $uri = shift || undef;
187
188pb_log(2,"DEBUG: uri:$uri\n");
189my ($scheme, $authority, $path, $query, $fragment) =
190 $uri =~ m|(?:([^:/?#]+):)?(?://([^/?#]*))?([^?#]*)(?:\?([^#]*))?(?:#(.*))?| if (defined $uri);
191my ($account,$host,$port) = $authority =~ m|(?:([^\@]+)\@)?([^:]+)(:(?:[0-9]+))?| if (defined $authority);
192
193$scheme = "" if (not defined $scheme);
194$authority = "" if (not defined $authority);
195$path = "" if (not defined $path);
196$account = "" if (not defined $account);
197$host = "" if (not defined $host);
198$port = "" if (not defined $port);
199
200pb_log(2,"DEBUG: scheme:$scheme ac:$account host:$host port:$port path:$path\n");
201return($scheme, $account, $host, $port, $path);
202}
203
204=item B<pb_get_date>
205
206This function returns a list of 9 parameters indicating the seconds, minutes, hours, day, month, year, day in the week, day in the year, and daylight saving time flag of the current time.
207
208Cf: man ctime and description of the struct tm.
209
210=cut
211
212sub pb_get_date {
213
214return(localtime->sec(), localtime->min(), localtime->hour(), localtime->mday(), localtime->mon(), localtime->year(), localtime->wday(), localtime->yday(), localtime->isdst());
215}
216
217=item B<pb_log_init>
218
219This function initializes the global variables used by the pb_log function.
220
221The first parameter is the debug level which will be considered during the run of the program?
222The second parameter is a pointer on a file descriptor used to print the log info.
223
224As an example, if you set the debug level to 2 in that function, every call to pb_log which contains a value less or equal than 2 will be printed. Calls with a value greater than 2 won't be printed.
225
226The call to B<pb_log_init> is typically done after getting a parameter on the CLI indicating the level of verbosity expected.
227
228=cut
229
230sub pb_log_init {
231
232$pbdebug = shift || 0;
233$pbLOG = shift || \*STDOUT;
234pb_log(1,"Debug value: $pbdebug\n");
235
236}
237
238=item B<pb_log>
239
240This function logs the messages passed as the second parameter if the value passed as first parameter is lesser or equal than the value passed to the B<pb_log_init> function.
241
242Here is a usage example:
243
244 pb_log_init(2,\*STDERR);
245 pb_log(1,"Hello World 1\n");
246 pb_log(2,"Hello World 2\n");
247 pb_log(3,"Hello World 3\n");
248
249 will print:
250
251 Hello World 1
252 Hello World 2
253
254=cut
255
256sub pb_log {
257
258my $dlevel = shift;
259my $msg = shift;
260
261print $pbLOG "$msg" if ($dlevel <= $pbdebug);
262print "$msg" if (($dlevel == 0) && ($pbLOG != \*STDOUT));
263}
264
265
266=item B<pb_display_file>
267
268This function print the content of the file passed in parameter.
269
270This is a cat equivalent function.
271
272=cut
273
274sub pb_display_file {
275
276my $file=shift;
277
278return if (not -f $file);
279printf "%s\n",pb_get_content($file);
280}
281
282=item B<pb_get_content>
283
284This function returns the content of the file passed in parameter.
285
286=cut
287
288sub pb_get_content {
289
290my $file=shift;
291
292my $bkp = $/;
293undef $/;
294open(R,$file) || die "Unable to open $file: $!";
295my $content=<R>;
296close(R);
297chomp($content);
298$/ = $bkp;
299return($content);
300}
301
302
303=item B<pb_set_content>
304
305This function put the content of a file into the file passed in parameter.
306
307=cut
308
309sub pb_set_content {
310
311my $file=shift;
312my $content=shift;
313
314my $bkp = $/;
315undef $/;
316open(R,"> $file") || die "Unable to write to $file: $!";
317print R "$content";
318close(R);
319$/ = $bkp;
320}
321
322=item B<pb_syntax_init>
323
324This function initializes the global variable used by the pb_syntax function.
325
326The parameter is the message string which will be printed when calling pb_syntax
327
328=cut
329
330sub pb_syntax_init {
331
332$pbsynmsg = shift || "Error";
333}
334
335=item B<pb_syntax>
336
337This function prints the syntax expected by the application, based on pod2usage, and exits.
338The first parameter is the return value of the exit.
339The second parameter is the verbosity as expected by pod2usage.
340
341Cf: man Pod::Usage
342
343=cut
344
345sub pb_syntax {
346
347my $exit_status = shift || -1;
348my $verbose_level = shift || 0;
349
350my $filehandle = \*STDERR;
351
352$filehandle = \*STDOUT if ($exit_status == 0);
353
354pod2usage( { -message => $pbsynmsg,
355 -exitval => $exit_status ,
356 -verbose => $verbose_level,
357 -output => $filehandle } );
358}
359
360=item B<pb_temp_init>
361
362This function initializes the environemnt variable PBTMP to a random value. This directory can be safely used during the whole program, it will be removed at the end automatically.
363
364=cut
365
366sub pb_temp_init {
367
368if (not defined $ENV{'TMPDIR'}) {
369 $ENV{'TMPDIR'}="/tmp";
370}
371$ENV{'PBTMP'} = tempdir( "pb.XXXXXXXXXX", DIR => $ENV{'TMPDIR'}, CLEANUP => 1 );
372}
373
374=item B<pb_get_arch>
375
376This function returns the architecture of our local environment and
377standardize on i386 for those platforms. It also solves issues where a i386 VE on x86_64 returns x86_64 wrongly
378
379=cut
380
381sub pb_get_arch {
382
383my $arch = `uname -m`;
384chomp($arch);
385$arch =~ s/i.86/i386/;
386# For Solaris
387$arch =~ s/i86pc/i386/;
388
389return($arch);
390}
391
392=item B<pb_check_requirements>
393
394This function checks that the commands needed for the subsystem are indeed present.
395The required comands are passed as a coma separated string as first parameter.
396The optional comands are passed as a coma separated string as second parameter.
397
398=cut
399
400sub pb_check_requirements {
401
402my $cmds = shift || "";
403my $options = shift || "";
404
405# cmds is a string of coma separated commands
406foreach my $file (split(/,/,$cmds)) {
407 pb_check_req($file,0);
408}
409
410# opts is a string of coma separated commands
411foreach my $file (split(/,/,$options)) {
412 pb_check_req($file,1);
413}
414}
415
416sub pb_check_req {
417
418my $file = shift;
419my $opt = shift || 1;
420my $found = 0;
421
422pb_log(2,"Checking availability of $file...");
423# Check for all dirs in the PATH
424foreach my $p (split(/:/,$ENV{'PATH'})) {
425 $found = 1 if (-x "$p/$file");
426}
427if ($found eq 0) {
428 pb_log(2,"KO\n");
429 if ($opt eq 1) {
430 pb_log(2,"Unable to find optional command $file\n");
431 } else {
432 die pb_log(0,"Unable to find required command $file\n");
433 }
434} else {
435 pb_log(2,"OK\n");
436}
437}
438
439=back
440
441=head1 WEB SITES
442
443The 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/>.
444
445=head1 USER MAILING LIST
446
447None exists for the moment.
448
449=head1 AUTHORS
450
451The Project-Builder.org team L<http://trac.project-builder.org/> lead by Bruno Cornec L<mailto:bruno@project-builder.org>.
452
453=head1 COPYRIGHT
454
455Project-Builder.org is distributed under the GPL v2.0 license
456described in the file C<COPYING> included with the distribution.
457
458=cut
459
4601;
Note: See TracBrowser for help on using the repository browser.