~drizzle-trunk/drizzle/development

« back to all changes in this revision

Viewing changes to scripts/mysql_install_db.pl.in

  • Committer: brian
  • Date: 2008-06-25 05:29:13 UTC
  • Revision ID: brian@localhost.localdomain-20080625052913-6upwo0jsrl4lnapl
clean slate

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
#!/usr/bin/perl
 
2
# -*- cperl -*-
 
3
#
 
4
# Copyright (C) 2007 MySQL AB
 
5
#
 
6
# This program is free software; you can redistribute it and/or modify
 
7
# it under the terms of the GNU General Public License as published by
 
8
# the Free Software Foundation; version 2 of the License.
 
9
#
 
10
# This program is distributed in the hope that it will be useful,
 
11
# but WITHOUT ANY WARRANTY; without even the implied warranty of
 
12
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 
13
# GNU General Public License for more details.
 
14
#
 
15
# You should have received a copy of the GNU General Public License
 
16
# along with this program; if not, write to the Free Software
 
17
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
 
18
 
 
19
##############################################################################
 
20
#
 
21
#  This scripts creates the MySQL Server system tables.
 
22
#
 
23
#  This script try to match the shell script version as close as possible,
 
24
#  but in addition being compatible with ActiveState Perl on Windows.
 
25
#
 
26
#  All unrecognized arguments to this script are passed to mysqld.
 
27
#
 
28
#  NOTE: This script in 5.0 doesn't really match the shell script
 
29
#        version 100%, it is more close to the 5.1 version.
 
30
#
 
31
#  NOTE: This script was deliberately written to be as close to the shell
 
32
#        script as possible, to make the maintenance of both in parallel
 
33
#        easier.
 
34
#
 
35
##############################################################################
 
36
 
 
37
use File::Basename;
 
38
use Getopt::Long;
 
39
use Sys::Hostname;
 
40
use Data::Dumper;
 
41
use strict;
 
42
 
 
43
Getopt::Long::Configure("pass_through");
 
44
 
 
45
my @args;                       # Argument list filled in
 
46
 
 
47
##############################################################################
 
48
#
 
49
#  Usage information
 
50
#
 
51
##############################################################################
 
52
 
 
53
sub usage
 
54
{
 
55
  print <<EOF;
 
56
Usage: $0 [OPTIONS]
 
57
  --basedir=path       The path to the MySQL installation directory.
 
58
  --builddir=path      If using --srcdir with out-of-directory builds, you
 
59
                       will need to set this to the location of the build
 
60
                       directory where built files reside.
 
61
  --cross-bootstrap    For internal use.  Used when building the MySQL system
 
62
                       tables on a different host than the target.
 
63
  --datadir=path       The path to the MySQL data directory.
 
64
  --force              Causes mysql_install_db to run even if DNS does not
 
65
                       work.  In that case, grant table entries that normally
 
66
                       use hostnames will use IP addresses.
 
67
  --ldata=path         The path to the MySQL data directory. Same as --datadir.
 
68
  --rpm                For internal use.  This option is used by RPM files
 
69
                       during the MySQL installation process.
 
70
  --skip-name-resolve  Use IP addresses rather than hostnames when creating
 
71
                       grant table entries.  This option can be useful if
 
72
                       your DNS does not work.
 
73
  --srcdir=path        The path to the MySQL source directory.  This option
 
74
                       uses the compiled binaries and support files within the
 
75
                       source tree, useful for if you don't want to install
 
76
                       MySQL yet and just want to create the system tables.
 
77
  --user=user_name     The login username to use for running mysqld.  Files
 
78
                       and directories created by mysqld will be owned by this
 
79
                       user.  You must be root to use this option.  By default
 
80
                       mysqld runs using your current login name and files and
 
81
                       directories that it creates will be owned by you.
 
82
 
 
83
All other options are passed to the mysqld program
 
84
 
 
85
EOF
 
86
  exit 1;
 
87
}
 
88
 
 
89
##############################################################################
 
90
#
 
91
#  Parse an argument list
 
92
#
 
93
#  We only need to pass arguments through to the server if we don't
 
94
#  handle them here.  So, we collect unrecognized options (passed on
 
95
#  the command line) into the args variable.
 
96
#
 
97
##############################################################################
 
