PHP:读取远程文件(理想情况下使用fopen)

问题描述 投票:2回答:3

我想使用PHP读取远程文本文件(理想情况下使用fopen)。当我在本地文件上使用此函数时,我的脚本使用fopen工作。

我试过了:

$file = fopen ("http://abc.abc.abc", "r");
if (!$file) {
    echo "<p>Unable to open remote file.\n";
    exit;
}

我得到了:

警告:fopen(http://abc.abc.abc):无法打开流:无法建立连接,因为目标计算机主动拒绝它。在第2行的C:\ xampp \ htdocs \ NMR \ nmrTest5.php中无法打开远程文件。

我已经读过phpseclib可能是一个不错的选择,因为我可以使用WinSCP(SFTP)或使用Puttyfor访问我的文件,我尝试了这个(在将所有文件从phpseclib复制到我的目录之后)希望我可以在本地复制文件,然后用fopen读取它(不是最好的东西,但我可以忍受):

include('Net/SFTP.php');

$sftp = new Net_SFTP('abc.abc.abc');
if (!$sftp->login('username', 'pass')) {
    exit('Login Failed');
} 

我得到了:

注意:在1561行的C:\ xampp \ htdocs \ NMR \ Net \ SSH2.php中找不到与客户端加密算法兼容的服务器登录失败

有趣的是,如果我连接到服务器(使用WinSCP),我收到了不同的消息:

注意:从第3362行的C:\ xampp \ htdocs \ NMR \ Net \ SSH2.php中的套接字读取错误

注意:服务器在第1471行的C:\ xampp \ htdocs \ NMR \ Net \ SSH2.php中关闭连接登录失败

有关如何让它工作的任何想法?理想情况下我会使用fopen,但我对其他解决方案持开放态度。

php fopen phpseclib
3个回答
1
投票

我遇到了与fopen类似的问题。卷曲对于这些目的很有用。

请检查以下基本示例函数(如果url是https,请取消注释CURLOPT_SSL_VERIFYPEER = FALSE行)。

$url = '***THE URL***';
$result = get_web_page_by_curl($url);

if ($result['errno'] != 0) echo 'error: bad url, timeout, redirect loop ...';
if ($result['http_code'] != 200) echo 'error: no page, no permissions, no service ...';
else {
    $page = $result['content'];
    echo $page;
}

function get_web_page_by_curl($url) {
    $agent = "Mozilla/5.0 (Windows; U; Windows NT 5.0; en-US; rv:1.4) Gecko/20030624 Netscape/7.1 (ax)";

    $options = array(
        CURLOPT_RETURNTRANSFER => true,     // return web page
        CURLOPT_HEADER         => false,    // don't return headers
        CURLOPT_FOLLOWLOCATION => true,     // follow redirects
        CURLOPT_ENCODING       => "",       // handle all encodings
        CURLOPT_USERAGENT      => $agent,   // who am i
        CURLOPT_AUTOREFERER    => true,     // set referer on redirect
        CURLOPT_CONNECTTIMEOUT => 120,      // timeout on connect
        CURLOPT_TIMEOUT        => 120,      // timeout on response
        CURLOPT_MAXREDIRS      => 10,       // stop after 10 redirects
        //CURLOPT_SSL_VERIFYPEER => FALSE   // this line makes it work under https 
    );

    $ch = curl_init($url);
    curl_setopt_array($ch, $options);
    $content = curl_exec($ch);
    $err     = curl_errno($ch);
    $errmsg  = curl_error($ch);
    $header  = curl_getinfo($ch);
    curl_close($ch);

    $header['errno']   = $err;
    $header['errmsg']  = $errmsg;
    $header['content'] = $content;
    return $header;
}

1
投票

我自己一直在解决这个问题,并且无法在任何一个地方找到任何好的文档来完成这个任务。

我刚刚创建了一个使用Monolog的日志服务,基本上根据正在写入/创建的日志文件创建自定义流处理程序。因此,它需要一个资源(例如由fopen创建的资源)才能将日志文件写入SFTP服务器。

我让它使用ssh2库这样工作:

$connection = ssh2_connect($this->host, 22);
ssh2_auth_password($connection, $this->user, $this->password);

$sftp = ssh2_sftp($connection);

//some stuff to do with whether the file already exists or not

$fh=fopen("ssh2.sftp://$sftp".ssh2_sftp_realpath($sftp,".")."/$this->logName/$this->fileName", 'a+');

return new StreamHandler($fh);

一切都工作得很漂亮,直到我将服务集成到一个不同的项目,并意识到这只是在我的开发机器上工作,因为它有libssh2库安装as outlined in this question

不幸的是,生成服务器并不容易添加库。因此,我发现自己在寻找不同的解决方案。

我在其他项目中使用过phpseclib,但仅限于基本的get()put()和一些nlist()电话。

为了使这个工作,我不得不使用Stream对象。没有很好的记录,但有a good discussion here

基于那里的信息,加上SFTP类中的一些挖掘,特别是get()函数,这就是我如何使用phpseclib实现相同的功能

SFTP\Stream::register();
$sftpFileSystem = new SFTP($this->host);
if (!$sftpFileSystem->login($this->user, $this->password)) {
    throw new Exception("Error logging in to central logging system. Please check the local logs and email for details", 1);
}

$context = [
    'sftp' => [
        'sftp' => $sftpFileSystem
    ],
];

//some stuff to do with whether the file already exists or not
$remote_file = $sftpFileSystem->realpath('test.txt');

$sftpStream = fopen("sftp://.{$remote_file}", 'a+', null, stream_context_create($context));
if (!$sftpStream) {
    exit(1);
}

return new StreamHandler($sftpStream);

请注意.sftp://召唤之后的点(fopen())。我浪费了半个小时!


1
投票

这就是我在问题的评论中根据@neubert的建议来解决问题的问题。

我首先在我的服务器上添加了phpseclib文件夹。然后我在我的PHP文件中使用此代码来访问远程服务器上的文件:

//needed for phpseclib
set_include_path(get_include_path() . PATH_SEPARATOR . 'phpseclib');
include_once('Net/SFTP.php');

//connection to the server
$sftp = new Net_SFTP('abc.abc.abc');
if (!$sftp->login('my_login', 'my_password')) {
  exit('Login Failed');
}

//dump entire file in a string, convert to an array and check number of lines

else {
  $text = $sftp->get('full_path_to_my_file');
}
$myArray = explode("\n", $text);
$nbline = count($myArray);
© www.soinside.com 2019 - 2024. All rights reserved.