2011-04-24 7 views
0

제 영어를 사과합니다. 소켓에 문제가 있습니다.Perl : 소켓 문제

#!/usr/bin/perl -w 

use strict; 
use POSIX; 
use POSIX ":sys_wait_h"; 
use IO::Socket; 
use IO::Handle; 
use DBI; 

my $pid= fork(); 
exit() if $pid; 
die "Couldn't fork: $! " unless defined($pid); 

POSIX::setsid() or die "Can't start a new session $!"; 
my $time_to_die =0; 
my $server; 

sub signal_handler { 
    $time_to_die = 1; 
    close($server); 
} 

$SIG{INT}= $SIG{TERM} = \&signal_handler; 

sub REAPER { 
    while ((my $waitedpid = waitpid(-1,WNOHANG)) > 0) { } 
    $SIG{CHLD} = \&REAPER; 
} 

my $server_port = 6741; 
$server= new IO::Socket::INET(LocalPort  => $server_port, 
           TYPE  => SOCK_STREAM, 
        Reuse  => 1, 
        Listen => 10) or die "Couldn't be a tcp server on port $server_port: [email protected]\n"; 

until($time_to_die) { 
    my $client; 

    while($client = $server->accept()) { 
     $SIG{CHLD} = \&REAPER;  
     defined(my $child_pid=fork()) or die "Can't fork new child $!"; 
     next if $child_pid; 

     if($child_pid == 0) { 
      close($server); 
     } 

     $client->autoflush(1); 
     print $client "Command :"; 

     while(<$client>) { 
      next unless /\S/; 

     my $full_enter_str = $_; 
     chomp($full_enter_str); 

      if($full_enter_str =~ /(<\w\d\d\d\w>),(\d{5}),(M|O),(\d{6}),(\d{6}),(\d{4}\.\d{4}\w),(\d{5}\.\d{4}\w),(\d{2}.\d),(\d{3}\.\d),(\d{3}\.\d),(\d),(\d{2})/) { 
       my $dbm = DBI->connect("DBI:mysql:database=homepage;host=127.0.0.1;port=3306", "av", "") or die "MySQL connect error"; 
       $dbm->do("INSERT INTO `new_table` VALUES (NULL, '".$1."')"); 
       $dbm->disconnect(); 
      } 
    } 
    continue { 
     print $client "Command :"; 
    } 
    exit; 
} 
continue { 
    close($client); 
} 

첫 번째 문제 : 내가 텔넷 클라이언트를 통해 문자열을 보낼 때 : 모든 바로 그때

<T060M>,00287,M,124427,220411,5800.1577N,04200.1038E,01.0,073.4,196.4,1,69 

, 항목은 데이터베이스에 추가됩니다. 그러나 항구가 (tcpdump)와 같은 것으로 올 때 :

0x0000: 4500 0073 0003 0000 f606 d057 d557 5cc6 E..s.......W.W\. 
0x0010: c0a8 0164 dcc4 1a55 4016 741d b90a 79e4 [email protected] 
0x0020: 5018 2c60 3b1f 0000 3c54 3036 304d 3e2c P.,`;...<T060M>, 
0x0030: 3030 3238 372c 4d2c 3130 3432 3339 2c32 00287,M,104239,2 
0x0040: 3430 3431 312c 3537 3434 2e35 3432 384e 40411,5744.5428N 
0x0050: 2c30 3431 3030 2e34 3133 3445 2c31 312e ,04100.4134E,11. 
0x0060: 302c 3030 302e 302c 3039 362e 342c 312c 0,000.0,096.4,1, 
0x0070: 3646 0d         6F. 

아무 일도 일어나지 않습니다.

두 번째 문제점 : 얼마 지나지 않아 클라이언트가 정확히 2 개가 무엇인지 정확히 알지만 충분한 시간이 지나면 좀비 (ps ax | grep '% scriptname %')가 표시됩니다.

답변

0

첫 번째 문제에 대해 - 저는 잘 모르겠습니다 만 정규 표현식과 일치합니까? 귀하의 회선은 6F으로 끝나지만, 정규 표현식은 \d{2}으로 끝납니다. 일치하는 것처럼 보이지 않습니다.

리퍼가 작동해야하는 것처럼 보이기 때문에 두 번째 사항에 대해서도 관심이 있습니다.

편집 : AFAIK, $SIG{CHLD}에서 "IGNORE"으로 설정하면 더 나은 해결책이됩니다. 어쨌든 REAPER 함수에서 유용한 것은하지 않기 때문입니다. 그러나 귀하의 접근 방식이 효과가없는 이유는 설명하지 않습니다.