98
 
 
99
sub parse_arguments
 
100
{
 
101
  my $opt = shift;
 
102
 
 
103
  my @saved_ARGV = @ARGV;
 
104
  @ARGV = @_;                           # Set ARGV so GetOptions works
 
105
 
 
106
  my $pick_args;
 
107
  if (@ARGV and $ARGV[0] eq 'PICK-ARGS-FROM-ARGV')
 
108
  {
 
109
    $pick_args = 1;
 
110
    shift @ARGV;
 
111
  }
 
112
 
 
113
  GetOptions(
 
114
             $opt,
 
115
             "force",
 
116
             "basedir=s",
 
117
             "builddir=s",      # FIXME not documented
 
118
             "srcdir=s",
 
119
             "ldata|datadir=s",
 
120
 
 
121
             # Note that the user will be passed to mysqld so that it runs
 
122
             # as 'user' (crucial e.g. if log-bin=/some_other_path/
 
123
             # where a chown of datadir won't help)
 
124
             "user=s",
 
125
 
 
126
             "skip-name-resolve",
 
127
             "verbose",
 
128
             "rpm",
 
129
             "help",
 
130
             "defaults-file|defaults-extra-file|no-defaults:s",
 
131
 
 
132
             # Used when building the MySQL system tables on a different host than
 
133
             # the target. The platform-independent files that are created in
 
134
             # --datadir on the host can be copied to the target system.
 
135
             #
 
136
             # The most common use for this feature is in the Windows installer
 
137
             # which will take the files from datadir and include them as part of
 
138
             # the install package.  See top-level 'dist-hook' make target.
 
139
             #
 
140
             # --windows is a deprecated alias
 
141
             "cross-bootstrap|windows", # FIXME undocumented, even needed?
 
142
  ) or usage();
 
143
 
 
144
  usage() if $opt->{help};
 
145
 
 
146
  @args =  @ARGV if $pick_args;
 
147
 
 
148
  @ARGV = @saved_ARGV;                  # Set back ARGV
 
149
}
 
150
 
 
151
##############################################################################
 
152
#
 
153
#  Try to find a specific file within --basedir which can either be a binary
 
154
#  release or installed source directory and return the path.
 
155
#
 
156
##############################################################################
 
157
 
 
158
sub find_in_basedir
 
159
{
 
160
  my $opt   = shift;
 
161
  my $mode  = shift;            # "dir" or "file"
 
162
  my $files = shift;
 
163
 
 
164
  foreach my $file ( @{ref($files) ? $files : [$files]} )
 
165
  {
 
166
    foreach my $dir ( @_ )
 
167
    {
 
168
      foreach my $part ( "$file","$file.exe","release/$file.exe",
 
169
                         "debug/$file.exe","relwithdebinfo/$file.exe" )
 
170
      {
 
171
        my $path = "$opt->{basedir}/$dir/$part";
 
172
        if ( -f $path )
 
173
        {
 
174
          return $mode eq "dir" ? dirname($path) : $path;
 
175
        }
 
176
      }
 
177
    }
 
178
  }
 
179
}
 
180
 
 
181
##############################################################################
 
182
#
 
183
#  Just a function to write out an error report
 
184
#
 
185
##############################################################################
 
186
 
 
187
sub cannot_find_file
 
188
{
 
189
  my $file = shift;
 
190
 
 
191
  print "FATAL ERROR: Could not find $file\n";
 
192
  print "\n";
 
193
  print "If you compiled from source, you need to run 'make install' to\n";
 
194
  print "copy the software into the correct location ready for operation.\n";
 
195
  print "\n";
 
196
  print "If you are using a binary release, you must either be at the top\n";
 
197
  print "level of the extracted archive, or pass the --basedir option\n";
 
198
  print "pointing to that location.\n";
 
199
  print "\n";
 
200
 
 
201
  exit 1;
 
202
}
 
203
 
 
204
##############################################################################
 
205
#
 
206
#  Form a command line that can handle spaces in paths and arguments
 
207
#
 
208
##############################################################################
 
209
 
 
210
# FIXME this backslash escaping needed if using '"..."' ?
 
211
# This regexp makes sure that any special chars are quoted,
 
212
# so the arg gets passed exactly to the server.
 
213
# XXX: This is broken; true fix requires using eval and proper
 
