mirror of
https://github.com/esp8266/Arduino.git
synced 2025-04-25 20:02:37 +03:00
Rewrite multipart boundary detection (#7728)
Use a simpler, cleaner implementation of multipart form detection as defined in https://tools.ietf.org/html/rfc7578 . Implements a simple state machine that detects the `\r\n--<boundary>` stream in input a character at a time, instead of buffering and comparing in chunks which can miss things due to alignment issues and which also had a problem with replacing characters in a binary stream. Adjust the private _uploadReadByte function to return -1 on error (like a read()), and the main file upload handler to use that return value instead of duplicating logic. Fixes #7723
This commit is contained in:
parent
04b0c270e4
commit
92175d7090
@ -246,7 +246,7 @@ protected:
|
||||
bool _parseForm(ClientType& client, const String& boundary, uint32_t len);
|
||||
bool _parseFormUploadAborted();
|
||||
void _uploadWriteByte(uint8_t b);
|
||||
uint8_t _uploadReadByte(ClientType& client);
|
||||
int _uploadReadByte(ClientType& client);
|
||||
void _prepareHeader(String& response, int code, const char* content_type, size_t contentLength);
|
||||
bool _collectHeader(const char* headerName, const char* headerValue);
|
||||
|
||||
|
@ -347,14 +347,14 @@ void ESP8266WebServerTemplate<ServerType>::_uploadWriteByte(uint8_t b){
|
||||
}
|
||||
|
||||
template <typename ServerType>
|
||||
uint8_t ESP8266WebServerTemplate<ServerType>::_uploadReadByte(ClientType& client){
|
||||
int ESP8266WebServerTemplate<ServerType>::_uploadReadByte(ClientType& client){
|
||||
int res = client.read();
|
||||
if(res == -1){
|
||||
while(!client.available() && client.connected())
|
||||
yield();
|
||||
res = client.read();
|
||||
}
|
||||
return (uint8_t)res;
|
||||
return res;
|
||||
}
|
||||
|
||||
|
||||
@ -444,45 +444,34 @@ bool ESP8266WebServerTemplate<ServerType>::_parseForm(ClientType& client, const
|
||||
_currentHandler->upload(*this, _currentUri, *_currentUpload);
|
||||
_currentUpload->status = UPLOAD_FILE_WRITE;
|
||||
|
||||
int bLen = boundary.length();
|
||||
uint8_t boundBuf[2 + bLen + 1]; // "--" + boundary + null terminator
|
||||
boundBuf[2 + bLen] = '\0';
|
||||
uint8_t argByte;
|
||||
bool first = true;
|
||||
while (1) {
|
||||
//attempt to fill up boundary buffer with length of boundary string
|
||||
int i;
|
||||
for (i = 0; i < 2 + bLen; i++) {
|
||||
if (!client.connected()) return _parseFormUploadAborted();
|
||||
argByte = _uploadReadByte(client);
|
||||
if (argByte == '\r')
|
||||
int fastBoundaryLen = 4 /* \r\n-- */ + boundary.length() + 1 /* \0 */;
|
||||
char fastBoundary[ fastBoundaryLen ];
|
||||
snprintf(fastBoundary, fastBoundaryLen, "\r\n--%s", boundary.c_str());
|
||||
int boundaryPtr = 0;
|
||||
while ( true ) {
|
||||
int ret = _uploadReadByte(client);
|
||||
if (ret < 0) {
|
||||
// Unexpected, we should have had data available per above
|
||||
return _parseFormUploadAborted();
|
||||
}
|
||||
char in = (char) ret;
|
||||
if (in == fastBoundary[ boundaryPtr ]) {
|
||||
// The input matched the current expected character, advance and possibly exit this file
|
||||
boundaryPtr++;
|
||||
if (boundaryPtr == fastBoundaryLen - 1) {
|
||||
// We read the whole boundary line, we're done here!
|
||||
break;
|
||||
boundBuf[i] = argByte;
|
||||
}
|
||||
if ((strncmp((const char*)boundBuf, "--", 2) == 0) && (strcmp((const char*)(boundBuf + 2), boundary.c_str()) == 0))
|
||||
break; //found the boundary, done parsing this file
|
||||
if (first) first = false; //only add newline characters after the first line
|
||||
else {
|
||||
_uploadWriteByte('\r');
|
||||
_uploadWriteByte('\n');
|
||||
}
|
||||
// current line does not contain boundary, upload all bytes in boundary buffer
|
||||
for (int j = 0; j < i; j++)
|
||||
_uploadWriteByte(boundBuf[j]);
|
||||
// the initial pass (filling up the boundary buffer) did not reach the end of the line. Upload the rest of the line now
|
||||
if (i >= 2 + bLen) {
|
||||
if (!client.connected()) return _parseFormUploadAborted();
|
||||
argByte = _uploadReadByte(client);
|
||||
while (argByte != '\r') {
|
||||
if (!client.connected()) return _parseFormUploadAborted();
|
||||
_uploadWriteByte(argByte);
|
||||
argByte = _uploadReadByte(client);
|
||||
}
|
||||
} else {
|
||||
// The char doesn't match what we want, so dump whatever matches we had, the read in char, and reset ptr to start
|
||||
for (int i = 0; i < boundaryPtr; i++) {
|
||||
_uploadWriteByte( fastBoundary[ i ] );
|
||||
}
|
||||
_uploadWriteByte( in );
|
||||
boundaryPtr = 0;
|
||||
}
|
||||
if (!client.connected()) return _parseFormUploadAborted();
|
||||
_uploadReadByte(client); // '\n'
|
||||
}
|
||||
//Found the boundary string, finish processing this file upload
|
||||
// Found the boundary string, finish processing this file upload
|
||||
if (_currentHandler && _currentHandler->canUpload(_currentUri))
|
||||
_currentHandler->upload(*this, _currentUri, *_currentUpload);
|
||||
_currentUpload->totalSize += _currentUpload->currentSize;
|
||||
|
Loading…
x
Reference in New Issue
Block a user