mysqld_multi
上传用户:dzyhzl
上传日期:2019-04-29
资源大小:56270k
文件大小:19k
源码类别:

模拟服务器

开发平台:

C/C++

  1. #!/usr/bin/perl
  2. use Getopt::Long;
  3. use POSIX qw(strftime);
  4. $|=1;
  5. $VER="2.2";
  6. $opt_config_file   = undef();
  7. $opt_example       = 0;
  8. $opt_help          = 0;
  9. $opt_log           = "/tmp/mysqld_multi.log";
  10. $opt_mysqladmin    = "/usr/local/mysql/bin/mysqladmin";
  11. $opt_mysqld        = "/usr/local/mysql/libexec/mysqld";
  12. $opt_no_log        = 0;
  13. $opt_password      = undef();
  14. $opt_tcp_ip        = 0;
  15. $opt_user          = "root";
  16. $opt_version       = 0;
  17. my ($mysqld, $mysqladmin, $groupids, $homedir, $my_progname);
  18. $homedir = $ENV{HOME};
  19. $my_progname = $0;
  20. $my_progname =~ s/.*[/]//;
  21. main();
  22. ####
  23. #### main sub routine
  24. ####
  25. sub main
  26. {
  27.   my ($flag_exit);
  28.   if (!defined(my_which(my_print_defaults)))
  29.   {
  30.     # We can't throw out yet, since --version, --help, or --example may
  31.     # have been given
  32.     print "WARNING! my_print_defaults command not found!n";
  33.     print "Please make sure you have this command available andn";
  34.     print "in your path. The command is available from the latestn";
  35.     print "MySQL distribution.n";
  36.   }
  37.   my @defops = `my_print_defaults mysqld_multi`;
  38.   chop @defops;
  39.   splice @ARGV, 0, 0, @defops;
  40.   GetOptions("help","example","version","mysqld=s","mysqladmin=s",
  41.      "config-file=s","user=s","password=s","log=s","no-log","tcp-ip")
  42.   || die "Wrong option! See $my_progname --help for detailed information!n";
  43.   $groupids = $ARGV[1];
  44.   if ($opt_version)
  45.   {
  46.     print "$my_progname version $VER by Jani Tolonenn";
  47.     exit(0);
  48.   }
  49.   example() if ($opt_example);
  50.   if (!defined(($mysqld = my_which($opt_mysqld))))
  51.   {
  52.     print "Couldn't find the mysqld binary! Tried: $opt_mysqldn";
  53.     $flag_exit=1;
  54.   }
  55.   if (!defined(($mysqladmin = my_which($opt_mysqladmin))))
  56.   {
  57.     print "Couldn't find the mysqladmin binary! Tried: $opt_mysqladminn";
  58.     $flag_exit=1;
  59.   }
  60.   usage() if ($opt_help);
  61.   if ($flag_exit)
  62.   {
  63.     print "Error with an option, see $my_progname --help for more info!n";
  64.     exit(1);
  65.   }
  66.   if (!defined(my_which(my_print_defaults)))
  67.   {
  68.     print "ABORT: Can't find command 'my_print_defaults'!n";
  69.     print "This command is available from the latest MySQLn";
  70.     print "distribution. Please make sure you have the commandn";
  71.     print "in your PATH.n";
  72.     exit(1);
  73.   }
  74.   usage() if (!defined($ARGV[0]) ||
  75.       ($ARGV[0] ne 'start' && $ARGV[0] ne 'START' &&
  76.        $ARGV[0] ne 'stop' && $ARGV[0] ne 'STOP' &&
  77.        $ARGV[0] ne 'report' && $ARGV[0] ne 'REPORT'));
  78.   if (!$opt_no_log)
  79.   {
  80.     w2log("$my_progname log file version $VER; run: ",
  81.   "$opt_log", 1, 0);
  82.   }
  83.   else
  84.   {
  85.     print "$my_progname log file version $VER; run: ";
  86.     print strftime "%a %b %e %H:%M:%S %Y", localtime;
  87.     print "n";
  88.   }
  89.   if ($ARGV[0] eq 'report' || $ARGV[0] eq 'REPORT')
  90.   {
  91.     report_mysqlds();
  92.   }
  93.   elsif ($ARGV[0] eq 'start' || $ARGV[0] eq 'START')
  94.   {
  95.     start_mysqlds();
  96.   }
  97.   else
  98.   {
  99.     stop_mysqlds();
  100.   }
  101. }
  102. ####
  103. #### Report living and not running MySQL servers
  104. ####
  105. sub report_mysqlds
  106. {
  107.   my (@groups, $com, $i, @options, $j, $pec);
  108.   print "Reporting MySQL serversn";
  109.   if (!$opt_no_log)
  110.   {
  111.     w2log("nReporting MySQL servers","$opt_log",0,0);
  112.   }
  113.   @groups = &find_groups($groupids);
  114.   for ($i = 0; defined($groups[$i]); $i++)
  115.   {
  116.     $com = "my_print_defaults";
  117.     $com.= defined($opt_config_file) ? " --config-file=$opt_config_file" : "";
  118.     $com.= " $groups[$i]";
  119.     @options = `$com`;
  120.     chop @options;
  121.     $com = "$mysqladmin -u $opt_user";
  122.     $com.= defined($opt_password) ? " -p$opt_password" : "";
  123.     $com.= $opt_tcp_ip ? " -h 127.0.0.1" : "";
  124.     for ($j = 0; defined($options[$j]); $j++)
  125.     {
  126.       if ((($options[$j] =~ m/^(--socket=)(.*)$/) && !$opt_tcp_ip) ||
  127.   ($options[$j] =~ m/^(--port=)(.*)$/))
  128.       {
  129. $com.= " $options[$j]";
  130.       }
  131.     }
  132.     $com.= " ping >> /dev/null 2>&1";
  133.     system($com);
  134.     $pec = $? >> 8;
  135.     if ($pec)
  136.     {
  137.       print "MySQL server from group: $groups[$i] is not runningn";
  138.       if (!$opt_no_log)
  139.       {
  140. w2log("MySQL server from group: $groups[$i] is not running",
  141.       "$opt_log", 0, 0);
  142.       }
  143.     }
  144.     else
  145.     {
  146.       print "MySQL server from group: $groups[$i] is runningn";
  147.       if (!$opt_no_log)
  148.       {
  149. w2log("MySQL server from group: $groups[$i] is running",
  150.       "$opt_log", 0, 0);
  151.       }
  152.     }
  153.   }
  154.   if (!$i)
  155.   {
  156.     print "No groups to be reported (check your GNRs)n";
  157.     if (!$opt_no_log)
  158.     {
  159.       w2log("No groups to be reported (check your GNRs)", "$opt_log", 0, 0);
  160.     }
  161.   }
  162. }
  163. ####
  164. #### start multiple servers
  165. ####
  166. sub start_mysqlds()
  167. {
  168.   my (@groups, $com, $i, @options, $j);
  169.   if (!$opt_no_log)
  170.   {
  171.     w2log("nStarting MySQL serversn","$opt_log",0,0);
  172.   }
  173.   else
  174.   {
  175.     print "nStarting MySQL serversn";
  176.   }
  177.   @groups = &find_groups($groupids);
  178.   for ($i = 0; defined($groups[$i]); $i++)
  179.   {
  180.     $com = "my_print_defaults";
  181.     $com.= defined($opt_config_file) ? " --config-file=$opt_config_file" : "";
  182.     $com.= " $groups[$i]";
  183.     @options = `$com`;
  184.     chop @options;
  185.     $com = "$mysqld";
  186.     for ($j = 0; defined($options[$j]); $j++)
  187.     {
  188.       $com.= " $options[$j]";
  189.     }
  190.     $com.= " >> $opt_log 2>&1" if (!$opt_no_log);
  191.     $com.= " &";
  192.     system($com);
  193.   }
  194.   if (!$i && !$opt_no_log)
  195.   {
  196.     w2log("No MySQL servers to be started (check your GNRs)",
  197.   "$opt_log", 0, 0);
  198.   }
  199. }
  200. ####
  201. #### stop multiple servers
  202. ####
  203. sub stop_mysqlds()
  204. {
  205.   my (@groups, $com, $i, @options, $j);
  206.   if (!$opt_no_log)
  207.   {
  208.     w2log("nStopping MySQL serversn","$opt_log",0,0);
  209.   }
  210.   else
  211.   {
  212.     print "nStopping MySQL serversn";
  213.   }
  214.   @groups = &find_groups($groupids);
  215.   for ($i = 0; defined($groups[$i]); $i++)
  216.   {
  217.     $com = "my_print_defaults";
  218.     $com.= defined($opt_config_file) ? " --config-file=$opt_config_file" : "";
  219.     $com.= " $groups[$i]";
  220.     @options = `$com`;
  221.     chop @options;
  222.     $com = "$mysqladmin -u $opt_user";
  223.     $com.= defined($opt_password) ? " -p$opt_password" : "";
  224.     $com.= $opt_tcp_ip ? " -h 127.0.0.1" : "";
  225.     for ($j = 0; defined($options[$j]); $j++)
  226.     {
  227.       if ((($options[$j] =~ m/^(--socket=)(.*)$/) && !$opt_tcp_ip) ||
  228.   ($options[$j] =~ m/^(--port=)(.*)$/))
  229.       {
  230. $com.= " $options[$j]";
  231.       }
  232.     }
  233.     $com.= " shutdown";
  234.     $com.= " >> $opt_log 2>&1" if (!$opt_no_log);
  235.     $com.= " &";
  236.     system($com);
  237.   }
  238.   if (!$i && !$opt_no_log)
  239.   {
  240.     w2log("No MySQL servers to be stopped (check your GNRs)",
  241.   "$opt_log", 0, 0);
  242.   }
  243. }
  244. ####
  245. #### Find groups. Takes the valid group numbers as an argument, parses
  246. #### them, puts them in the ascending order, removes duplicates and
  247. #### returns the wanted groups accordingly.
  248. ####
  249. sub find_groups
  250. {
  251.   my ($raw_gids) = @_;
  252.   my (@groups, @data, @tmp, $line, $i, $k, @pre_gids, @gids, @tmp2,
  253.       $prev_value);
  254.   # Read the lines from the config file to variable 'data'
  255.   if (defined($opt_config_file))
  256.   {
  257.     open(MY_CNF, "<$opt_config_file") && (@data=<MY_CNF>) && close(MY_CNF);
  258.   }
  259.   else
  260.   {
  261.     if (-f "/etc/my.cnf" && -r "/etc/my.cnf")
  262.     {
  263.       open(MY_CNF, "</etc/my.cnf") && (@tmp=<MY_CNF>) && close(MY_CNF);
  264.     }
  265.     for ($i = 0; ($line = shift @tmp); $i++)
  266.     {
  267.       $data[$i] = $line;
  268.     }
  269.     if (-f "$homedir/.my.cnf" && -r "$homedir/.my.cnf")
  270.     {
  271.       open(MY_CNF, "<$homedir/.my.cnf") && (@tmp=<MY_CNF>) && close(MY_CNF);
  272.     }
  273.     for (; ($line = shift @tmp); $i++)
  274.     {
  275.       $data[$i] = $line;
  276.     }
  277.   }
  278.   chop @data;
  279.   # Make a list of the wanted group ids
  280.   if (defined($raw_gids))
  281.   {
  282.     @pre_gids = split(',', $raw_gids);
  283.   }
  284.   if (defined($raw_gids))
  285.   {
  286.     for ($i = 0, $j = 0; defined($pre_gids[$i]); $i++)
  287.     {
  288.       if ($pre_gids[$i] =~ m/^(d+)$/)
  289.       {
  290. $gids[$j] = $1;
  291. $j++;
  292.       }
  293.       elsif ($pre_gids[$i] =~ m/^(d+)(-)(d+)$/)
  294.       {
  295. for ($k = $1; $k <= $3; $k++)
  296. {
  297.   $gids[$j] = $k;
  298.   $j++;
  299. }
  300.       }
  301.       else
  302.       {
  303. print "ABORT: Bad GNR: $pre_gids[$i] See $my_progname --helpn";
  304. exit(1);
  305.       }
  306.     }
  307.   }
  308.   # Sort the list of gids numerically in ascending order
  309.   @gids = sort {$a <=> $b} @gids;
  310.   # Remove non-positive integers and duplicates
  311.   for ($i = 0, $j = 0; defined($gids[$i]); $i++)
  312.   {
  313.     next if ($gids[$i] <= 0);
  314.     if (!$i || $prev_value != $gids[$i])
  315.     {
  316.       $tmp2[$j] = $gids[$i];
  317.       $j++;
  318.     }
  319.     $prev_value = $gids[$i];
  320.   }
  321.   @gids = @tmp2;
  322.   # Find and return the wanted groups
  323.   for ($i = 0, $j = 0; defined($data[$i]); $i++)
  324.   {
  325.     if ($data[$i] =~ m/^(s*[s*)(mysqld)(d+)(s*]s*)$/)
  326.     {
  327.       if (defined($raw_gids))
  328.       {
  329. for ($k = 0; defined($gids[$k]); $k++)
  330. {
  331.   if ($gids[$k] == $3)
  332.   {
  333.     $groups[$j] = $2 . $3;
  334.     $j++;
  335.   }
  336. }
  337.       }
  338.       else
  339.       {
  340. $groups[$j] = $2 . $3;
  341. $j++;
  342.       }
  343.     }
  344.   }
  345.   return @groups;
  346. }
  347. ####
  348. #### w2log: Write to a logfile.
  349. #### 1.arg: append to the log file (given string, or from a file. if a file,
  350. ####        file will be read from $opt_logdir)
  351. #### 2.arg: logfile -name (w2log assumes that the logfile is in $opt_logdir).
  352. #### 3.arg. 0 | 1, if true, print current date to the logfile. 3. arg will
  353. ####        be ignored, if 1. arg is a file.
  354. #### 4.arg. 0 | 1, if true, first argument is a file, else a string
  355. ####
  356. sub w2log
  357. {
  358.   my ($msg, $file, $date_flag, $is_file)= @_;
  359.   my (@data);
  360.   open (LOGFILE, ">>$opt_log")
  361.     or die "FATAL: w2log: Couldn't open log file: $opt_logn";
  362.   if ($is_file)
  363.   {
  364.     open (FROMFILE, "<$msg") && (@data=<FROMFILE>) &&
  365.       close(FROMFILE)
  366. or die "FATAL: w2log: Couldn't open file: $msgn";
  367.     foreach my $line (@data)
  368.     {
  369.       print LOGFILE "$line";
  370.     }
  371.   }
  372.   else
  373.   {
  374.     print LOGFILE "$msg";
  375.     print LOGFILE strftime "%a %b %e %H:%M:%S %Y", localtime if ($date_flag);
  376.     print LOGFILE "n";
  377.   }
  378.   close (LOGFILE);
  379.   return;
  380. }
  381. ####
  382. #### my_which is used, because we can't assume that every system has the
  383. #### which -command. my_which can take only one argument at a time.
  384. #### Return values: requested system command with the first found path,
  385. #### or undefined, if not found.
  386. ####
  387. sub my_which
  388. {
  389.   my ($command) = @_;
  390.   my (@paths, $path);
  391.   return $command if (-f $command && -x $command);
  392.   @paths = split(':', $ENV{'PATH'});
  393.   foreach $path (@paths)
  394.   {
  395.     $path .= "/$command";
  396.     return $path if (-f $path && -x $path);
  397.   }
  398.   return undef();
  399. }
  400. ####
  401. #### example
  402. ####
  403. sub example
  404. {
  405.   print <<EOF;
  406. # This is an example of a my.cnf file on behalf of $my_progname.
  407. # This file should probably be in your home dir (~/.my.cnf) or /etc/my.cnf
  408. # Version $VER by Jani Tolonen
  409. # NOTES:
  410. # 1.Make sure that the MySQL user, who is stopping the mysqld services (e.g
  411. #   using the mysqladmin) have the same password and username for all the
  412. #   data directories accessed (to the 'mysql' database) And make sure that
  413. #   the user has the 'Shutdown_priv' privilege! If you have many data-
  414. #   directories and many different 'mysql' databases with different passwords
  415. #   for the MySQL 'root' user, you may want to create a common 'multi_admin'
  416. #   user for each using the same password (see below). Example how to do it:
  417. #   shell> mysql -u root -S /tmp/mysql.sock -proot_password -e
  418. #   "GRANT SHUTDOWN ON *.* TO multi_admin@localhost IDENTIFIED BY 'multipass'"
  419. #   You will have to do the above for each mysqld running in each data
  420. #   directory, that you have (just change the socket, -S=...)
  421. #   See more detailed information from chapter:
  422. #   '6 The MySQL Access Privilege System' from the MySQL manual.
  423. # 2.pid-file is very important, if you are using safe_mysqld to start mysqld
  424. #   (e.g. --mysqld=safe_mysqld) Every mysqld should have it's own pid-file.
  425. #   The advantage using safe_mysqld instead of mysqld directly here is, that
  426. #   safe_mysqld 'guards' every mysqld process and will restart it, if mysqld
  427. #   process fails due to signal kill -9, or similar. (Like segmentation fault,
  428. #   which MySQL should never do, of course ;) Please note that safe_mysqld
  429. #   script may require that you start it from a certain place. This means that
  430. #   you may have to CD to a certain directory, before you start the
  431. #   mysqld_multi. If you have problems starting, please see the script.
  432. #   Check especially the lines:
  433. #   --------------------------------------------------------------------------
  434. #   MY_PWD=`pwd`
  435. #   Check if we are starting this relative (for the binary release)
  436. #   if test -d $MY_PWD/data/mysql -a -f ./share/mysql/english/errmsg.sys -a 
  437. #   -x ./bin/mysqld
  438. #   --------------------------------------------------------------------------
  439. #   The above test should be successful, or you may encounter problems.
  440. # 3.Beware of the dangers starting multiple mysqlds in the same data directory.
  441. #   Use separate data directories, unless you *KNOW* what you are doing!
  442. # 4.The socket file and the TCP/IP port must be different for every mysqld.
  443. # 5.The first and fifth mysqld was intentionally left out from the example.
  444. #   You may have 'gaps' in the config file. This gives you more flexibility.
  445. #   The order in which the mysqlds are started or stopped depends on the order
  446. #   in which they appear in the config file.
  447. # 6.When you want to refer to a certain group with GNR with this program,
  448. #   just use the number in the end of the group name ( [mysqld# <== )
  449. # 7.You may want to use option '--user' for mysqld, but in order to do this
  450. #   you need to be root when you start this script. Having the option
  451. #   in the config file doesn't matter; you will just get a warning, if you are
  452. #   not the superuser and the mysqlds are started under *your* unix account.
  453. #   IMPORTANT: Make sure that the pid-file and the data directory are
  454. #   read+write(+execute for the latter one) accessible for *THAT* UNIX user,
  455. #   who the specific mysqld process is started as. *DON'T* use the UNIX root
  456. #   account for this, unless you *KNOW* what you are doing!
  457. # 8.MOST IMPORTANT: Make sure that you understand the meanings of the options
  458. #   that are passed to the mysqlds and why *WOULD YOU WANT* to have separate
  459. #   mysqld processes. Starting multiple mysqlds in one data directory *WON'T*
  460. #   give you extra performance in a threaded system!
  461. #
  462. [mysqld_multi]
  463. mysqld     = /usr/local/mysql/bin/safe_mysqld
  464. mysqladmin = /usr/local/mysql/bin/mysqladmin
  465. user       = multi_admin
  466. password   = multipass
  467. [mysqld2]
  468. socket     = /tmp/mysql.sock2
  469. port       = 3307
  470. pid-file   = /usr/local/mysql/var2/hostname.pid2
  471. datadir    = /usr/local/mysql/var2
  472. language   = /usr/local/mysql/share/mysql/english
  473. user       = john
  474. [mysqld3]
  475. socket     = /tmp/mysql.sock3
  476. port       = 3308
  477. pid-file   = /usr/local/mysql/var3/hostname.pid3
  478. datadir    = /usr/local/mysql/var3
  479. language   = /usr/local/mysql/share/mysql/swedish
  480. user       = monty
  481. [mysqld4]
  482. socket     = /tmp/mysql.sock4
  483. port       = 3309
  484. pid-file   = /usr/local/mysql/var4/hostname.pid4
  485. datadir    = /usr/local/mysql/var4
  486. language   = /usr/local/mysql/share/mysql/estonia
  487. user       = tonu
  488.  
  489. [mysqld6]
  490. socket     = /tmp/mysql.sock6
  491. port       = 3311
  492. pid-file   = /usr/local/mysql/var6/hostname.pid6
  493. datadir    = /usr/local/mysql/var6
  494. language   = /usr/local/mysql/share/mysql/japanese
  495. user       = jani
  496. EOF
  497.   exit(0);
  498. }
  499. ####
  500. #### usage
  501. ####
  502. sub usage
  503. {
  504.   print <<EOF;
  505. $my_progname version $VER by Jani Tolonen
  506. This software comes with ABSOLUTELY NO WARRANTY. This is free software,
  507. and you are welcome to modify and redistribute it under the GPL license.
  508. Description:
  509. $my_progname can be used to start, or stop any number of separate
  510. mysqld processes running in different TCP/IP ports and UNIX sockets.
  511. This program can read group [mysqld_multi] from my.cnf file.
  512. You may want to put options mysqld=... and mysqladmin=... there.
  513. The program will search for group(s) named [mysqld#] from my.cnf (or
  514. the given --config-file=...), where # can be any positive number
  515. starting from 1. These groups should be the same as the usual [mysqld]
  516. group (e.g. options to mysqld, see MySQL manual for detailed
  517. information about this group), but with those port, socket
  518. etc. options that are wanted for each separate mysqld processes. The
  519. number in the group name has another function; it can be used for
  520. starting, stopping, or reporting some specific mysqld servers with
  521. this program. See the usage and options below for more information.
  522. Usage: $my_progname [OPTIONS] {start|stop|report} [GNR,GNR,GNR...]
  523. or     $my_progname [OPTIONS] {start|stop|report} [GNR-GNR,GNR,GNR-GNR,...]
  524. The GNR above means the group number. You can start, stop or report
  525. any GNR, or several of them at the same time. (See --example) The GNRs
  526. list can be comma separated, or a dash combined, of which the latter
  527. means that all the GNRs between GNR1-GNR2 will be affected. Without
  528. GNR argument all the found groups will be either started, stopped, or
  529. reported. Note that you must not have any white spaces in the GNR
  530. list. Anything after a white space are ignored.
  531. Options:
  532. --config-file=...  Alternative config file. NOTE: This will not affect
  533.                    this program's own options (group [mysqld_multi]),
  534.                    but only groups [mysqld#]. Without this option everything
  535.                    will be searched from the ordinary my.cnf file.
  536.                    Using: $opt_config_file
  537. --example          Give an example of a config file. (PLEASE DO CHECK THIS!)
  538. --help             Print this help and exit.
  539. --log=...          Log file. Full path to and the name for the log file. NOTE:
  540.                    If the file exists, everything will be appended.
  541.                    Using: $opt_log
  542. --mysqladmin=...   mysqladmin binary to be used for a server shutdown.
  543.                    Using: $mysqladmin
  544. --mysqld=...       mysqld binary to be used. Note that you can give safe_mysqld
  545.                    to this option also. The options are passed to mysqld. Just
  546.                    make sure you have mysqld in your PATH or fix safe_mysqld.
  547.                    Using: $mysqld
  548. --no-log           Print to stdout instead of the log file. By default the log
  549.                    file is turned on.
  550. --password=...     Password for user for mysqladmin.
  551. --tcp-ip           Connect to the MySQL server(s) via the TCP/IP port instead
  552.                    of the UNIX socket. This affects stopping and reporting.
  553.                    If a socket file is missing, the server may still be
  554.                    running, but can be accessed only via the TCP/IP port.
  555.                    By default connecting is done via the UNIX socket.
  556. --user=...         MySQL user for mysqladmin. Using: $opt_user
  557. --version          Print the version number and exit.
  558. EOF
  559.   exit(0);
  560. }