214
# quoting of every single arg ($opt->{basedir}, $opt->{ldata}, etc.)
 
215
#  join(" ", map {s/([^\w\_\.\-])/\\$1/g}
 
216
 
 
217
sub quote_options {
 
218
  my @cmd;
 
219
  foreach my $opt ( @_ )
 
220
  {
 
221
    next unless $opt;           # If undefined or empty, just skip
 
222
    push(@cmd, "\"$opt\"");     # Quote argument
 
223
  }
 
224
  return join(" ", @cmd);
 
225
}
 
226
 
 
227
##############################################################################
 
228
#
 
229
#  Ok, let's go.  We first need to parse arguments which are required by
 
230
#  my_print_defaults so that we can execute it first, then later re-parse
 
231
#  the command line to add any extra bits that we need.
 
232
#
 
233
##############################################################################
 
234
 
 
235
my $opt = {};
 
236
parse_arguments($opt, 'PICK-ARGS-FROM-ARGV', @ARGV);
 
237
 
 
238
# ----------------------------------------------------------------------
 
239
# We can now find my_print_defaults.  This script supports:
 
240
#
 
241
#   --srcdir=path pointing to compiled source tree
 
242
#   --basedir=path pointing to installed binary location
 
243
#
 
244
# or default to compiled-in locations.
 
245
# ----------------------------------------------------------------------
 
246
 
 
247
my $print_defaults;
 
248
 
 
249
if ( $opt->{srcdir} and $opt->{basedir} )
 
250
{
 
251
  error("Specify either --basedir or --srcdir, not both");
 
252
}
 
253
if ( $opt->{srcdir} )
 
254
{
 
255
  $opt->{builddir} = $opt->{srcdir} unless $opt->{builddir};
 
256
  $print_defaults = "$opt->{builddir}/extra/my_print_defaults";
 
257
}
 
258
elsif ( $opt->{basedir} )
 
259
{
 
260
  $print_defaults = find_in_basedir($opt,"file","my_print_defaults","bin","extra");
 
261
}
 
262
else
 
263
{
 
264
  $print_defaults='@bindir@/my_print_defaults';
 
265
}
 
266
 
 
267
-x $print_defaults or -f "$print_defaults.exe"
 
268
  or cannot_find_file($print_defaults);
 
269
 
 
270
# ----------------------------------------------------------------------
 
271
# Now we can get arguments from the groups [mysqld] and [mysql_install_db]
 
272
# in the my.cfg file, then re-run to merge with command line arguments.
 
273
# ----------------------------------------------------------------------
 
274
 
 
275
my @default_options;
 
276
my $cmd = quote_options($print_defaults,$opt->{'defaults-file'},
 
277
                        "mysqld","mysql_install_db");
 
278
open(PIPE, "$cmd |") or error($opt,"can't run $cmd: $!");
 
279
while ( <PIPE> )
 
280
{
 
281
  chomp;
 
282
  next unless /\S/;
 
283
  push(@default_options, $_);
 
284
}
 
285
close PIPE;
 
286
$opt = {};                              # Reset the arguments FIXME ?
 
287
parse_arguments($opt, @default_options);
 
288
parse_arguments($opt, 'PICK-ARGS-FROM-ARGV', @ARGV);
 
289
 
 
290
# ----------------------------------------------------------------------
 
291
# Configure paths to support files
 
292
# ----------------------------------------------------------------------
 
293
 
 
294
# FIXME $extra_bindir is not used
 
295
my ($bindir,$extra_bindir,$mysqld,$pkgdatadir,$mysqld_opt,$scriptdir);
 
296
 
 
297
if ( $opt->{srcdir} )
 
298
{
 
299
  $opt->{basedir} = $opt->{builddir};
 
300
  $bindir         = "$opt->{basedir}/client";
 
301
  $extra_bindir   = "$opt->{basedir}/extra";
 
302
  $mysqld         = "$opt->{basedir}/sql/mysqld";
 
303
  $mysqld_opt     = "--language=$opt->{srcdir}/sql/share/english";
 
304
  $pkgdatadir     = "$opt->{srcdir}/scripts";
 
305
  $scriptdir      = "$opt->{srcdir}/scripts";
 
306
}
 
307
elsif ( $opt->{basedir} )
 
