Important changes to repositories hosted on mbed.com
Mbed hosted mercurial repositories are deprecated and are due to be permanently deleted in July 2026.
To keep a copy of this software download the repository Zip archive or clone locally using Mercurial.
It is also possible to export all your personal repositories from the account settings page.
Dependents: MQTTGateway2 MQTTGatewayK64 http-example-wnc GuardRoom ... more
Revision 34:6daf67a96a91, committed 2019-01-04
- Comitter:
- Jan Jongboom
- Date:
- Fri Jan 04 13:27:32 2019 +0100
- Parent:
- 33:5b2869cc8934
- Child:
- 35:b3ee394d1d2e
- Commit message:
- Add integration tests
Changed in this revision
--- /dev/null Thu Jan 01 00:00:00 1970 +0000
+++ b/TESTS/COMMON/test_setup.h Fri Jan 04 13:27:32 2019 +0100
@@ -0,0 +1,101 @@
+#ifndef _MBED_HTTP_TEST_SETUP_H_
+#define _MBED_HTTP_TEST_SETUP_H_
+
+#include "mbed.h"
+#include "NetworkInterface.h"
+
+/**
+ * Connect to the network using the default networking interface,
+ * you can also swap this out with a driver for a different networking interface
+ * if you use WiFi: see mbed_app.json for the credentials
+ */
+NetworkInterface *connect_to_default_network_interface() {
+ NetworkInterface* network = NetworkInterface::get_default_instance();
+
+ if (!network) {
+ return NULL;
+ }
+
+ nsapi_error_t connect_status = network->connect();
+
+ if (connect_status != NSAPI_ERROR_OK) {
+ return NULL;
+ }
+
+ return network;
+}
+
+
+/* List of trusted root CA certificates
+ *
+ * - Amazon, the CA for os.mbed.com
+ * - Let's Encrypt, the CA for httpbin.org
+ * - Comodo, the CA for reqres.in
+ *
+ * To add more root certificates, just concatenate them.
+ */
+const char SSL_CA_PEM[] = "-----BEGIN CERTIFICATE-----\n"
+ "MIIDQTCCAimgAwIBAgITBmyfz5m/jAo54vB4ikPmljZbyjANBgkqhkiG9w0BAQsF\n"
+ "ADA5MQswCQYDVQQGEwJVUzEPMA0GA1UEChMGQW1hem9uMRkwFwYDVQQDExBBbWF6\n"
+ "b24gUm9vdCBDQSAxMB4XDTE1MDUyNjAwMDAwMFoXDTM4MDExNzAwMDAwMFowOTEL\n"
+ "MAkGA1UEBhMCVVMxDzANBgNVBAoTBkFtYXpvbjEZMBcGA1UEAxMQQW1hem9uIFJv\n"
+ "b3QgQ0EgMTCCASIwDQYJKoZIhvcNAQEBBQADggEPADCCAQoCggEBALJ4gHHKeNXj\n"
+ "ca9HgFB0fW7Y14h29Jlo91ghYPl0hAEvrAIthtOgQ3pOsqTQNroBvo3bSMgHFzZM\n"
+ "9O6II8c+6zf1tRn4SWiw3te5djgdYZ6k/oI2peVKVuRF4fn9tBb6dNqcmzU5L/qw\n"
+ "IFAGbHrQgLKm+a/sRxmPUDgH3KKHOVj4utWp+UhnMJbulHheb4mjUcAwhmahRWa6\n"
+ "VOujw5H5SNz/0egwLX0tdHA114gk957EWW67c4cX8jJGKLhD+rcdqsq08p8kDi1L\n"
+ "93FcXmn/6pUCyziKrlA4b9v7LWIbxcceVOF34GfID5yHI9Y/QCB/IIDEgEw+OyQm\n"
+ "jgSubJrIqg0CAwEAAaNCMEAwDwYDVR0TAQH/BAUwAwEB/zAOBgNVHQ8BAf8EBAMC\n"
+ "AYYwHQYDVR0OBBYEFIQYzIU07LwMlJQuCFmcx7IQTgoIMA0GCSqGSIb3DQEBCwUA\n"
+ "A4IBAQCY8jdaQZChGsV2USggNiMOruYou6r4lK5IpDB/G/wkjUu0yKGX9rbxenDI\n"
+ "U5PMCCjjmCXPI6T53iHTfIUJrU6adTrCC2qJeHZERxhlbI1Bjjt/msv0tadQ1wUs\n"
+ "N+gDS63pYaACbvXy8MWy7Vu33PqUXHeeE6V/Uq2V8viTO96LXFvKWlJbYK8U90vv\n"
+ "o/ufQJVtMVT8QtPHRh8jrdkPSHCa2XV4cdFyQzR1bldZwgJcJmApzyMZFo6IQ6XU\n"
+ "5MsI+yMRQ+hDKXJioaldXgjUkK642M4UwtBV8ob2xJNDd2ZhwLnoQdeXeGADbkpy\n"
+ "rqXRfboQnoZsG4q5WTP468SQvvG5\n"
+ "-----END CERTIFICATE-----\n"
+ "-----BEGIN CERTIFICATE-----\n"
+ "MIIEkjCCA3qgAwIBAgIQCgFBQgAAAVOFc2oLheynCDANBgkqhkiG9w0BAQsFADA/\n"
+ "MSQwIgYDVQQKExtEaWdpdGFsIFNpZ25hdHVyZSBUcnVzdCBDby4xFzAVBgNVBAMT\n"
+ "DkRTVCBSb290IENBIFgzMB4XDTE2MDMxNzE2NDA0NloXDTIxMDMxNzE2NDA0Nlow\n"
+ "SjELMAkGA1UEBhMCVVMxFjAUBgNVBAoTDUxldCdzIEVuY3J5cHQxIzAhBgNVBAMT\n"
+ "GkxldCdzIEVuY3J5cHQgQXV0aG9yaXR5IFgzMIIBIjANBgkqhkiG9w0BAQEFAAOC\n"
+ "AQ8AMIIBCgKCAQEAnNMM8FrlLke3cl03g7NoYzDq1zUmGSXhvb418XCSL7e4S0EF\n"
+ "q6meNQhY7LEqxGiHC6PjdeTm86dicbp5gWAf15Gan/PQeGdxyGkOlZHP/uaZ6WA8\n"
+ "SMx+yk13EiSdRxta67nsHjcAHJyse6cF6s5K671B5TaYucv9bTyWaN8jKkKQDIZ0\n"
+ "Z8h/pZq4UmEUEz9l6YKHy9v6Dlb2honzhT+Xhq+w3Brvaw2VFn3EK6BlspkENnWA\n"
+ "a6xK8xuQSXgvopZPKiAlKQTGdMDQMc2PMTiVFrqoM7hD8bEfwzB/onkxEz0tNvjj\n"
+ "/PIzark5McWvxI0NHWQWM6r6hCm21AvA2H3DkwIDAQABo4IBfTCCAXkwEgYDVR0T\n"
+ "AQH/BAgwBgEB/wIBADAOBgNVHQ8BAf8EBAMCAYYwfwYIKwYBBQUHAQEEczBxMDIG\n"
+ "CCsGAQUFBzABhiZodHRwOi8vaXNyZy50cnVzdGlkLm9jc3AuaWRlbnRydXN0LmNv\n"
+ "bTA7BggrBgEFBQcwAoYvaHR0cDovL2FwcHMuaWRlbnRydXN0LmNvbS9yb290cy9k\n"
+ "c3Ryb290Y2F4My5wN2MwHwYDVR0jBBgwFoAUxKexpHsscfrb4UuQdf/EFWCFiRAw\n"
+ "VAYDVR0gBE0wSzAIBgZngQwBAgEwPwYLKwYBBAGC3xMBAQEwMDAuBggrBgEFBQcC\n"
+ "ARYiaHR0cDovL2Nwcy5yb290LXgxLmxldHNlbmNyeXB0Lm9yZzA8BgNVHR8ENTAz\n"
+ "MDGgL6AthitodHRwOi8vY3JsLmlkZW50cnVzdC5jb20vRFNUUk9PVENBWDNDUkwu\n"
+ "Y3JsMB0GA1UdDgQWBBSoSmpjBH3duubRObemRWXv86jsoTANBgkqhkiG9w0BAQsF\n"
+ "AAOCAQEA3TPXEfNjWDjdGBX7CVW+dla5cEilaUcne8IkCJLxWh9KEik3JHRRHGJo\n"
+ "uM2VcGfl96S8TihRzZvoroed6ti6WqEBmtzw3Wodatg+VyOeph4EYpr/1wXKtx8/\n"
+ "wApIvJSwtmVi4MFU5aMqrSDE6ea73Mj2tcMyo5jMd6jmeWUHK8so/joWUoHOUgwu\n"
+ "X4Po1QYz+3dszkDqMp4fklxBwXRsW10KXzPMTZ+sOPAveyxindmjkW8lGy+QsRlG\n"
+ "PfZ+G6Z6h7mjem0Y+iWlkYcV4PIWL1iwBi8saCbGS5jN2p8M+X+Q7UNKEkROb3N6\n"
+ "KOqkqm57TH2H3eDJAkSnh6/DNFu0Qg==\n"
+ "-----END CERTIFICATE-----\n"
+ "-----BEGIN CERTIFICATE-----\n"
+ "MIICiTCCAg+gAwIBAgIQH0evqmIAcFBUTAGem2OZKjAKBggqhkjOPQQDAzCBhTEL\n"
+ "MAkGA1UEBhMCR0IxGzAZBgNVBAgTEkdyZWF0ZXIgTWFuY2hlc3RlcjEQMA4GA1UE\n"
+ "BxMHU2FsZm9yZDEaMBgGA1UEChMRQ09NT0RPIENBIExpbWl0ZWQxKzApBgNVBAMT\n"
+ "IkNPTU9ETyBFQ0MgQ2VydGlmaWNhdGlvbiBBdXRob3JpdHkwHhcNMDgwMzA2MDAw\n"
+ "MDAwWhcNMzgwMTE4MjM1OTU5WjCBhTELMAkGA1UEBhMCR0IxGzAZBgNVBAgTEkdy\n"
+ "ZWF0ZXIgTWFuY2hlc3RlcjEQMA4GA1UEBxMHU2FsZm9yZDEaMBgGA1UEChMRQ09N\n"
+ "T0RPIENBIExpbWl0ZWQxKzApBgNVBAMTIkNPTU9ETyBFQ0MgQ2VydGlmaWNhdGlv\n"
+ "biBBdXRob3JpdHkwdjAQBgcqhkjOPQIBBgUrgQQAIgNiAAQDR3svdcmCFYX7deSR\n"
+ "FtSrYpn1PlILBs5BAH+X4QokPB0BBO490o0JlwzgdeT6+3eKKvUDYEs2ixYjFq0J\n"
+ "cfRK9ChQtP6IHG4/bC8vCVlbpVsLM5niwz2J+Wos77LTBumjQjBAMB0GA1UdDgQW\n"
+ "BBR1cacZSBm8nZ3qQUfflMRId5nTeTAOBgNVHQ8BAf8EBAMCAQYwDwYDVR0TAQH/\n"
+ "BAUwAwEB/zAKBggqhkjOPQQDAwNoADBlAjEA7wNbeqy3eApyt4jf/7VGFAkK+qDm\n"
+ "fQjGGoe9GKhzvSbKYAydzpmfz1wPMOG+FDHqAjAU9JM8SaczepBGR7NjfRObTrdv\n"
+ "GDeAU/7dIOA1mjbRxwG55tzd8/8dLDoWV9mSOdY=\n"
+ "-----END CERTIFICATE-----\n";
+
+#endif // MBED_HTTP_TEST_SETUP_H_
--- /dev/null Thu Jan 01 00:00:00 1970 +0000
+++ b/TESTS/tests/integration/main.cpp Fri Jan 04 13:27:32 2019 +0100
@@ -0,0 +1,266 @@
+/*
+ * PackageLicenseDeclared: Apache-2.0
+ * Copyright (c) 2018 ARM Limited
+ *
+ * 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.
+ */
+
+#include "mbed.h"
+#include "http_request.h"
+#include "https_request.h"
+#include "test_setup.h"
+#include "utest/utest.h"
+#include "unity/unity.h"
+#include "greentea-client/test_env.h"
+
+using namespace utest::v1;
+
+static NetworkInterface *network;
+
+static void setup_verify_network() {
+ if (!network) network = connect_to_default_network_interface();
+ TEST_ASSERT_NOT_NULL(network);
+}
+
+// verifies that the header is present and has a certain value
+static void assert_header(HttpResponse *res, const char *header, const char *value) {
+ bool headerPresent = false;
+ for (size_t ix = 0; ix < res->get_headers_length(); ix++) {
+ if (res->get_headers_fields()[ix]->compare(header)) {
+ headerPresent = true;
+
+ TEST_ASSERT(res->get_headers_values()[ix]->compare(value));
+ }
+ }
+ TEST_ASSERT_EQUAL(true, headerPresent);
+}
+
+static control_t http_get(const size_t call_count) {
+ setup_verify_network();
+
+ HttpRequest *req = new HttpRequest(network, HTTP_GET, "http://httpbin.org/status/418");
+
+ HttpResponse* res = req->send();
+ TEST_ASSERT(res);
+ TEST_ASSERT_EQUAL(418, res->get_status_code());
+
+ delete req;
+
+ return CaseNext;
+}
+
+static control_t http_post(const size_t call_count) {
+ setup_verify_network();
+
+ HttpRequest* req = new HttpRequest(network, HTTP_POST, "http://httpbin.org/post");
+ req->set_header("Content-Type", "application/json");
+
+ const char body[] = "{\"mykey\":\"mbedvalue\"}";
+
+ HttpResponse* res = req->send(body, strlen(body));
+ TEST_ASSERT(res);
+ TEST_ASSERT_EQUAL(200, res->get_status_code());
+
+ // verify that the Content-Type header is present, and set to application/json
+ assert_header(res, "Content-Type", "application/json");
+
+ // verify that both the key and value are present in the response
+ TEST_ASSERT(res->get_body_length() > 0);
+ TEST_ASSERT_NOT_EQUAL(res->get_body_as_string().find("mykey"), string::npos);
+ TEST_ASSERT_NOT_EQUAL(res->get_body_as_string().find("mbedvalue"), string::npos);
+
+ delete req;
+
+ return CaseNext;
+}
+
+static control_t http_socket_reuse(const size_t call_count) {
+ setup_verify_network();
+
+ TCPSocket socket;
+ nsapi_error_t open_result = socket.open(network);
+ TEST_ASSERT_EQUAL(NSAPI_ERROR_OK, open_result);
+
+ nsapi_error_t connect_result = socket.connect("httpbin.org", 80);
+ TEST_ASSERT_EQUAL(NSAPI_ERROR_OK, connect_result);
+
+ {
+ HttpRequest *req = new HttpRequest(&socket, HTTP_GET, "http://httpbin.org/status/404");
+
+ HttpResponse* res = req->send();
+ TEST_ASSERT(res);
+ TEST_ASSERT_EQUAL(404, res->get_status_code());
+
+ delete req;
+ }
+
+ {
+ HttpRequest *req = new HttpRequest(&socket, HTTP_GET, "http://httpbin.org/status/403");
+
+ HttpResponse* res = req->send();
+ TEST_ASSERT(res);
+ TEST_ASSERT_EQUAL(403, res->get_status_code());
+
+ delete req;
+ }
+
+ return CaseNext;
+}
+
+static control_t https_get(const size_t call_count) {
+ setup_verify_network();
+
+ HttpsRequest *req = new HttpsRequest(network, SSL_CA_PEM, HTTP_GET, "https://os.mbed.com/media/uploads/mbed_official/hello.txt");
+
+ HttpResponse* res = req->send();
+ TEST_ASSERT(res);
+ TEST_ASSERT_EQUAL(200, res->get_status_code());
+ TEST_ASSERT(res->get_body_length() > 0);
+ TEST_ASSERT_NOT_EQUAL(res->get_body_as_string().find("Hello world!"), string::npos);
+
+ delete req;
+
+ return CaseNext;
+}
+
+static control_t https_post(const size_t call_count) {
+ setup_verify_network();
+
+ HttpsRequest* req = new HttpsRequest(network, SSL_CA_PEM, HTTP_POST, "https://httpbin.org/post");
+ req->set_header("Content-Type", "application/json");
+
+ const char body[] = "{\"myhttpskey\":\"janjanjan\"}";
+
+ HttpResponse* res = req->send(body, strlen(body));
+ TEST_ASSERT(res);
+ TEST_ASSERT_EQUAL(200, res->get_status_code());
+
+ // verify that the Content-Type header is present, and set to application/json
+ assert_header(res, "Content-Type", "application/json");
+
+ // verify that both the key and value are present in the response
+ TEST_ASSERT(res->get_body_length() > 0);
+ TEST_ASSERT_NOT_EQUAL(res->get_body_as_string().find("myhttpskey"), string::npos);
+ TEST_ASSERT_NOT_EQUAL(res->get_body_as_string().find("janjanjan"), string::npos);
+
+ delete req;
+
+ return CaseNext;
+}
+
+static control_t https_socket_reuse(const size_t call_count) {
+ setup_verify_network();
+
+ TLSSocket socket;
+ nsapi_error_t open_result = socket.open(network);
+ TEST_ASSERT_EQUAL(NSAPI_ERROR_OK, open_result);
+
+ nsapi_error_t ca_result = socket.set_root_ca_cert(SSL_CA_PEM);
+ TEST_ASSERT_EQUAL(NSAPI_ERROR_OK, ca_result);
+
+ nsapi_error_t connect_result = socket.connect("httpbin.org", 443);
+ TEST_ASSERT_EQUAL(NSAPI_ERROR_OK, connect_result);
+
+ {
+ HttpsRequest *req = new HttpsRequest(&socket, HTTP_GET, "http://httpbin.org/status/404");
+
+ HttpResponse* res = req->send();
+ TEST_ASSERT(res);
+ TEST_ASSERT_EQUAL(404, res->get_status_code());
+
+ delete req;
+ }
+
+ {
+ HttpsRequest *req = new HttpsRequest(&socket, HTTP_GET, "http://httpbin.org/status/403");
+
+ HttpResponse* res = req->send();
+ TEST_ASSERT(res);
+ TEST_ASSERT_EQUAL(403, res->get_status_code());
+
+ delete req;
+ }
+
+ return CaseNext;
+}
+
+// Spread the message out over 3 different chunks
+const char * chunks[] = {
+ "{\"message\":",
+ "\"this is an example",
+ " of chunked encoding\"}"
+};
+
+int chunk_ix = 0;
+
+// Callback function, grab the next chunk and return it
+const void * get_chunk(uint32_t* out_size) {
+ // If you don't have any data left, set out_size to 0 and return a null pointer
+ if (chunk_ix == (sizeof(chunks) / sizeof(chunks[0]))) {
+ *out_size = 0;
+ return NULL;
+ }
+ const char *chunk = chunks[chunk_ix];
+ *out_size = strlen(chunk);
+ chunk_ix++;
+
+ return chunk;
+}
+
+static control_t chunked_request(const size_t call_count) {
+ setup_verify_network();
+
+ HttpsRequest *req = new HttpsRequest(network, SSL_CA_PEM, HTTP_POST, "https://reqres.in/api/users");
+ req->set_header("Content-Type", "application/json");
+
+ HttpResponse* res = req->send(&get_chunk);
+ TEST_ASSERT(res);
+ TEST_ASSERT_EQUAL(201, res->get_status_code());
+ TEST_ASSERT(res->get_body_length() > 0);
+ TEST_ASSERT_NOT_EQUAL(res->get_body_as_string().find("message"), string::npos);
+ TEST_ASSERT_NOT_EQUAL(res->get_body_as_string().find("this is an example of chunked encoding"), string::npos);
+ TEST_ASSERT_NOT_EQUAL(res->get_body_as_string().find("createdAt"), string::npos);
+
+ delete req;
+
+ return CaseNext;
+}
+
+utest::v1::status_t greentea_setup(const size_t number_of_cases) {
+ GREENTEA_SETUP(1*60, "default_auto");
+ return greentea_test_setup_handler(number_of_cases);
+}
+
+Case cases[] = {
+ Case("http get", http_get),
+ Case("http post", http_post),
+ Case("http socket reuse", http_socket_reuse),
+ Case("https get", https_get),
+ Case("https post", https_post),
+ Case("https socket reuse", https_socket_reuse),
+ Case("chunked request", chunked_request)
+};
+
+Specification specification(greentea_setup, cases);
+
+void blink_led() {
+ static DigitalOut led(LED1);
+ led = !led;
+}
+
+int main() {
+ Ticker t;
+ t.attach(blink_led, 0.5);
+
+ return !Harness::run(specification);
+}