From bf1f1076a8e4b47bb5eb0d712e300b06c1a3c3e2 Mon Sep 17 00:00:00 2001 From: Jeroen88 Date: Sun, 1 Sep 2019 12:24:56 +0200 Subject: [PATCH 1/9] Because of git problems, start from a new fork and create a new PR. This was PR #6457 --- .../ReuseConnectionV2/ReuseConnectionV2.ino | 76 +++++++++++++++++++ .../src/ESP8266HTTPClient.cpp | 19 +++-- 2 files changed, 90 insertions(+), 5 deletions(-) create mode 100644 libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino diff --git a/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino b/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino new file mode 100644 index 0000000000..5a208e282b --- /dev/null +++ b/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino @@ -0,0 +1,76 @@ +/** + reuseConnectionV2.ino + + Created on: 22.11.2015 + + This example reuses the http connection and also restores the connection if the connection is lost +*/ + + +#include + +#include +#include + +#include + +ESP8266WiFiMulti WiFiMulti; + +HTTPClient http; +WiFiClient client; + +void setup() { + + Serial.begin(115200); + // Serial.setDebugOutput(true); + + Serial.println(); + Serial.println(); + Serial.println("Connecting to WiFi..."); + + WiFi.mode(WIFI_STA); + WiFiMulti.addAP("SSID", "PASSWORD"); + + // wait for WiFi connection + while ((WiFiMulti.run() != WL_CONNECTED)) { + Serial.write('.'); + delay(500); + } + Serial.println(" connected to WiFi"); + + // allow reuse (if server supports it) + http.setReuse(true); + + + http.begin(client, "http://jigsaw.w3.org/HTTP/connection.html"); + //http.begin(client, "jigsaw.w3.org", 80, "/HTTP/connection.html"); +} + +void loop() { + for (int i = 0; i < 10; i++) { + Serial.printf("Reuse connection example, GET url for the %d time\n", i + 1); + int httpCode = http.GET(); + if (httpCode > 0) { + Serial.printf("[HTTP] GET... code: %d\n", httpCode); + + // file found at server + if (httpCode == HTTP_CODE_OK) { + http.writeToStream(&Serial); + } + } else { + Serial.printf("[HTTP] GET... failed, error: %s\n", http.errorToString(httpCode).c_str()); + // Something went wrong with the connection, try to reconnect + http.end(); + http.begin(client, "http://jigsaw.w3.org/HTTP/connection.html"); + //http.begin(client, "jigsaw.w3.org", 80, "/HTTP/connection.html"); + } + + Serial.println("\n\n\nWait 5 second...\n"); + delay(5000); + } + + http.end(); + + Serial.println("Done testing, now wait forever"); + for (;;) delay(100); // Wait forever +} diff --git a/libraries/ESP8266HTTPClient/src/ESP8266HTTPClient.cpp b/libraries/ESP8266HTTPClient/src/ESP8266HTTPClient.cpp index 55608f0d3b..b553c404d3 100644 --- a/libraries/ESP8266HTTPClient/src/ESP8266HTTPClient.cpp +++ b/libraries/ESP8266HTTPClient/src/ESP8266HTTPClient.cpp @@ -456,6 +456,10 @@ void HTTPClient::disconnect(bool preserveClient) #endif } } else { + if (!preserveClient && _client) { // Also destroy _client if not connected() + _client = nullptr; + } + DEBUG_HTTPCLIENT("[HTTP-Client][end] tcp is closed\n"); } } @@ -922,7 +926,9 @@ int HTTPClient::writeToStream(Stream * stream) return returnError(HTTPC_ERROR_NO_STREAM); } - if(!connected()) { + // Only return error if not connected and no data available, because otherwise ::getString() will return an error instead of an empty + // string when the server returned a http code 204 (no content) + if(!connected() && _transferEncoding != HTTPC_TE_IDENTITY && _size > 0) { return returnError(HTTPC_ERROR_NOT_CONNECTED); } @@ -931,11 +937,13 @@ int HTTPClient::writeToStream(Stream * stream) int ret = 0; if(_transferEncoding == HTTPC_TE_IDENTITY) { - ret = writeToStreamDataBlock(stream, len); + if(len > 0) { + ret = writeToStreamDataBlock(stream, len); - // have we an error? - if(ret < 0) { - return returnError(ret); + // have we an error? + if(ret < 0) { + return returnError(ret); + } } } else if(_transferEncoding == HTTPC_TE_CHUNKED) { int size = 0; @@ -1289,6 +1297,7 @@ int HTTPClient::handleHeaderResponse() _canReuse = (headerLine[sizeof "HTTP/1." - 1] != '0'); } _returnCode = headerLine.substring(9, headerLine.indexOf(' ', 9)).toInt(); + _canReuse = (_returnCode != '0'); } else if(headerLine.indexOf(':')) { String headerName = headerLine.substring(0, headerLine.indexOf(':')); String headerValue = headerLine.substring(headerLine.indexOf(':') + 1); From 6e80ffd5e58610bab0b947234e37d044bbd8b9b2 Mon Sep 17 00:00:00 2001 From: Jeroen88 Date: Sun, 1 Sep 2019 13:57:15 +0200 Subject: [PATCH 2/9] Style update to pass Travis --- .../examples/ReuseConnectionV2/ReuseConnectionV2.ino | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino b/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino index 5a208e282b..65e8b8a07b 100644 --- a/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino +++ b/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino @@ -72,5 +72,7 @@ void loop() { http.end(); Serial.println("Done testing, now wait forever"); - for (;;) delay(100); // Wait forever + for (;;) { + delay(100); // Wait forever + } } From a1f275f051c9dbd034ea02d546bd3af3d867107a Mon Sep 17 00:00:00 2001 From: david gauchard Date: Wed, 18 Dec 2019 11:22:46 +0100 Subject: [PATCH 3/9] Update ReuseConnectionV2.ino --- .../examples/ReuseConnectionV2/ReuseConnectionV2.ino | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino b/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino index 65e8b8a07b..d1c04927d0 100644 --- a/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino +++ b/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino @@ -7,13 +7,15 @@ */ -#include - #include #include - #include +#ifndef STASSID +#define STASSID "your-ssid" +#define STAPSK "your-password" +#endif + ESP8266WiFiMulti WiFiMulti; HTTPClient http; @@ -29,7 +31,7 @@ void setup() { Serial.println("Connecting to WiFi..."); WiFi.mode(WIFI_STA); - WiFiMulti.addAP("SSID", "PASSWORD"); + WiFiMulti.addAP(STASSID, STAPSK); // wait for WiFi connection while ((WiFiMulti.run() != WL_CONNECTED)) { From c7a19f55c80390068abe2278bad5267853e5cc30 Mon Sep 17 00:00:00 2001 From: david gauchard Date: Sat, 11 Apr 2020 01:23:18 +0200 Subject: [PATCH 4/9] fix + enforce testing http code per @earlephilhower review --- libraries/ESP8266HTTPClient/src/ESP8266HTTPClient.cpp | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/libraries/ESP8266HTTPClient/src/ESP8266HTTPClient.cpp b/libraries/ESP8266HTTPClient/src/ESP8266HTTPClient.cpp index 8a9cacb4a5..0ea4b61ae3 100644 --- a/libraries/ESP8266HTTPClient/src/ESP8266HTTPClient.cpp +++ b/libraries/ESP8266HTTPClient/src/ESP8266HTTPClient.cpp @@ -1351,9 +1351,10 @@ int HTTPClient::handleHeaderResponse() if (headerLine.startsWith(F("HTTP/1."))) { - _canReuse = _canReuse && (headerLine[sizeof "HTTP/1." - 1] != '0'); - _returnCode = headerLine.substring(9, headerLine.indexOf(' ', 9)).toInt(); - _canReuse = _canReuse && (_returnCode != '0'); + constexpr auto httpVersionIdx = sizeof "HTTP/1." - 1; + _canReuse = _canReuse && (headerLine[httpVersionIdx] != '0'); + _returnCode = headerLine.substring(httpVersionIdx + 2, headerLine.indexOf(' ', httpVersionIdx + 2)).toInt(); + _canReuse = _canReuse && (_returnCode > 0) && (_returnCode < 500); } else if ((headerSeparator = headerLine.indexOf(':')) > 0) { String headerName = headerLine.substring(0, headerSeparator); From f947b86de6c3fcfff5bb4d3e436313ea8e0d7153 Mon Sep 17 00:00:00 2001 From: "Earle F. Philhower, III" Date: Fri, 15 May 2020 08:53:29 -0700 Subject: [PATCH 5/9] Close connection before ::connecting on HTTP/1.0 HTTPClient never actually closes the TCP connection on its own. It will leave the TCP connection open unless you explicitly do a getString which makes a StreamString and stuffs it with the HTTP server response, at which point the HTTP server itself will close the connection. If you check the HTTP error code and find failure, unless you do a getString and throw it away, it won't disconnect. Even in HTTP/1.0 or in cases when you haven't enabled _reuse. Change the logic in ::connect to only reuse the connection when it is specifically allowed. Otherwise, fall back to re-connection. --- libraries/ESP8266HTTPClient/src/ESP8266HTTPClient.cpp | 8 ++------ 1 file changed, 2 insertions(+), 6 deletions(-) diff --git a/libraries/ESP8266HTTPClient/src/ESP8266HTTPClient.cpp b/libraries/ESP8266HTTPClient/src/ESP8266HTTPClient.cpp index 0ea4b61ae3..85d27d5c6c 100644 --- a/libraries/ESP8266HTTPClient/src/ESP8266HTTPClient.cpp +++ b/libraries/ESP8266HTTPClient/src/ESP8266HTTPClient.cpp @@ -1206,12 +1206,8 @@ bool HTTPClient::hasHeader(const char* name) */ bool HTTPClient::connect(void) { - if(connected()) { - if(_reuse) { - DEBUG_HTTPCLIENT("[HTTP-Client] connect: already connected, reusing connection\n"); - } else { - DEBUG_HTTPCLIENT("[HTTP-Client] connect: already connected, try reuse!\n"); - } + if(_reuse && _canReuse && connected()) { + DEBUG_HTTPCLIENT("[HTTP-Client] connect: already connected, reusing connection\n"); while(_client->available() > 0) { _client->read(); } From e92905e3473f7eb256a97068a5f9a0f7b89da237 Mon Sep 17 00:00:00 2001 From: "Earle F. Philhower, III" Date: Fri, 15 May 2020 09:07:24 -0700 Subject: [PATCH 6/9] Adjust example per request Do single URL get in each loop, avoid infinite for loop at end. --- .../ReuseConnectionV2/ReuseConnectionV2.ino | 26 +++++++++++-------- 1 file changed, 15 insertions(+), 11 deletions(-) diff --git a/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino b/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino index d1c04927d0..0b36fc7027 100644 --- a/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino +++ b/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino @@ -48,8 +48,12 @@ void setup() { //http.begin(client, "jigsaw.w3.org", 80, "/HTTP/connection.html"); } +int pass = 0; + void loop() { - for (int i = 0; i < 10; i++) { + // First 10 loop()s, retrieve the URL + if (pass < 10) { + pass++; Serial.printf("Reuse connection example, GET url for the %d time\n", i + 1); int httpCode = http.GET(); if (httpCode > 0) { @@ -66,15 +70,15 @@ void loop() { http.begin(client, "http://jigsaw.w3.org/HTTP/connection.html"); //http.begin(client, "jigsaw.w3.org", 80, "/HTTP/connection.html"); } - - Serial.println("\n\n\nWait 5 second...\n"); - delay(5000); - } - - http.end(); - - Serial.println("Done testing, now wait forever"); - for (;;) { - delay(100); // Wait forever + + if (pass == 10) { + // On the 11th loop(), c + pass++; + http.end(); + Serial.println("Done testing, now wait forever"); + } else { + Serial.println("\n\n\nWait 5 second...\n"); + delay(5000); + } } } From d708587a2d377d18b9c3fd4dd19593cfefcfbc61 Mon Sep 17 00:00:00 2001 From: "Earle F. Philhower, III" Date: Fri, 15 May 2020 09:14:22 -0700 Subject: [PATCH 7/9] Fix astyle --- .../examples/ReuseConnectionV2/ReuseConnectionV2.ino | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino b/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino index 0b36fc7027..3a3354d09d 100644 --- a/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino +++ b/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino @@ -70,7 +70,7 @@ void loop() { http.begin(client, "http://jigsaw.w3.org/HTTP/connection.html"); //http.begin(client, "jigsaw.w3.org", 80, "/HTTP/connection.html"); } - + if (pass == 10) { // On the 11th loop(), c pass++; From 3782845aae88e45036fb2ce05fbd7953e4ca75aa Mon Sep 17 00:00:00 2001 From: "Earle F. Philhower, III" Date: Fri, 15 May 2020 09:15:57 -0700 Subject: [PATCH 8/9] Clean up final pass notice --- .../examples/ReuseConnectionV2/ReuseConnectionV2.ino | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino b/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino index 3a3354d09d..9c6ab020a5 100644 --- a/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino +++ b/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino @@ -72,10 +72,8 @@ void loop() { } if (pass == 10) { - // On the 11th loop(), c - pass++; http.end(); - Serial.println("Done testing, now wait forever"); + Serial.println("Done testing"); } else { Serial.println("\n\n\nWait 5 second...\n"); delay(5000); From e62e6a084827b7d8ac3be8e178515e04c3822a46 Mon Sep 17 00:00:00 2001 From: "Earle F. Philhower, III" Date: Fri, 15 May 2020 09:23:39 -0700 Subject: [PATCH 9/9] Fix example syntax error Editing code in a web textbox without running it is a painful process. --- .../examples/ReuseConnectionV2/ReuseConnectionV2.ino | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino b/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino index 9c6ab020a5..42a89beced 100644 --- a/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino +++ b/libraries/ESP8266HTTPClient/examples/ReuseConnectionV2/ReuseConnectionV2.ino @@ -54,7 +54,7 @@ void loop() { // First 10 loop()s, retrieve the URL if (pass < 10) { pass++; - Serial.printf("Reuse connection example, GET url for the %d time\n", i + 1); + Serial.printf("Reuse connection example, GET url for the %d time\n", pass); int httpCode = http.GET(); if (httpCode > 0) { Serial.printf("[HTTP] GET... code: %d\n", httpCode);