308
{
 
309
  $bindir         = "$opt->{basedir}/bin";
 
310
  $extra_bindir   = $bindir;
 
311
  $mysqld         = find_in_basedir($opt,"file",["mysqld-nt","mysqld"],
 
312
                                    "libexec","sbin","bin") ||  # ,"sql"
 
313
                    find_in_basedir($opt,"file","mysqld-nt",
 
314
                                  "bin");  # ,"sql"
 
315
  $pkgdatadir     = find_in_basedir($opt,"dir","fill_help_tables.sql",
 
316
                                    "share","share/mysql");  # ,"scripts"
 
317
  $scriptdir      = "$opt->{basedir}/scripts";
 
318
}
 
319
else
 
320
{
 
321
  $opt->{basedir} = '@prefix@';
 
322
  $bindir         = '@bindir@';
 
323
  $extra_bindir   = $bindir;
 
324
  $mysqld         = '@libexecdir@/mysqld';
 
325
  $pkgdatadir     = '@pkgdatadir@';
 
326
  $scriptdir      = '@scriptdir@';
 
327
}
 
328
 
 
329
unless ( $opt->{ldata} )
 
330
{
 
331
  $opt->{ldata} = '@localstatedir@';
 
332
}
 
333
 
 
334
if ( $opt->{srcdir} )
 
335
{
 
336
  $pkgdatadir = "$opt->{srcdir}/scripts";
 
337
}
 
338
 
 
339
# ----------------------------------------------------------------------
 
340
# Set up paths to SQL scripts required for bootstrap
 
341
# ----------------------------------------------------------------------
 
342
 
 
343
my $fill_help_tables     = "$pkgdatadir/fill_help_tables.sql";
 
344
my $create_system_tables = "$pkgdatadir/mysql_system_tables.sql";
 
345
my $fill_system_tables   = "$pkgdatadir/mysql_system_tables_data.sql";
 
346
 
 
347
foreach my $f ( $fill_help_tables,$create_system_tables,$fill_system_tables )
 
348
{
 
349
  -f $f or cannot_find_file($f);
 
350
}
 
351
 
 
352
-x $mysqld or -f "$mysqld.exe" or cannot_find_file($mysqld);
 
353
# Try to determine the hostname
 
354
my $hostname = hostname();
 
355
 
 
356
# ----------------------------------------------------------------------
 
357
# Check if hostname is valid
 
358
# ----------------------------------------------------------------------
 
359
 
 
360
my $resolved;
 
361
if ( !$opt->{'cross-bootstrap'} and !$opt->{rpm} and !$opt->{force} )
 
362
{
 
363
  my $resolveip;
 
364
 
 
365
  $resolved = `$resolveip $hostname 2>&1`;
 
366
  if ( $? != 0 )
 
367
  {
 
368
    $resolved=`$resolveip localhost 2>&1`;
 
369
    if ( $? != 0 )
 
370
    {
 
371
      error($opt,
 
372
            "Neither host '$hostname' nor 'localhost' could be looked up with",
 
373
            "$bindir/resolveip",
 
374
            "Please configure the 'hostname' command to return a correct",
 
375
            "hostname.",
 
376
            "If you want to solve this at a later stage, restart this script",
 
377
            "with the --force option");
 
378
    }
 
379
    warning($opt,
 
380
            "The host '$hostname' could not be looked up with resolveip.",
 
381
            "This probably means that your libc libraries are not 100 % compatible",
 
382
            "with this binary MySQL version. The MySQL daemon, mysqld, should work",
 
383
            "normally with the exception that host name resolving will not work.",
 
384
            "This means that you should use IP addresses instead of hostnames",
 
385
            "when specifying MySQL privileges !");
 
386
  }
 
387
}
 
388
 
 
389
# FIXME what does this really mean....
 
390
if ( $opt->{'skip-name-resolve'} and $resolved and $resolved =~ /\s/ )
 
391
{
 
392
  $hostname = (split(' ', $resolved))[5];
 
393
}
 
394
 
 
395
# ----------------------------------------------------------------------
 
396
# Create database directories mysql & test
 
397
# ----------------------------------------------------------------------
 
398
 
 
399
foreach my $dir ( $opt->{ldata}, "$opt->{ldata}/mysql", "$opt->{ldata}/test" )
 
