spectrum2/3rdparty/cpprestsdk/tests/functional/http/client/progress_handler_tests.cpp
2015-11-19 15:19:14 +01:00

431 lines
13 KiB
C++

/***
* ==++==
*
* Copyright (c) Microsoft Corporation. All rights reserved.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
* ==--==
* =+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+=+
*
* Tests cases manually building up HTTP requests with progress handlers.
*
* =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
****/
#include "stdafx.h"
#ifdef _WIN32
#include <WinError.h>
#endif
using namespace web::http;
using namespace web::http::client;
using namespace tests::functional::http::utilities;
namespace tests { namespace functional { namespace http { namespace client {
SUITE(progress_handler_tests)
{
TEST_FIXTURE(uri_address, set_progress_handler_no_bodies)
{
http_client_config config;
config.set_chunksize(512);
http_client client(m_uri, config);
const method mtd = methods::GET;
utility::size64_t upsize = 4711u, downsize = 4711u;
int calls = 0;
http_request msg(mtd);
msg.set_progress_handler(
[&](message_direction::direction direction, utility::size64_t so_far)
{
calls += 1;
if (direction == message_direction::upload)
upsize = so_far;
else
downsize = so_far;
});
test_http_server::scoped_server scoped(m_uri);
scoped.server()->next_request().then([&](test_request *p_request)
{
http_asserts::assert_test_request_equals(p_request, mtd, U("/"));
std::map<utility::string_t, utility::string_t> headers;
p_request->reply(200, utility::string_t(U("OK")), headers);
});
auto response = client.request(msg).get();
http_asserts::assert_response_equals(response, status_codes::OK);
VERIFY_ARE_EQUAL(0, upsize);
response.content_ready().wait();
VERIFY_ARE_EQUAL(0, downsize);
VERIFY_ARE_EQUAL(2, calls);
}
TEST_FIXTURE(uri_address, set_progress_handler_upload)
{
http_client_config config;
config.set_chunksize(512);
http_client client(m_uri, config);
const method mtd = methods::POST;
utility::string_t data;
utility::string_t content_type = U("text/plain; charset=utf-8");
const size_t repeats = 5500;
for (size_t i = 0; i < repeats; ++i)
data.append(U("abcdefghihklmnopqrstuvwxyz"));
utility::size64_t upsize = 4711u, downsize = 4711u;
int calls = 0;
http_request msg(mtd);
msg.set_progress_handler(
[&](message_direction::direction direction, utility::size64_t so_far)
{
calls += 1;
if (direction == message_direction::upload)
upsize = so_far;
else
downsize = so_far;
});
msg.set_body(data);
test_http_server::scoped_server scoped(m_uri);
scoped.server()->next_request().then([&](test_request *p_request)
{
http_asserts::assert_test_request_equals(p_request, mtd, U("/"), content_type, data);
std::map<utility::string_t, utility::string_t> headers;
p_request->reply(200, utility::string_t(U("OK")), headers);
});
auto response = client.request(msg).get();
http_asserts::assert_response_equals(response, status_codes::OK);
VERIFY_ARE_EQUAL(26u*repeats, upsize);
response.content_ready().wait();
VERIFY_ARE_EQUAL(0, downsize);
// We don't have very precise control over how much of a message is transferred
// in each chunk being sent or received, so we can't make an exact comparison here.
VERIFY_IS_TRUE(calls >= 3);
}
TEST_FIXTURE(uri_address, set_progress_handler_download)
{
http_client_config config;
config.set_chunksize(512);
http_client client(m_uri, config);
const method mtd = methods::GET;
utility::size64_t upsize = 4711u, downsize = 4711u;
int calls = 0;
http_request msg(mtd);
msg.set_progress_handler(
[&](message_direction::direction direction, utility::size64_t so_far)
{
calls += 1;
if (direction == message_direction::upload)
upsize = so_far;
else
downsize = so_far;
});
const size_t repeats = 6000;
test_http_server::scoped_server scoped(m_uri);
scoped.server()->next_request().then([&](test_request *p_request)
{
http_asserts::assert_test_request_equals(p_request, mtd, U("/"));
std::string resp_data;
for (size_t i = 0; i < repeats; ++i)
resp_data.append("abcdefghihklmnopqrstuvwxyz");
std::map<utility::string_t, utility::string_t> headers;
headers[U("Content-Type")] = U("text/plain");
p_request->reply(200, utility::string_t(U("OK")), headers, resp_data);
});
auto response = client.request(msg).get();
http_asserts::assert_response_equals(response, status_codes::OK);
VERIFY_ARE_EQUAL(0, upsize);
response.content_ready().wait();
VERIFY_ARE_EQUAL(26u*repeats, downsize);
// We don't have very precise control over how much of a message is transferred
// in each chunk being sent or received, so we can't make an exact comparison here.
VERIFY_IS_TRUE(calls > 4);
}
TEST_FIXTURE(uri_address, set_progress_handler_upload_and_download)
{
http_client_config config;
config.set_chunksize(512);
http_client client(m_uri, config);
const method mtd = methods::POST;
utility::string_t data;
utility::string_t content_type = U("text/plain; charset=utf-8");
const size_t repeats = 5500;
for (size_t i = 0; i < repeats; ++i)
data.append(U("abcdefghihklmnopqrstuvwxyz"));
utility::size64_t upsize = 4711u, downsize = 4711u;
int calls = 0;
http_request msg(mtd);
msg.set_progress_handler(
[&](message_direction::direction direction, utility::size64_t so_far)
{
calls += 1;
if (direction == message_direction::upload)
upsize = so_far;
else
downsize = so_far;
});
msg.set_body(data);
test_http_server::scoped_server scoped(m_uri);
scoped.server()->next_request().then([&](test_request *p_request)
{
http_asserts::assert_test_request_equals(p_request, mtd, U("/"), content_type, data);
std::string resp_data;
for (size_t i = 0; i < repeats*2; ++i)
resp_data.append("abcdefghihklmnopqrstuvwxyz");
std::map<utility::string_t, utility::string_t> headers;
headers[U("Content-Type")] = U("text/plain");
p_request->reply(200, utility::string_t(U("OK")), headers, resp_data);
});
auto response = client.request(msg).get();
http_asserts::assert_response_equals(response, status_codes::OK);
VERIFY_ARE_EQUAL(26u*repeats, upsize);
response.content_ready().wait();
VERIFY_ARE_EQUAL(26u*repeats*2, downsize);
// We don't have very precise control over how much of a message is transferred
// in each chunk being sent or received, so we can't make an exact comparison here.
VERIFY_IS_TRUE(calls > 4);
}
TEST_FIXTURE(uri_address, set_progress_handler_open_failure)
{
http_client client(U("http://localhost323:-1"));
const method mtd = methods::POST;
utility::string_t data;
utility::string_t content_type = U("text/plain; charset=utf-8");
const size_t repeats = 5500;
for (size_t i = 0; i < repeats; ++i)
data.append(U("abcdefghihklmnopqrstuvwxyz"));
utility::size64_t upsize = 4711u, downsize = 4711u;
int calls = 0;
http_request msg(mtd);
// We should never see this handler called.
msg.set_progress_handler(
[&](message_direction::direction direction, utility::size64_t so_far)
{
calls += 1;
if (direction == message_direction::upload)
upsize = so_far;
else
downsize = so_far;
});
msg.set_body(data);
auto response = client.request(msg);
VERIFY_THROWS(response.get(), web::http::http_exception);
VERIFY_ARE_EQUAL(4711u, upsize);
VERIFY_ARE_EQUAL(4711u, downsize);
VERIFY_ARE_EQUAL(0, calls);
}
TEST_FIXTURE(uri_address, set_progress_handler_request_timeout)
{
test_http_server::scoped_server scoped(m_uri);
http_client_config config;
config.set_chunksize(512);
config.set_timeout(utility::seconds(1));
http_client client(m_uri, config);
const method mtd = methods::POST;
utility::string_t data;
utility::string_t content_type = U("text/plain; charset=utf-8");
const size_t repeats = 5500;
for (size_t i = 0; i < repeats; ++i)
data.append(U("abcdefghihklmnopqrstuvwxyz"));
utility::size64_t upsize = 4711u, downsize = 4711u;
int calls = 0;
http_request msg(mtd);
// We should never see this handler called for download, but for upload should still happen, since
// there's a server (just not a very responsive one) and we're sending data to it.
msg.set_progress_handler(
[&](message_direction::direction direction, utility::size64_t so_far)
{
calls += 1;
if (direction == message_direction::upload)
upsize = so_far;
else
downsize = so_far;
});
msg.set_body(data);
auto response = client.request(msg);
#ifdef __APPLE__
// CodePlex 295
VERIFY_THROWS(response.get(), http_exception);
#else
VERIFY_THROWS_HTTP_ERROR_CODE(response.get(), std::errc::timed_out);
#endif
VERIFY_ARE_EQUAL(26u*repeats, upsize);
VERIFY_ARE_EQUAL(4711u, downsize);
// We don't have very precise control over how much of the message is transferred
// before the exception occurs, so we can't make an exact comparison here.
VERIFY_IS_TRUE(calls >= 2);
}
TEST_FIXTURE(uri_address, upload_nobody_exception)
{
test_http_server::scoped_server scoped(m_uri);
http_client client(m_uri);
http_request msg(methods::GET);
scoped.server()->next_request().then([&](test_request *p_request)
{
p_request->reply(200, utility::string_t(U("OK")));
});
msg.set_progress_handler([](message_direction::direction, utility::size64_t)
{
// First all is for data upload completion
throw std::invalid_argument("fake error");
});
VERIFY_THROWS(client.request(msg).get(), std::invalid_argument);
// Codeplex 328.
#if !defined(_WIN32)
tests::common::utilities::os_utilities::sleep(1000);
#endif
}
TEST_FIXTURE(uri_address, download_nobody_exception)
{
test_http_server::scoped_server scoped(m_uri);
http_client client(m_uri);
http_request msg(methods::GET);
scoped.server()->next_request().then([&](test_request *p_request)
{
p_request->reply(200, utility::string_t(U("OK")));
});
int numCalls = 0;
msg.set_progress_handler([&](message_direction::direction, utility::size64_t)
{
if(++numCalls == 2)
{
// second is for data download
throw std::invalid_argument("fake error");
}
});
VERIFY_THROWS(client.request(msg).get().content_ready().get(), std::invalid_argument);
}
TEST_FIXTURE(uri_address, data_upload_exception)
{
test_http_server::scoped_server scoped(m_uri);
http_client client(m_uri);
http_request msg(methods::PUT);
msg.set_body(U("A"));
msg.set_progress_handler([&](message_direction::direction, utility::size64_t)
{
throw std::invalid_argument("fake error");
});
VERIFY_THROWS(client.request(msg).get(), std::invalid_argument);
// Codeplex 328.
#if !defined(_WIN32)
tests::common::utilities::os_utilities::sleep(1000);
#endif
}
TEST_FIXTURE(uri_address, data_download_exception)
{
test_http_server::scoped_server scoped(m_uri);
http_client client(m_uri);
http_request msg(methods::GET);
scoped.server()->next_request().then([&](test_request *p_request)
{
std::string resp_data("abc");
std::map<utility::string_t, utility::string_t> headers;
headers[U("Content-Type")] = U("text/plain");
p_request->reply(200, utility::string_t(U("OK")), headers, resp_data);
});
int numCalls = 0;
msg.set_progress_handler([&](message_direction::direction, utility::size64_t)
{
if(++numCalls == 2)
{
// 2rd is for data download
throw std::invalid_argument("fake error");
}
});
try
{
handle_timeout([&]
{
client.request(msg).get().content_ready().get();
});
}
catch (std::invalid_argument const &)
{
// Expected.
}
}
}
}}}}