客户端 - 服务器2路通信不成功(Perl) [英] Client-Server 2 way communication unsuccessful (Perl)

查看:160
本文介绍了客户端 - 服务器2路通信不成功(Perl)的处理方法,对大家解决问题具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧!

问题描述

REQ流量:

____________      LOGIN           ___________
|          |------------------->>|          |
|  CLIENT  |      LOGIN ACK      |  SERVER  |
|__________|<<------------------ |__________|

客户端和服务器发送他们打开的文件的内容。

The client and the server sends across the contents of the files they open.

CODE:

p>

SERVER

$socket = new IO::Socket::INET ( LocalHost => '127.0.0.1', LocalPort => '0155', Proto => 'tcp', Listen => 1, Reuse => 1 ) or die "Oops: $! \n";

print "Waiting for the Client.\n";

while($clientsocket = $socket->accept()){


print   "Connected from : ", $clientsocket->peerhost();
print   ", Port : ", $clientsocket->peerport(), "\n";

while(<$clientsocket>){
    if($_ ne "\$END\$\n"){
            print"Message received from Client : $_";
        print $clientsocket $_;
}
    else{
    print"\$END\$\n";  
}   
} last;
}

login_ack();
sub login_ack{

# Some code removed for the purpose of posting                          
open (LOGINACK, "login_ack.txt") || die "Cannot open login acknowledgment file $!\n";
my @loginack=<LOGINACK>;
close LOGINACK;                         
open(LOG, ">>logfile.txt");

foreach $loginack(@loginack) {
    if($loginack ne "\$END\$\n"){
        print $clientsocket $loginack;
        print LOG $loginack;
        print $loginack;
    }
    else{
        print"\$END\$\n";   
    }
}

$socket = new IO::Socket::INET ( PeerHost => '127.0.0.1', PeerPort => '0155', Proto => 'tcp', Reuse => 1) or die "$!\n";

print "Connected to the Server.\n";

send_login();
sub send_login{

# Some code removed for the purpose of posting                              
open (LOGIN, "Login.txt") || die "Cannot open login file $!\n";
my @login=<LOGIN>;
close LOGIN;                            
open(LOG, ">>logfile.txt");

foreach $login(@login) {
     if($login ne "\$END\$\n"){
        print $socket $login;
        print LOG $login;
        print $login;
    }
    else{
    print"\$END\$\n"; 
    }
}
} 
LOGINACK : while(<$socket>){
        print"Message received from Server : $_";
        print $socket $_; 
        last LOGINACK;
}

PRESENT OUTPUT:

>

希望的输出:

客户
$ b

CLIENT

Connected to the Server.
this is Login
SAS4
50
SAS_ACTION LOGIN
LOGIN bss
PASSWORD cleint
$END$
Message received from Server : this is Login Ack
Message received from Server : SAS4
Message received from Server : 61
Message received from Server : SAS_ACTION LOGIN_ACK
Message received from Server : ACK_STATUS 0
Message received from Server : ACK_MESSAGE Logged In
Message received from Server : $END$        

SERVER

Waiting for the Client.
Connected from : 127.0.0.1, Port : 1862
Message received from Client : this is Login
Message received from Client : SAS4
Message received from Client : 50
Message received from Client : SAS_ACTION LOGIN
Message received from Client : LOGIN bss
Message received from Client : PASSWORD cleint
Message received from Client : $END$
this is Login Ack
SAS4
61
SAS_ACTION LOGIN_ACK
ACK_STATUS 0
ACK_MESSAGE Logged In
$END$


推荐答案

回答修订版



此代码包含3个脚本: / p>

Revised Edition of Answer

This code includes 3 scripts:


  1. serverJ.pl 这是一个迭代服务器,但继续处理多个请求;

  2. clientJ.pl 这是一个简单的客户端,发送来自 Login.txt 文件并回显从服务器接收的内容;和

  3. clientQ.pl ,它向服务器发送QUIT消息。

  1. serverJ.pl which is an iterative server, processing one request at a time, but continuing to process multiple requests;
  2. clientJ.pl which is a simple client that sends the login request from the Login.txt file and echoes what it receives from the server; and
  3. clientQ.pl which sends a QUIT message to the server.

我使用老式的文件句柄而不是升级到现代的词法文件句柄,但我改为3参数形式的 open 。另一个变化是大多数消息前缀有 $$:,Perl进程ID。我不会标记 INNER:循环,除了与 OUTER:循环的对称性,它需要一个标签。不清楚 clientJ中需要 send_login() abc() .pl ;它们依赖于原始代码。

I've left the code using old-fashioned file handles rather than upgrading to modern lexical file handles, but I changed to the 3-argument form of open. One other change is that most messages are prefixed with $$:, the Perl process ID. I would not have labelled the INNER: loop except for symmetry with the OUTER: loop, which does need a label. It is not clear that the functions send_login() and abc() are needed in clientJ.pl; they are left in deference to the original code.

#!/usr/bin/env perl
use strict;
use warnings;
use IO::Socket::INET;

my $socket = new IO::Socket::INET (
    LocalHost => '127.0.0.1',
    LocalPort => '1055',
    Proto => 'tcp',
    Listen => 1,
    Reuse => 1
) or die "Oops: $! \n";

print "$$: waiting for clients.\n";
server_loop();
$socket->close();

sub server_loop
{
OUTER:
    while (my $clientsocket = $socket->accept())
    {
        print "$$: connected from : ", $clientsocket->peerhost();
        print ", port : ", $clientsocket->peerport(), "\n";

INNER:
        while (<$clientsocket>)
        {
            print"$$: Message received from Client : $_";
            last INNER if ($_ eq "\$END\$\n");
            last OUTER if ($_ eq "\$QUIT\$\n");
            print $clientsocket $_;
        }

        my $login_ack = "login_ack.txt";
        open LOGINACK, '<', $login_ack or die "Cannot open login acknowledgment file $login_ack ($!)\n";
        my @loginack = <LOGINACK>;
        close LOGINACK;

        my $logfile = "logfile.txt";
        open LOG, ">>", $logfile or die "$$: cannot open $logfile ($!)\n";

        foreach my $loginack (@loginack)
        {
            last if ($loginack eq "\$END\$\n");
            print $clientsocket $loginack;
            print LOG "$$: $loginack";
            print "$$: server to client: $loginack";
        }

        close LOG;
        close $clientsocket;
    }
}



clientJ.pl



clientJ.pl

#!/usr/bin/env perl
use strict;
use warnings;
use IO::Socket::INET;

my $socket = new IO::Socket::INET (
    PeerHost => '127.0.0.1',
    PeerPort => '1055',
    Proto    => 'tcp',
    Reuse    => 1
) or die "$!\n";

print "$$: connected to the server.\n";

send_login();
abc();
$socket->close();

sub send_login
{
    my $login_txt = "Login.txt";
    open LOGIN, '<', $login_txt or die "Cannot open $login_txt $!\n";
    my @login = <LOGIN>;
    close LOGIN;

    my $logfile = "logfile.txt";
    open LOG, '>>', $logfile or die "Cannot open $logfile ($!)\n";

    foreach my $login (@login)
    {
        print $socket $login;
        print LOG "$$: $login";
        print "$$: sent to server: $login";
        last if ($login eq "\$END\$\n");
    }
    close LOG;
}

sub abc
{
    while (<$socket>)
    {
        print"$$: Message received from Server: $_";
    }
}



clientQ.pl



clientQ.pl

#!/usr/bin/env perl
use strict;
use warnings;
use IO::Socket::INET;

my $socket = new IO::Socket::INET(
  PeerHost => '127.0.0.1',
  PeerPort => '1055',
  Proto    => 'tcp',
  Reuse    => 1
) or die "$!\n";

print $socket "\$QUIT\$\n";

$socket->close();



示例日志



Sample log

$ perl serverJ.pl 2>&1 &
[1] 87610
$ 87610: waiting for clients.

$ perl clientJ.pl | so
87610: connected from : 127.0.0.1, port : 57143
87610: Message received from Client : SAS4
87610: Message received from Client : 50
87610: Message received from Client : SAS_ACTION LOGIN
87610: Message received from Client : LOGIN bss
87610: Message received from Client : PASSWORD cleint
87610: Message received from Client : $END$
87610: server to client: SAS4
87610: server to client: 61
87610: server to client: SAS_ACTION LOGIN_ACK
87610: server to client: ACK_STATUS 0
87610: server to client: ACK_MESSAGE Logged In
87614: connected to the server.
87614: sent to server: SAS4
87614: sent to server: 50
87614: sent to server: SAS_ACTION LOGIN
87614: sent to server: LOGIN bss
87614: sent to server: PASSWORD cleint
87614: sent to server: $END$
87614: Message received from Server: SAS4
87614: Message received from Server: 50
87614: Message received from Server: SAS_ACTION LOGIN
87614: Message received from Server: LOGIN bss
87614: Message received from Server: PASSWORD cleint
87614: Message received from Server: SAS4
87614: Message received from Server: 61
87614: Message received from Server: SAS_ACTION LOGIN_ACK
87614: Message received from Server: ACK_STATUS 0
87614: Message received from Server: ACK_MESSAGE Logged In
$ perl clientJ.pl | so
87610: connected from : 127.0.0.1, port : 57144
87610: Message received from Client : SAS4
87610: Message received from Client : 50
87610: Message received from Client : SAS_ACTION LOGIN
87610: Message received from Client : LOGIN bss
87610: Message received from Client : PASSWORD cleint
87610: Message received from Client : $END$
87610: server to client: SAS4
87610: server to client: 61
87610: server to client: SAS_ACTION LOGIN_ACK
87610: server to client: ACK_STATUS 0
87610: server to client: ACK_MESSAGE Logged In
87617: connected to the server.
87617: sent to server: SAS4
87617: sent to server: 50
87617: sent to server: SAS_ACTION LOGIN
87617: sent to server: LOGIN bss
87617: sent to server: PASSWORD cleint
87617: sent to server: $END$
87617: Message received from Server: SAS4
87617: Message received from Server: 50
87617: Message received from Server: SAS_ACTION LOGIN
87617: Message received from Server: LOGIN bss
87617: Message received from Server: PASSWORD cleint
87617: Message received from Server: SAS4
87617: Message received from Server: 61
87617: Message received from Server: SAS_ACTION LOGIN_ACK
87617: Message received from Server: ACK_STATUS 0
87617: Message received from Server: ACK_MESSAGE Logged In
$ perl clientQ.pl | so
87610: connected from : 127.0.0.1, port : 57147
87610: Message received from Client : $QUIT$
[1]+  Done                    perl serverJ.pl 2>&1
$ 






h3>


Post loop code executes


在服务器脚本中,发送确认时, foreach my $ loginack(@loginack)循环不会被执行。我认为这是因为代码不能突破循环。

In the server script, while sending the acknowledgement, the code after the foreach my $loginack (@loginack) loop does not get executed. I think that is because the code cannot break out of the loop. How shall it be dealt with?

中的轻度修改代码serverJ.pl

Mildly modified code in serverJ.pl:

        foreach my $loginack (@loginack)
        {
            last if ($loginack eq "\$END\$\n");
            print $clientsocket $loginack;
            print LOG "$$: $loginack";
            print "$$: server to client: $loginack";
        }

        print "$$: End of Loop\n";
        close LOG;
        close $clientsocket;
        print "$$: After close\n";

运行示例:

$ perl serverJ.pl &
[1] 6913
$ 6913: waiting for clients.

$ perl clientJ.pl
6915: connected to the server.
6913: connected from : 127.0.0.1, port : 52454
6915: sent to server: This is the file Login.txt
6913: Message received from Client : This is the file Login.txt
6915: sent to server: It contains more than 1 line
6915: sent to server: It also contains this one
6915: sent to server: $END$
6913: Message received from Client : It contains more than 1 line
6913: Message received from Client : It also contains this one
6913: Message received from Client : $END$
6915: Message received from Server: This is the file Login.txt
6915: Message received from Server: It contains more than 1 line
6915: Message received from Server: It also contains this one
6913: server to client: This is the file login_ack.txt
6913: server to client: As with the other file, it contains multiple lines
6913: server to client: Several of them.
6913: server to client: This is one of them.
6915: Message received from Server: This is the file login_ack.txt
6913: End of Loop
6915: Message received from Server: As with the other file, it contains multiple lines
6915: Message received from Server: Several of them.
6915: Message received from Server: This is one of them.
6913: After close
$ perl clientQ.pl
6913: connected from : 127.0.0.1, port : 52455
6913: Message received from Client : $QUIT$
[1]+  Done                    perl serverJ.pl
$






回答的原始版本



这段代码似乎对我有用(对于'works'的宽松定义)。它是基于你的代码,但使用套接字1055而不是0155(我运行在Unix;我必须是root使用小于1024的端口号),并与 use strict; / code>和使用警告; 和其他次要清理。

#!/usr/bin/env perl
use strict;
use warnings;
use IO::Socket::INET;

my $socket = new IO::Socket::INET( LocalHost => '127.0.0.1', LocalPort => '1055', Proto => 'tcp', Listen => 1, Reuse => 1 ) or die "Oops: $! \n";

print "Waiting for the Client.\n";

my $clientsocket;

while ($clientsocket = $socket->accept())
{
    print   "Connected from : ", $clientsocket->peerhost();
    print   ", Port : ", $clientsocket->peerport(), "\n";

    while (<$clientsocket>)
    {
        if ($_ ne "\$END\$\n")
        {
            print"Message received from Client : $_";
            print $clientsocket $_;
        }
        else
        {
            print"\$END\$\n";  
        }   
    }
    last;
}

login_ack();

sub login_ack
{
    # Some code removed for the purpose of posting                          
    open (LOGINACK, "login_ack.txt") || die "Cannot open login acknowledgment file $!\n";
    my @loginack=<LOGINACK>;
    close LOGINACK;                         
    open(LOG, ">>logfile.txt");

    foreach my $loginack (@loginack)
    {
        if ($loginack ne "\$END\$\n")
        {
            print $clientsocket $loginack;
            print LOG $loginack;
            print $loginack;
        }
        else
        {
            print"\$END\$\n";   
        }
    }
}



client.pl



client.pl

#!/usr/bin/env perl
use strict;
use warnings;
use IO::Socket::INET;

my $socket = new IO::Socket::INET ( PeerHost => '127.0.0.1', PeerPort => '1055', Proto => 'tcp', Reuse => 1) or die "$!\n";

print "Connected to the Server.\n";

send_login();

sub send_login
{
# Some code removed for the purpose of posting                              
    open (LOGIN, "Login.txt") || die "Cannot open login file $!\n";
    my @login=<LOGIN>;
    close LOGIN;                            
    open(LOG, ">>logfile.txt");

    foreach my $login (@login)
    {
        if ($login ne "\$END\$\n")
        {
            print $socket $login;
            print LOG $login;
            print $login;
        }
        else
        {
            print"\$END\$\n"; 
        }
    }
} 

LOGINACK:
while (<$socket>)
{
    print"Message received from Server : $_";
    print $socket $_; 
    last LOGINACK;
}



Login.txt



Login.txt

This is the content of the Login.txt file.
$END$



login_ack.txt


$ b b

login_ack.txt

This is the contents of the login_ack.txt file.
It has two lines of text plus the $END$ line.
$END$



示例输出



Sample output

$ perl server.pl &
[1] 74838
$ Waiting for the Client.

$ perl client.pl | so
Connected from : 127.0.0.1, Port : 51109
Message received from Client : This is the content of the Login.txt file.
Message received from Client : This is the content of the Login.txt file.
Connected to the Server.
This is the content of the Login.txt file.
$END$
Message received from Server : This is the content of the Login.txt file.
This is the contents of the login_ack.txt file.
It has two lines of text plus the $END$ line.
$END$
[1]+  Done                    perl server.pl
$ 


$ b b

服务器在响应单个请求后终止。

The server terminates after responding to a single request.

这篇关于客户端 - 服务器2路通信不成功(Perl)的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持IT屋!

查看全文
登录 关闭
扫码关注1秒登录
发送“验证码”获取 | 15天全站免登陆