400
{
 
401
  # FIXME not really the same as original "mkdir -p", but ok?
 
402
  mkdir($dir, 0700) unless -d $dir;
 
403
  chown($opt->{user}, $dir) if -w "/" and !$opt->{user};
 
404
}
 
405
 
 
406
push(@args, "--user=$opt->{user}") if $opt->{user};
 
407
 
 
408
# ----------------------------------------------------------------------
 
409
# Configure mysqld command line
 
410
# ----------------------------------------------------------------------
 
411
 
 
412
# FIXME use --init-file instead of --bootstrap ?!
 
413
 
 
414
my $mysqld_bootstrap = $ENV{MYSQLD_BOOTSTRAP} || $mysqld;
 
415
my $mysqld_install_cmd_line = quote_options($mysqld_bootstrap,
 
416
                                            $opt->{'defaults-file'},
 
417
                                            $mysqld_opt,
 
418
                                            "--bootstrap",
 
419
                                            "--basedir=$opt->{basedir}",
 
420
                                            "--datadir=$opt->{ldata}",
 
421
                                            "--skip-innodb",
 
422
                                            "--skip-bdb",
 
423
                                            "--skip-ndbcluster",
 
424
                                            "--max_allowed_packet=8M",
 
425
                                            "--net_buffer_length=16K",
 
426
                                            @args,
 
427
                                          );
 
428
 
 
429
# ----------------------------------------------------------------------
 
430
# Create the system and help tables by passing them to "mysqld --bootstrap"
 
431
# ----------------------------------------------------------------------
 
432
 
 
433
report_verbose_wait($opt,"Installing MySQL system tables...");
 
434
 
 
435
open(SQL, $create_system_tables)
 
436
  or error($opt,"can't open $create_system_tables for reading: $!");
 
437
# FIXME  > /dev/null ?
 
438
if ( open(PIPE, "| $mysqld_install_cmd_line") )
 
439
{
 
440
  print PIPE "use mysql;\n";
 
441
  while ( <SQL> )
 
442
  {
 
443
    # When doing a "cross bootstrap" install, no reference to the current
 
444
    # host should be added to the system tables.  So we filter out any
 
445
    # lines which contain the current host name.
 
446
    next if $opt->{'cross-bootstrap'} and /\@current_hostname/;
 
447
 
 
448
    print PIPE $_;
 
449
  }
 
450
  close PIPE;
 
451
  close SQL;
 
452
 
 
453
  report_verbose($opt,"OK");
 
454
 
 
455
  # ----------------------------------------------------------------------
 
456
  # Pipe fill_help_tables.sql to "mysqld --bootstrap"
 
457
  # ----------------------------------------------------------------------
 
458
 
 
459
  report_verbose_wait($opt,"Filling help tables...");
 
460
  open(SQL, $fill_help_tables)
 
461
    or error($opt,"can't open $fill_help_tables for reading: $!");
 
462
  # FIXME  > /dev/null ?
 
463
  if ( open(PIPE, "| $mysqld_install_cmd_line") )
 
464
  {
 
465
    print PIPE "use mysql;\n";
 
466
    while ( <SQL> )
 
467
    {
 
468
      print PIPE $_;
 
469
    }
 
470
    close PIPE;
 
471
    close SQL;
 
472
 
 
473
    report_verbose($opt,"OK");
 
474
  }
 
475
  else
 
476
  {
 
477
    warning($opt,"HELP FILES ARE NOT COMPLETELY INSTALLED!",
 
478
                 "The \"HELP\" command might not work properly");
 
479
  }
 
480
 
 
481
  report_verbose($opt,"To start mysqld at boot time you have to copy",
 
482
                      "support-files/mysql.server to the right place " .
 
483
                      "for your system");
 
484
 
 
485
  if ( !$opt->{'cross-bootstrap'} )
 
486
  {
 
487
    # This is not a true installation on a running system.  The end user must
 
488
    # set a password after installing the data files on the real host system.
 
489
    # At this point, there is no end user, so it does not make sense to print
 
490
    # this reminder.
 
491
    report($opt,
 
492
           "PLEASE REMEMBER TO SET A PASSWORD FOR THE MySQL root USER !",
 
493
           "To do so, start the server, then issue the following commands:",
 
494
           "",
 
495
           "  $bindir/mysqladmin -u root password 'new-password'",
 
496
           "  $bindir/mysqladmin -u root -h $hostname password 'new-password'",
 
497
           "",
 
498
           "Alternatively you can run:",
 
499
           "",
 
500
           "  $bindir/mysql_secure_installation",
 
501
           "",
 
502
           "which will also give you the option of removing the test",
 
503
           "databases and anonymous user created by default.  This is",
 
504
           "strongly recommended for production servers.",
 
505
           "",
 
506
           "See the manual for more instructions.");
 
507
 
 
508
    if ( !$opt->{rpm} )
 
509
    {
 
510
      report($opt,
 
511
             "You can start the MySQL daemon with:",
 
512
             "",
 
513
             "  cd " . '@prefix@' . " ; $bindir/mysqld_safe &",
 
514
             "",
 
515
             "You can test the MySQL daemon with mysql-test-run.pl",
 
516
             "",
 
517
             "  cd mysql-test ; perl mysql-test-run.pl");
 
518
    }
 
519
    report($opt,
 
520
           "Please report any problems with the " . '@scriptdir@' . "/mysqlbug script!",
 
521
           "",
 
522
           "The latest information about MySQL is available on the web at",
 
523
           "",
 
524
           "  http://www.mysql.com",
 
525
           "",
 
526
           "Support MySQL by buying support/licenses at http://shop.mysql.com");
 
527
  }
 
528
  exit 0
 
529
}
 
