c++ - 使用 Boost [C++] 将 http 文件内容读取为字符串

标签 c++ string boost webserver

我必须在 Visual Studio C++ 中读取文件(在网络服务器中)的内容以进行字符串处理。我有一个在一个网络服务器上运行良好的代码。但是如果我使用另一个网络服务器,我的程序只会读取前 200 个字符。除了网络服务器,一切都一样。

这是有效的 URL:http://www.fxcoder.hu/fxc_esopus/clients.dat

这是最大值。 200 个字符 URL:http://www.forexhelpmate.com/app/webroot/forex/clients.dat

clients.dat文件在每台服务器上也是相同的。

我的代码:

#include <iostream>
#include <istream>
#include <ostream>
#include <string>
#include <boost/asio.hpp>
using boost::asio::ip::tcp;
std::ostringstream ostringstream_content;
int GetPageContent(char* argv[]){
    try{
        boost::asio::io_service io_service;
        // Get a list of endpoints corresponding to the server name.
        tcp::resolver resolver(io_service);
        tcp::resolver::query query(argv[0], "http");
        tcp::resolver::iterator endpoint_iterator = resolver.resolve(query);
        tcp::resolver::iterator end;
        // Try each endpoint until we successfully establish a connection.
        tcp::socket socket(io_service);
        boost::system::error_code error = boost::asio::error::host_not_found;
        while (error && endpoint_iterator != end){
            socket.close();
            socket.connect(*endpoint_iterator++, error);
        }
        if (error) throw boost::system::system_error(error);
        // Form the request. We specify the "Connection: close" header so that the
        // server will close the socket after transmitting the response. This will
        // allow us to treat all data up until the EOF as the content.
        boost::asio::streambuf request;
        std::ostream request_stream(&request);
        request_stream << "GET " << argv[1] << " HTTP/1.0\r\n";
        request_stream << "Host: " << argv[0] << "\r\n";
        request_stream << "Accept: */*\r\n";
        request_stream << "Connection: close\r\n\r\n";
        // Send the request.
        boost::asio::write(socket, request);
        // Read the response status line.
        boost::asio::streambuf response;
        boost::asio::read_until(socket, response, "\r\n");
        // Check that response is OK.
        std::istream response_stream(&response);
        std::string http_version;
        response_stream >> http_version;
        unsigned int status_code;
        response_stream >> status_code;
        std::string status_message;
        std::getline(response_stream, status_message);
        if (!response_stream || http_version.substr(0, 5) != "HTTP/"){
            std::cout << "Invalid response\n";
            return 1;
        }
        if (status_code != 200){
            std::cout << "Response returned with status code " << status_code << "\n";
            return 1;
        }
        // Read the response headers, which are terminated by a blank line.
        boost::asio::read_until(socket, response, "\r\n\r\n");
        // Process the response headers.
        std::string header;
        while (std::getline(response_stream, header) && header != "\r")
            std::cout << header << "\n";
        std::cout << "\n";
        // Write whatever content we already have to output.
        if (response.size() > 0){
            ostringstream_content << &response;
        }
        // Read until EOF, writing data to output as we go.
        while (boost::asio::read(socket, response, boost::asio::transfer_at_least(1), error)) std::cout << &response;
        if (error != boost::asio::error::eof) throw boost::system::system_error(error);
    }
    catch (std::exception& e){
        std::cout << "Exception: " << e.what() << "\n";
    }
    return 0;
}

两个 Web 服务器之间的哪些差异可能会导致此问题?

为什么第二台服务器只读取前 200 个字符?

提前谢谢你。

亲戚


匈牙利,欧盟

最佳答案

您将第一部分流式传输到 ostringstream_content,将其余部分流式传输到 std::cout

以下为我正确下载了这两个文件

#include <iostream>
#include <istream>
#include <ostream>
#include <string>
#include <boost/asio.hpp>
using boost::asio::ip::tcp;

int GetPageContent(char const *argv[]) {
    try {
        boost::asio::io_service io_service;
        // Get a list of endpoints corresponding to the server name.
        tcp::resolver resolver(io_service);
        tcp::resolver::query query(argv[0], "http");
        tcp::resolver::iterator endpoint_iterator = resolver.resolve(query);
        tcp::resolver::iterator end;
        // Try each endpoint until we successfully establish a connection.
        tcp::socket socket(io_service);
        boost::system::error_code error = boost::asio::error::host_not_found;
        while (error && endpoint_iterator != end) {
            socket.close();
            socket.connect(*endpoint_iterator++, error);
        }
        if (error)
            throw boost::system::system_error(error);
        // Form the request. We specify the "Connection: close" header so that the
        // server will close the socket after transmitting the response. This will
        // allow us to treat all data up until the EOF as the content.
        boost::asio::streambuf request;
        std::ostream request_stream(&request);
        request_stream << "GET " << argv[1] << " HTTP/1.0\r\n";
        request_stream << "Host: " << argv[0] << "\r\n";
        request_stream << "Accept: */*\r\n";
        request_stream << "Connection: close\r\n\r\n";
        // Send the request.
        boost::asio::write(socket, request);
        // Read the response status line.
        boost::asio::streambuf response;
        boost::asio::read_until(socket, response, "\r\n");
        // Check that response is OK.
        std::istream response_stream(&response);
        std::string http_version;
        response_stream >> http_version;
        unsigned int status_code;
        response_stream >> status_code;
        std::string status_message;
        std::getline(response_stream, status_message);
        if (!response_stream || http_version.substr(0, 5) != "HTTP/") {
            std::cerr << "Invalid response\n";
            return 1;
        }
        if (status_code != 200) {
            std::cerr << "Response returned with status code " << status_code << "\n";
            return 1;
        }

        // Read the response headers, which are terminated by a blank line.
        boost::asio::read_until(socket, response, "\r\n\r\n");

        // Process the response headers.
        std::string header;

        while (std::getline(response_stream, header) && header != "\r")
            std::cerr << header << "\n";

        std::cerr << "\n";
        std::cerr << "Writing content data\n";
        // Write whatever content we already have to output.
        if (response.size() > 0) {
            std::cout << &response;
        }

        // Read until EOF, writing data to output as we go.
        while (true) { 
            size_t n = boost::asio::read(socket, response, boost::asio::transfer_at_least(1), error);

            if (!error)
            {
                if (n)
                    std::cout << &response;
            }

            if (error == boost::asio::error::eof)
                break;

            if (error)
                throw boost::system::system_error(error);
        }
    }
    catch (std::exception &e) {
        std::cerr << "Exception: " << e.what() << "\n";
    }

    std::cerr << "Done\n";
    return 0;
}

int main(){
    char const* argv[] = { "www.fxcoder.hu", "/fxc_esopus/clients.dat" };
    //char const* argv[] = { "www.forexhelpmate.com", "/app/webroot/forex/clients.dat" };

    return GetPageContent(argv);
}

关于c++ - 使用 Boost [C++] 将 http 文件内容读取为字符串,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27491288/

相关文章:

c++ - 为什么这样的循环在输入字符串时不会捕获错误?

c++ - 函数局部静态 const 对象的线程安全初始化

c++ - std::dynamic_pointer_cast 未正确向下转换

c++ - 如何正确使用外部变量?

php 用中间的连字符组合变量

arrays - swift 2.0 - 错误 : Immutable value of type '[String]' only has mutating members named 'append'

c++ - 子类化 std::thread:构造函数中可变参数模板函数的问题

java - 从任意字符串中获取数字

c++ - boost 静态链接 boost.locale 错误

boost - 在 Red Hat Linux 上安装 mapnik 失败