mirror of
				https://github.com/esp8266/Arduino.git
				synced 2025-10-24 07:13:45 +03:00 
			
		
		
		
	add Response header handling
This commit is contained in:
		| @@ -31,14 +31,24 @@ | |||||||
| httpClient::httpClient() { | httpClient::httpClient() { | ||||||
|     _tcp = NULL; |     _tcp = NULL; | ||||||
|     _tcps = NULL; |     _tcps = NULL; | ||||||
|  |  | ||||||
|  |     _headerKeysCount = 0; | ||||||
|  |     _currentHeaders = NULL; | ||||||
|  |  | ||||||
|  |     _returnCode = 0; | ||||||
|  |     _size = 0; | ||||||
| } | } | ||||||
|  |  | ||||||
| httpClient::~httpClient() { | httpClient::~httpClient() { | ||||||
|     if(connected()) { |     if(connected()) { | ||||||
|         _tcp->stop(); |         _tcp->stop(); | ||||||
|     } |     } | ||||||
| } |  | ||||||
|  |  | ||||||
|  |     if(_currentHeaders) { | ||||||
|  |         delete[] _currentHeaders; | ||||||
|  |     } | ||||||
|  |     _headerKeysCount = 0; | ||||||
|  | } | ||||||
|  |  | ||||||
| void httpClient::begin(const char *host, uint16_t port, const char * url, bool https, const char * httpsFingerprint) { | void httpClient::begin(const char *host, uint16_t port, const char * url, bool https, const char * httpsFingerprint) { | ||||||
|     _host = host; |     _host = host; | ||||||
| @@ -63,7 +73,11 @@ bool httpClient::connected() { | |||||||
|     return false; |     return false; | ||||||
| } | } | ||||||
|  |  | ||||||
| bool httpClient::GET() { | /** | ||||||
|  |  * send a GET request | ||||||
|  |  * @return http code | ||||||
|  |  */ | ||||||
|  | int httpClient::GET() { | ||||||
|  |  | ||||||
|     bool status; |     bool status; | ||||||
|     status = connect(); |     status = connect(); | ||||||
| @@ -71,16 +85,20 @@ bool httpClient::GET() { | |||||||
|         status = sendHeader("GET"); |         status = sendHeader("GET"); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     return status; |     if(status) { | ||||||
|  |         return handleHeaderResponse(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return 0; | ||||||
| } | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * sends a post request to the server |  * sends a post request to the server | ||||||
|  * @param payload uint8_t * |  * @param payload uint8_t * | ||||||
|  * @param size size_t |  * @param size size_t | ||||||
|  * @return status |  * @return http code | ||||||
|  */ |  */ | ||||||
| bool httpClient::POST(uint8_t * payload, size_t size) { | int httpClient::POST(uint8_t * payload, size_t size) { | ||||||
|  |  | ||||||
|     bool status; |     bool status; | ||||||
|     status = connect(); |     status = connect(); | ||||||
| @@ -93,10 +111,13 @@ bool httpClient::POST(uint8_t * payload, size_t size) { | |||||||
|         status = _tcp->write(&payload[0], size); |         status = _tcp->write(&payload[0], size); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     return status; |     if(status) { | ||||||
|  |         return handleHeaderResponse(); | ||||||
|  |     } | ||||||
|  |     return 0; | ||||||
| } | } | ||||||
|  |  | ||||||
| bool httpClient::POST(String payload) { | int httpClient::POST(String payload) { | ||||||
|     return POST((uint8_t *) payload.c_str(), payload.length()); |     return POST((uint8_t *) payload.c_str(), payload.length()); | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -131,6 +152,56 @@ void httpClient::addHeader(const String& name, const String& value, bool first) | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | void httpClient::collectHeaders(const char* headerKeys[], const size_t headerKeysCount) { | ||||||
|  |     _headerKeysCount = headerKeysCount; | ||||||
|  |     if(_currentHeaders) | ||||||
|  |         delete[] _currentHeaders; | ||||||
|  |     _currentHeaders = new RequestArgument[_headerKeysCount]; | ||||||
|  |     for(int i = 0; i < _headerKeysCount; i++) { | ||||||
|  |         _currentHeaders[i].key = headerKeys[i]; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | String httpClient::header(const char* name) { | ||||||
|  |     for(int i = 0; i < _headerKeysCount; ++i) { | ||||||
|  |         if(_currentHeaders[i].key == name) | ||||||
|  |             return _currentHeaders[i].value; | ||||||
|  |     } | ||||||
|  |     return String(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | String httpClient::header(int i) { | ||||||
|  |     if(i < _headerKeysCount) | ||||||
|  |         return _currentHeaders[i].value; | ||||||
|  |     return String(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | String httpClient::headerName(int i) { | ||||||
|  |     if(i < _headerKeysCount) | ||||||
|  |         return _currentHeaders[i].key; | ||||||
|  |     return String(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | int httpClient::headers() { | ||||||
|  |     return _headerKeysCount; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | bool httpClient::hasHeader(const char* name) { | ||||||
|  |     for(int i = 0; i < _headerKeysCount; ++i) { | ||||||
|  |         if((_currentHeaders[i].key == name) && (_currentHeaders[i].value.length() > 0)) | ||||||
|  |             return true; | ||||||
|  |     } | ||||||
|  |     return false; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * size of message body / payload | ||||||
|  |  * @return 0 if no info or > 0 when Content-Length is set by server | ||||||
|  |  */ | ||||||
|  | size_t httpClient::getSize(void) { | ||||||
|  |     return _size; | ||||||
|  | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * init TCP connection and handle ssl verify if needed |  * init TCP connection and handle ssl verify if needed | ||||||
|  * @return true if connection is ok |  * @return true if connection is ok | ||||||
| @@ -151,7 +222,6 @@ bool httpClient::connect(void) { | |||||||
|         _tcp = new WiFiClient(); |         _tcp = new WiFiClient(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|     if(!_tcp->connect(_host.c_str(), _port)) { |     if(!_tcp->connect(_host.c_str(), _port)) { | ||||||
|         DEBUG_HTTPCLIENT("[HTTP-Client] failed connect to %s:%u.\n", _host.c_str(), _port); |         DEBUG_HTTPCLIENT("[HTTP-Client] failed connect to %s:%u.\n", _host.c_str(), _port); | ||||||
|         return false; |         return false; | ||||||
| @@ -184,12 +254,64 @@ bool httpClient::connect(void) { | |||||||
|  * @return status |  * @return status | ||||||
|  */ |  */ | ||||||
| bool httpClient::sendHeader(const char * type) { | bool httpClient::sendHeader(const char * type) { | ||||||
|  |     if(!connected()) { | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|     String header = String(type) + " " + _url + " HTTP/1.1\r\n" |     String header = String(type) + " " + _url + " HTTP/1.1\r\n" | ||||||
|             "Host: " + _host + "\r\n" |             "Host: " + _host + "\r\n" | ||||||
|             "User-Agent: ESP8266httpClient\r\n" |             "User-Agent: ESP8266httpClient\r\n" | ||||||
|             "Connection: close\r\n" + |             "Connection: close\r\n" + _Headers + "\r\n"; | ||||||
|             _Headers + |  | ||||||
|             "\r\n"; |  | ||||||
|  |  | ||||||
|     return _tcp->write(header.c_str(), header.length()); |     return _tcp->write(header.c_str(), header.length()); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * reads the respone from the server | ||||||
|  |  * @return int http code | ||||||
|  |  */ | ||||||
|  | int httpClient::handleHeaderResponse() { | ||||||
|  |  | ||||||
|  |     if(!connected()) { | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     while(connected()) { | ||||||
|  |         size_t len = _tcp->available(); | ||||||
|  |         if(len > 0) { | ||||||
|  |             String headerLine = _tcp->readStringUntil('\n'); | ||||||
|  |             headerLine.trim(); // remove \r | ||||||
|  |  | ||||||
|  |             DEBUG_HTTPCLIENT("[HTTP][handleHeaderResponse] RX: '%s'\n", headerLine.c_str()); | ||||||
|  |  | ||||||
|  |             if(headerLine.startsWith("HTTP/1.")) { | ||||||
|  |                 _returnCode = headerLine.substring(9, headerLine.indexOf(' ', 9)).toInt(); | ||||||
|  |             } else if(headerLine.indexOf(':')) { | ||||||
|  |                 String headerName = headerLine.substring(0, headerLine.indexOf(':')); | ||||||
|  |                 String headerValue = headerLine.substring(headerLine.indexOf(':') + 2); | ||||||
|  |  | ||||||
|  |                 if(headerName.equalsIgnoreCase("Content-Length")) { | ||||||
|  |                     _size = headerValue.toInt(); | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 for(size_t i = 0; i < _headerKeysCount; i++) { | ||||||
|  |                     if(_currentHeaders[i].key == headerName) { | ||||||
|  |                         _currentHeaders[i].value = headerValue; | ||||||
|  |                         return true; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             if(headerLine == "") { | ||||||
|  |                 DEBUG_HTTPCLIENT("[HTTP][handleHeaderResponse] code: '%s'\n", String(_returnCode).c_str()); | ||||||
|  |                 if(_size) { | ||||||
|  |                     DEBUG_HTTPCLIENT("[HTTP][handleHeaderResponse] size: '%s'\n", String(_size).c_str()); | ||||||
|  |                 } | ||||||
|  |                 return _returnCode; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |         } else { | ||||||
|  |             delay(0); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|   | |||||||
| @@ -43,20 +43,38 @@ class httpClient { | |||||||
|  |  | ||||||
|         bool connected(void); |         bool connected(void); | ||||||
|  |  | ||||||
|         bool GET(); |         /// request handling | ||||||
|         bool POST(uint8_t * payload, size_t size); |         int GET(); | ||||||
|         bool POST(String payload); |         int POST(uint8_t * payload, size_t size); | ||||||
|  |         int POST(String payload); | ||||||
|  |  | ||||||
|         void addHeader(const String& name, const String& value, bool first = false); |         void addHeader(const String& name, const String& value, bool first = false); | ||||||
|  |  | ||||||
|  |         /// Response handling | ||||||
|  |         void collectHeaders(const char* headerKeys[], const size_t headerKeysCount); | ||||||
|  |         String header(const char* name);   // get request header value by name | ||||||
|  |         String header(int i);              // get request header value by number | ||||||
|  |         String headerName(int i);          // get request header name by number | ||||||
|  |         int headers();                     // get header count | ||||||
|  |         bool hasHeader(const char* name);  // check if header exists | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         size_t getSize(void); | ||||||
|  |  | ||||||
|         WiFiClient & getStream(void); |         WiFiClient & getStream(void); | ||||||
|  |  | ||||||
|     protected: |     protected: | ||||||
|  |  | ||||||
|  |         struct RequestArgument { | ||||||
|  |           String key; | ||||||
|  |           String value; | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |  | ||||||
|         WiFiClient * _tcp; |         WiFiClient * _tcp; | ||||||
|         WiFiClientSecure * _tcps; |         WiFiClientSecure * _tcps; | ||||||
|  |  | ||||||
|  |         /// request handling | ||||||
|         String _host; |         String _host; | ||||||
|         uint16_t _port; |         uint16_t _port; | ||||||
|  |  | ||||||
| @@ -66,10 +84,17 @@ class httpClient { | |||||||
|  |  | ||||||
|         String  _Headers; |         String  _Headers; | ||||||
|  |  | ||||||
|  |         /// Response handling | ||||||
|  |         RequestArgument* _currentHeaders; | ||||||
|  |         size_t           _headerKeysCount; | ||||||
|  |  | ||||||
|  |         int _returnCode; | ||||||
|  |         size_t _size; | ||||||
|  |  | ||||||
|  |  | ||||||
|         bool connect(void); |         bool connect(void); | ||||||
|  |  | ||||||
|         bool sendHeader(const char * type); |         bool sendHeader(const char * type); | ||||||
|  |         int handleHeaderResponse(); | ||||||
|  |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user