我在使用Net :: SSH2 :: Channel时遇到问题。我正在使用perl-5.10.1-136.el6.x86_64处理Oracle LINUX 2.6.39-400.214.5.el6uek.x86_64。

远程服务器运行Debian 3.2.60-1 + deb7u3 x86_64。

Perl代码将用作bash脚本中的函数。现在,这是一个简单的论文。

我对Perl编程不是很有经验。到现在为止,我只适应现有的脚本。

使用我的测试脚本,我可以使用SSH2登录到远程服务器。

然后,我可以使用$chan->shell(<conmmand>);执行一个-只能执行一个-命令
即使第二条命令将作为脚本中的第一个shell命令成功执行,它似乎也会失败。
  ->虽然据说$chan->exec ();在通道上仅支持一个命令,但是$chan->shell();在同一通道上应该可以多次使用。
      用$chan-->close;关闭通道并再次重新打开它也无济于事。
  ->如何解释$ssh2->error;的输出?

我的第二个问题是:阻塞到底在做什么?
在ssh连接(ssh->blocking();)和通道($chan->blocking(0);)上使用阻塞有什么区别?我都看过
对我来说,在频道上使用阻塞或不使用它都没有影响。

最后,我不知道如何获取已执行命令的返回值以及如何以某种方式读取它,而不必事先知道其长度。
(测试脚本中使用的目录/ tmpp不存在。使用现有目录不会更改任何内容。
我的输出始终与uname -a输出的前65个字符相同
当我设置bufleng =70时,脚本挂起。在这里,似乎有些错误。

任何建议都非常欢迎。
谢谢您的支持。

再见
拉尔夫



我的脚本如下:



#!/usr/bin/perl

  use warnings;
  use strict;
  use Net::SSH2;
  use Data::Dumper;

  # Handling options
  my $host = shift;
  my $user = shift;
  my $passwd = shift;
  my $command = shift;
  my $destination = shift;
  #my $source = $ARGV;

  # Defining variables
  my $LEC="";

  #my $scp_session=new Net::ssh2 ( Errmode=>'return',
  my $ssh2=Net::SSH2 -> new();
  $ssh2->debug(1);
##print Dumper($ENV);

  ### Connecting (10)
  #die "can\'t connect to ${host}, Error 10" unless $ssh2->connect($host);
  #print "Connected to ${host}\n";
  $ssh2->connect($host);
  if ("$?"==0) {
    print "Connected to ${host}\n";
  } else {
    print "can\'t connect to ${host}, Error 10\n";
    #print $ssh2->error;
    print "\n";
  }

  ### Logging in (20)
  #die "can't authenticate as ${user}" unless
  #$ssh2->auth(username => '${user}',password => '${passwd}');
  $ssh2->auth_password(${user},${passwd});
  #  print $ssh2->error;
  #  #print "\n";
  if ("$?"==0) {
    print "  Authenticated as ${user}\n";
  } else {
    print $ssh2->error;
    print "   Authentication failed\n";
    exit 20
  }

print "  ## step 0\n";

  ### workload (30)
  my $chan = $ssh2->channel();
    print $ssh2->error;
    print "\n  ## step 2\n";
  #$chan->blocking(0);
  #  print $ssh2->error;
  #  print "\n  ## step 3\n";
  #$chan->shell('set cli-parameters console pager disabled');
  #   print $ssh2->error;
  #   print "\n  ## step 4\n";
  #   $chan->close;
  #   my $chan = $ssh2->channel();
  ### testing
  $chan->shell('ls -la');
    print $ssh2->error;
    print "\n  ## step 5\n";
  my $buflen = 65;
  my $buf1 = '0' x $buflen;
  $chan->read($buf1, $buflen);
  print $buf1,"\n";
  $chan1->shell(${command});
    print $ssh2->error;
    print "\n  ## step 6\n";
  #my $buflen = 70;
    print $ssh2->error;
    print "\n  ## step 7\n";
  my $buf2 = '0' x $buflen;
    print $ssh2->error;
    print "\n  ## step 8\n";
  $chan->read($LEC, $buflen);
    print $ssh2->error;
    print "\n  ## step 9\n";
  print "${command}:\n", $buf1,"\n";
    print $ssh2->error;
    print "\n  ## step 10\n";
  $chan->shell('exit');
    print $ssh2->error;
    print "\n  ## step 11\n";


  ### Logging off and disconnecting (90)
  $ssh2->disconnect($host);
    #print $ssh2->error;
    #print "\n";


   #($ssh2->auth_password($user,$passwd)) {
        #print "\n Executing command...\n";
        my $cmd = "ls";
        #print " ==> Running $cmd\n";







输出为:



perl ./test2.perl ifbscdd root dcdiag "LEC=1 ; [ -d /tmp ] && LEC=9 ; export LEC ;  return $LEC"
Connected to ifbscdd
  Authenticated as root
  ## step 0
libssh2_channel_open_ex(ss->session, pv_channel_type, len_channel_type, window_size, packet_size, ((void *)0) , 0 ) -> 0x1b89fc0
0
  ## step 2
0
  ## step 5
Net::SSH2::Channel::read(size = 65, ext = 0)
- read 65 bytes
- read 65 total
Linux ifbscdd 3.2.0-4-amd64 #1 SMP Debian 3.2.60-1+deb7u3 x86_64

libssh2_channel_open_ex(ss->session, pv_channel_type, len_channel_type, window_size, packet_size, ((void *)0) , 0 ) -> 0x1ba6490
0
  ## step 6
0
  ## step 7
0
  ## step 8
Net::SSH2::Channel::read(size = 65, ext = 0)
- read 0 bytes
- read 0 total
0
  ## step 9
LEC=1 ; [ -d /tmp ] && LEC=9 ; export LEC ;  return :
Linux ifbscdd 3.2.0-4-amd64 #1 SMP Debian 3.2.60-1+deb7u3 x86_64

0
  ## step 10
-1LIBSSH2_ERROR_SOCKET_NONEFailed waiting for channel success
  ## step 11
Net::SSH2::Channel::DESTROY
Net::SSH2::Channel::DESTROY
Net::SSH2::DESTROY object 0x1aa6230

最佳答案

您只能为每个Channel对象调用一次shell

另外,请注意shell方法不带任何参数。为了执行命令,您必须将其写入通道:

$channel->write("$cmd\n");


或者通常,您为每个要运行的命令请求一个新的Channel对象,然后使用该命令作为参数调用其exec方法。通常,在尝试执行操作时与shell交谈不是一个好主意,而且很难正确地进行。

无论如何,当您在Linux环境上运行时,您可能希望使用Net::OpenSSH,而不是Net :: SSH2,而Net :: SSH2却是原始的和有问题的。

关于linux - Perl Net::SSH2::channel的问题,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/29798602/

10-12 12:54