| 1 | [[PageOutline]] |
| 2 | |
| 3 | = fork child process on CGI = |
| 4 | * [http://www.bioinfo.jp/tips.html Tips (CGI, Perl, Unix and etc.)] |
| 5 | * [http://www.tohoho-web.com/lng/199910/99100140.htm CGI-Perlのforkで子供を置き去りにするには?] |
| 6 | * [http://www.stackasterisk.jp/tech/program/perl04_02.jsp Perl第4回:PerlTips(バックグラウンドで処理を実行)] |
| 7 | |
| 8 | * sample code |
| 9 | * Apache2 mpm-worker + Perl 5.10 on x64 |
| 10 | {{{ |
| 11 | #!perl |
| 12 | #! /usr/bin/perl -w |
| 13 | |
| 14 | my $pid = fork; |
| 15 | if (! defined $pid) { |
| 16 | # cannot fork |
| 17 | die("cannot fork: $!"); |
| 18 | } |
| 19 | |
| 20 | if ($pid) { |
| 21 | # parent process |
| 22 | print "Content-type: text/html;\n\n"; |
| 23 | |
| 24 | # output something |
| 25 | print "parent: $$ / child: $pid\n"; |
| 26 | |
| 27 | close(STDIN); |
| 28 | close(STDOUT); |
| 29 | exit; |
| 30 | } |
| 31 | else { |
| 32 | # child proccess |
| 33 | close(STDIN); |
| 34 | close(STDOUT); |
| 35 | |
| 36 | # long long proccess ... |
| 37 | sleep(60); |
| 38 | } |
| 39 | }}} |
| 40 | * Apache2 mpm-prefork + Perl 5.10 on x64 |
| 41 | {{{ |
| 42 | #!perl |
| 43 | #! /usr/bin/perl -w |
| 44 | |
| 45 | my $pid = fork; |
| 46 | if (! defined $pid) { |
| 47 | # cannot fork |
| 48 | die("cannot fork: $!"); |
| 49 | } |
| 50 | |
| 51 | if ($pid) { |
| 52 | # parent process |
| 53 | print "Content-type: text/html;\n\n"; |
| 54 | |
| 55 | print "parent: $$ / child: $pid\n"; |
| 56 | } |
| 57 | else { |
| 58 | # child proccess |
| 59 | close(STDOUT); |
| 60 | |
| 61 | sleep(60); |
| 62 | } |
| 63 | }}} |
| 64 | * 子プロセスでclose(STDOUT)が必要とだけ書かれている文献が多いが、Apache2 mpm-worker環境では親子ともSTDINも閉じないと親プロセスは終了しなかった。 |
| 65 | * しかし、テストのため一度mpm-preforkに切り替えた後、再度mpm-workerに戻した後は何故かpreforkと同じ挙動(STDINを閉じなくてもOK)を示すようになった。 |
| 66 | * apache2 -V |
| 67 | {{{ |
| 68 | Server version: Apache/2.2.14 (Ubuntu) |
| 69 | Server built: Apr 13 2010 20:22:19 |
| 70 | Server's Module Magic Number: 20051115:23 |
| 71 | Server loaded: APR 1.3.8, APR-Util 1.3.9 |
| 72 | Compiled using: APR 1.3.8, APR-Util 1.3.9 |
| 73 | Architecture: 64-bit |
| 74 | Server MPM: Worker |
| 75 | threaded: yes (fixed thread count) |
| 76 | forked: yes (variable process count) |
| 77 | }}} |
| 78 | * Apache2 mpm-preforkでは子プロセスでclose(STDOUT)するだけで良かった。 |
| 79 | * apache2 -V |
| 80 | {{{ |
| 81 | Server version: Apache/2.2.14 (Ubuntu) |
| 82 | Server built: Apr 13 2010 20:21:26 |
| 83 | Server's Module Magic Number: 20051115:23 |
| 84 | Server loaded: APR 1.3.8, APR-Util 1.3.9 |
| 85 | Compiled using: APR 1.3.8, APR-Util 1.3.9 |
| 86 | Architecture: 64-bit |
| 87 | Server MPM: Prefork |
| 88 | threaded: no |
| 89 | forked: yes (variable process count) |
| 90 | }}} |
| 91 | * 親プロセスについてはSTDIN/STDOUTを閉じた時点でhttpdからプロセスが終了させられるので(see [http://www.bioinfo.jp/tips.html Tips (CGI, Perl, Unix and etc.)])、exit前に何かの処理を行うことは出来ない。 |
| 92 | * mpm-preforkでは子プロセスが終了するまで、親プロセスはゾンビとして残った。mpm-workerではそのようなことはない模様(こちらも、一度mpm-preforkに切り替えると、workerに戻してもゾンビが残るようになった)。 |
| 93 | * ps aux | grep www-data |
| 94 | {{{ |
| 95 | www-data 3767 0.0 0.0 0 0 ? Z 15:28 0:00 [fork.cgi] <defunct> |
| 96 | www-data 3768 0.0 0.1 16700 668 ? S 15:28 0:00 /usr/bin/perl -w /var/www/archive/fork.cgi |
| 97 | }}} |
| 98 | * ただし、ゾンビとなった親プロセスは入出力がないため、しばらくするとhttpdによって殺される(see [http://www.bioinfo.jp/tips.html Tips (CGI, Perl, Unix and etc.)])。 |
| 99 | |
| 100 | = making files with random name and random content = |
| 101 | * [http://d.hatena.ne.jp/perlcodesample/20100413/1270894115 File::Temp - 一時ファイルの作成 / Perlモジュール徹底解説 - サンプルコードによるPerl入門] |
| 102 | * [http://d.hatena.ne.jp/fbis/20080114/1200307393 ランダムな文字列を生成するString::Random - Unknown::Programming] |
| 103 | |
| 104 | * makerandom.pl |
| 105 | {{{ |
| 106 | #!perl |
| 107 | #! /usr/bin/perl -w |
| 108 | |
| 109 | use strict; |
| 110 | use warnings; |
| 111 | |
| 112 | use File::Temp qw(tempfile); |
| 113 | use String::Random; |
| 114 | |
| 115 | my $filenum = $ARGV[0]; |
| 116 | |
| 117 | while ($filenum-- > 0) { |
| 118 | my ($fh, $fname) = tempfile(DIR => '.'); |
| 119 | |
| 120 | my $length = int(rand(100)); |
| 121 | print "making $fname ...\n"; |
| 122 | print $fh String::Random->new->randregex("[A-Za-z0-9]{$length}"); |
| 123 | } |
| 124 | }}} |
| 125 | |
| 126 | = $line = <$hash{'filehandle'}>; # => GLOB = |
| 127 | * [http://harapeko.asablo.jp/blog/2006/10/13/559447 山カッコ演算子にハッシュの要素を使用することはできない: 国民宿舎はらぺこ 大浴場] |
| 128 | {{{ |
| 129 | #!perl |
| 130 | my %file = (); |
| 131 | |
| 132 | { |
| 133 | open my $fh, '<test.txt' or die; |
| 134 | $file{'fh'} = $fh; |
| 135 | } |
| 136 | |
| 137 | my $line = <$file{'fh'}>; # NG |
| 138 | print $line; |
| 139 | }}} |
| 140 | * 実行結果 |
| 141 | {{{ |
| 142 | GLOB(0x161aa34)my %file = (); |
| 143 | |
| 144 | }}} |