530
else
 
531
{
 
532
  error($opt,
 
533
        "Installation of system tables failed!",
 
534
         "",
 
535
        "Examine the logs in $opt->{ldata} for more information.",
 
536
        "You can try to start the mysqld daemon with:",
 
537
        "$mysqld --skip-grant &",
 
538
        "and use the command line tool",
 
539
        "$bindir/mysql to connect to the mysql",
 
540
        "database and look at the grant tables:",
 
541
        "",
 
542
        "shell> $bindir/mysql -u root mysql",
 
543
        "mysql> show tables",
 
544
        "",
 
545
        "Try 'mysqld --help' if you have problems with paths. Using --log",
 
546
        "gives you a log in $opt->{ldata} that may be helpful.",
 
547
        "",
 
548
        "The latest information about MySQL is available on the web at",
 
549
        "http://www.mysql.com",
 
550
        "Please consult the MySQL manual section: 'Problems running mysql_install_db',",
 
551
        "and the manual section that describes problems on your OS.",
 
552
        "Another information source is the MySQL email archive.",
 
553
        "Please check all of the above before mailing us!",
 
554
        "And if you do mail us, you MUST use the " . '@scriptdir@' . "/mysqlbug script!")
 
555
}
 
556
 
 
557
##############################################################################
 
558
#
 
559
#  Misc
 
560
#
 
561
##############################################################################
 
562
 
 
563
sub report_verbose
 
564
{
 
565
  my $opt  = shift;
 
566
  my $text = shift;
 
567
 
 
568
  report_verbose_wait($opt, $text, @_);
 
569
  print "\n\n";
 
570
}
 
571
 
 
572
sub report_verbose_wait
 
573
{
 
574
  my $opt  = shift;
 
575
  my $text = shift;
 
576
 
 
577
  if ( $opt->{verbose} or (!$opt->{rpm} and !$opt->{'cross-bootstrap'}) )
 
578
  {
 
579
    print "$text";
 
580
    map {print "\n$_"} @_;
 
581
  }
 
582
}
 
583
 
 
584
sub report
 
585
{
 
586
  my $opt  = shift;
 
587
  my $text = shift;
 
588
 
 
589
  print "$text\n";
 
590
  map {print "$_\n"} @_;
 
591
  print "\n";
 
592
}
 
593
 
 
594
sub error
 
595
{
 
596
  my $opt  = shift;
 
597
  my $text = shift;
 
598
 
 
599
  print "FATAL ERROR: $text\n";
 
600
  map {print "$_\n"} @_;
 
601
  exit 1;
 
602
}
 
603
 
 
604
sub warning
 
605
{
 
606
  my $opt  = shift;
 
607
  my $text = shift;
 
608
 
 
609
  print "WARNING: $text\n";
 
610
  map {print "$_\n"} @_;
 
611
  print "\n";
 
612
}