2019-09-10 22:05:28 +02:00
|
|
|
#include <ESPAsyncWebServer.h>
|
2020-06-05 20:55:01 +02:00
|
|
|
#include "FilesystemAbstraction.h"
|
|
|
|
|
2019-09-10 22:05:28 +02:00
|
|
|
|
2020-05-19 20:31:09 +02:00
|
|
|
#define FLASH_TEXT(name) const char *name
|
2019-09-10 22:05:28 +02:00
|
|
|
|
2020-05-19 20:31:09 +02:00
|
|
|
namespace webdav_constants
|
|
|
|
{
|
2019-09-10 22:05:28 +02:00
|
|
|
FLASH_TEXT(MULTISTATUS_START) = "<?xml version=\"1.0\" ?><D:multistatus xmlns:D=\"DAV:\">";
|
|
|
|
FLASH_TEXT(MULTISTATUS_END) = "</D:multistatus>";
|
|
|
|
FLASH_TEXT(RESPONSE_START) = "<D:response>";
|
2020-05-19 20:31:09 +02:00
|
|
|
FLASH_TEXT(RESPONSE_END) = "</D:response>\n";
|
2019-09-10 22:05:28 +02:00
|
|
|
FLASH_TEXT(HREF_START) = "<D:href>";
|
|
|
|
FLASH_TEXT(HREF_END) = "</D:href>";
|
|
|
|
FLASH_TEXT(PROPSTAT_START) = "<D:propstat>";
|
|
|
|
FLASH_TEXT(PROPSTAT_END) = "</D:propstat>";
|
|
|
|
FLASH_TEXT(PROP_START) = "<D:prop>";
|
|
|
|
FLASH_TEXT(PROP_END) = "</D:prop>";
|
|
|
|
FLASH_TEXT(RESOURCETYPE_START) = "<D:resourcetype>";
|
|
|
|
FLASH_TEXT(RESOURCETYPE_END) = "</D:resourcetype>";
|
|
|
|
FLASH_TEXT(RESOURCE_COLLECTION) = "<D:collection/>";
|
2020-05-16 12:33:53 +02:00
|
|
|
FLASH_TEXT(HTTP_204_NO_CONTENT) = "HTTP/1.1 204 No Content";
|
2019-09-10 22:05:28 +02:00
|
|
|
|
|
|
|
FLASH_TEXT(CONTENTLEN_START) = "<D:getcontentlength>";
|
|
|
|
FLASH_TEXT(CONTENTLEN_END) = "</D:getcontentlength>";
|
|
|
|
FLASH_TEXT(CREATEDATE_START) = "<D:creationdate>";
|
|
|
|
FLASH_TEXT(CREATEDATE_END) = "</D:creationdate>";
|
|
|
|
FLASH_TEXT(MODDATE_START) = "<D:getlastmodified>";
|
|
|
|
FLASH_TEXT(MODDATE_END) = "</D:getlastmodified>";
|
|
|
|
FLASH_TEXT(STATUS_OK) = "<D:status>HTTP/1.1 200 OK</D:status>";
|
2020-05-19 20:31:09 +02:00
|
|
|
} // namespace webdav_constants
|
2019-09-10 22:05:28 +02:00
|
|
|
|
2020-05-19 20:31:09 +02:00
|
|
|
class WebdavFileListCallback
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
WebdavFileListCallback(const String &path)
|
|
|
|
: path_(path), headerWritten_(false), finished_(false)
|
|
|
|
{
|
2020-06-05 20:55:01 +02:00
|
|
|
dir_ = portablefs::openDir(path);
|
2020-05-19 20:31:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
size_t operator()(uint8_t *buffer, size_t maxLen, size_t index)
|
|
|
|
{
|
|
|
|
Serial.print("index ");
|
|
|
|
Serial.println(index);
|
|
|
|
|
|
|
|
using namespace webdav_constants;
|
|
|
|
uint8_t *bufferStart = buffer;
|
|
|
|
|
|
|
|
if (finished_)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (!headerWritten_)
|
|
|
|
{
|
|
|
|
toBuffer(buffer, MULTISTATUS_START);
|
|
|
|
headerWritten_ = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool fileFound = false;
|
|
|
|
while (dir_.next())
|
|
|
|
{
|
|
|
|
if (isFirstFileOfTrainingGroup())
|
|
|
|
{
|
|
|
|
fileFound = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fileFound)
|
|
|
|
{
|
2020-05-19 22:03:10 +02:00
|
|
|
//toBuffer(buffer, path_.c_str());
|
2020-05-19 20:31:09 +02:00
|
|
|
toBuffer(buffer, RESPONSE_START);
|
|
|
|
toBuffer(buffer, HREF_START);
|
2020-05-19 22:03:10 +02:00
|
|
|
const auto fileName = dir_.fileName();
|
|
|
|
const auto fileNameWithoutDir = fileName.substring(fileName.lastIndexOf("/") + 1);
|
|
|
|
String fileBaseName = fileNameWithoutDir.substring(0, fileNameWithoutDir.indexOf('_'));
|
2020-05-19 20:31:09 +02:00
|
|
|
fileBaseName += ".st";
|
|
|
|
toBuffer(buffer, fileBaseName.c_str());
|
|
|
|
toBuffer(buffer, HREF_END);
|
|
|
|
toBuffer(buffer, PROPSTAT_START);
|
|
|
|
toBuffer(buffer, PROP_START);
|
|
|
|
if (dir_.isDirectory())
|
|
|
|
{
|
|
|
|
toBuffer(buffer, RESOURCETYPE_START);
|
|
|
|
toBuffer(buffer, RESOURCE_COLLECTION);
|
|
|
|
toBuffer(buffer, RESOURCETYPE_END);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
toBuffer(buffer, CONTENTLEN_START);
|
2020-05-19 22:03:10 +02:00
|
|
|
String fileSizeStr(getFileSize(fileName));
|
2020-05-19 20:31:09 +02:00
|
|
|
toBuffer(buffer, fileSizeStr.c_str());
|
|
|
|
toBuffer(buffer, CONTENTLEN_END);
|
|
|
|
}
|
|
|
|
|
|
|
|
toBuffer(buffer, PROP_END);
|
|
|
|
toBuffer(buffer, STATUS_OK);
|
|
|
|
toBuffer(buffer, PROPSTAT_END);
|
|
|
|
toBuffer(buffer, webdav_constants::RESPONSE_END);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
toBuffer(buffer, MULTISTATUS_END);
|
|
|
|
finished_ = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t bytesWritten = buffer - bufferStart;
|
2020-06-05 20:55:01 +02:00
|
|
|
assert_msg(bytesWritten < maxLen, "Written too much!");
|
|
|
|
//Serial.print("Bytes written ");
|
|
|
|
//Serial.println(bytesWritten);
|
|
|
|
//Serial.print("Max bytes ");
|
|
|
|
//Serial.println(maxLen);
|
2020-05-19 20:31:09 +02:00
|
|
|
return bytesWritten;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
bool isFirstFileOfTrainingGroup()
|
|
|
|
{
|
|
|
|
return !dir_.isDirectory() && dir_.fileName().endsWith("_0");
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t getFileSize(const String &fileZero)
|
|
|
|
{
|
|
|
|
size_t size = 0;
|
|
|
|
auto fileBase = fileZero.substring(0, fileZero.indexOf('_'));
|
2020-06-05 20:55:01 +02:00
|
|
|
auto newDirInstance = portablefs::openDir(path_);
|
2020-05-19 20:31:09 +02:00
|
|
|
while (newDirInstance.next())
|
|
|
|
if (newDirInstance.isFile() && newDirInstance.fileName().startsWith(fileBase))
|
|
|
|
size += newDirInstance.fileSize();
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
void toBuffer(uint8_t *&buffer, const char *text)
|
|
|
|
{
|
|
|
|
auto len = strlen(text);
|
|
|
|
memcpy(buffer, text, len);
|
|
|
|
buffer += len;
|
|
|
|
}
|
|
|
|
|
2020-06-05 20:55:01 +02:00
|
|
|
portablefs::Dir dir_;
|
2020-05-19 20:31:09 +02:00
|
|
|
const String path_;
|
|
|
|
bool headerWritten_;
|
|
|
|
bool finished_;
|
|
|
|
};
|
|
|
|
|
2020-05-19 22:03:10 +02:00
|
|
|
bool deleteMeasurementFiles(const String &stName, const String &folder)
|
2019-09-10 22:05:28 +02:00
|
|
|
{
|
2020-05-19 22:03:10 +02:00
|
|
|
String baseName = folder + "/" + stName.substring(0, stName.indexOf("."));
|
|
|
|
int counter = 0;
|
2020-05-19 20:31:09 +02:00
|
|
|
{
|
2020-06-05 20:55:01 +02:00
|
|
|
auto d = portablefs::openDir(folder);
|
2020-05-19 22:03:10 +02:00
|
|
|
while (d.next())
|
|
|
|
if (d.isFile() && d.fileName().startsWith(baseName))
|
|
|
|
++counter;
|
|
|
|
}
|
|
|
|
if (counter == 0)
|
|
|
|
return false;
|
2019-09-10 22:05:28 +02:00
|
|
|
|
2020-05-19 22:03:10 +02:00
|
|
|
for (int i = 0; i < counter; ++i)
|
|
|
|
{
|
|
|
|
const String pathToDelete = baseName + "_" + String(i);
|
|
|
|
if (!SPIFFS.remove(pathToDelete))
|
|
|
|
return false;
|
2019-09-10 22:05:28 +02:00
|
|
|
}
|
2020-05-19 22:03:10 +02:00
|
|
|
return true;
|
2019-09-10 22:05:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
class SpiffsWebDavHandler : public AsyncWebHandler
|
|
|
|
{
|
|
|
|
public:
|
2020-05-19 20:31:09 +02:00
|
|
|
SpiffsWebDavHandler(const String &prefix, const String &folder)
|
2019-09-10 22:05:28 +02:00
|
|
|
: prefix_(prefix), folder_(folder)
|
2020-05-19 20:31:09 +02:00
|
|
|
{
|
|
|
|
}
|
2019-09-10 22:05:28 +02:00
|
|
|
|
|
|
|
virtual bool canHandle(AsyncWebServerRequest *request) override final
|
|
|
|
{
|
|
|
|
Serial.print("Can handle for url : ");
|
|
|
|
Serial.println(request->url());
|
|
|
|
return request->url().startsWith(prefix_);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void handleRequest(AsyncWebServerRequest *request) override final
|
|
|
|
{
|
2020-05-19 20:31:09 +02:00
|
|
|
if (request->url() == prefix_ + "/" && (request->method() == HTTP_GET || request->method() == HTTP_PROPFIND))
|
|
|
|
{
|
2020-05-19 22:03:10 +02:00
|
|
|
// send chunked response - it is too large to send in one go
|
2020-05-19 20:31:09 +02:00
|
|
|
auto response = request->beginChunkedResponse("application/xml",
|
|
|
|
WebdavFileListCallback(folder_));
|
2019-09-10 22:05:28 +02:00
|
|
|
request->send(response);
|
2020-05-19 20:31:09 +02:00
|
|
|
} /*
|
|
|
|
else if(request->url() == prefix_ + "/" && request->method() == HTTP_GET) {
|
2020-05-16 12:33:53 +02:00
|
|
|
AsyncResponseStream * response = request->beginResponseStream("text/plain", 1460*10);
|
2019-09-10 22:05:28 +02:00
|
|
|
Dir dir = SPIFFS.openDir(folder_);
|
|
|
|
Serial.print("Opening folder ");
|
|
|
|
Serial.println(folder_);
|
|
|
|
while (dir.next()) {
|
|
|
|
Serial.print(" File: ");
|
|
|
|
Serial.println(dir.fileName());
|
|
|
|
response->println(dir.fileName());
|
|
|
|
}
|
|
|
|
request->send(response);
|
2020-05-19 20:31:09 +02:00
|
|
|
}*/
|
|
|
|
else if (request->method() == HTTP_GET)
|
|
|
|
{
|
2019-09-10 22:05:28 +02:00
|
|
|
auto path = folder_ + request->url().substring(prefix_.length());
|
2020-05-19 20:31:09 +02:00
|
|
|
if (SPIFFS.exists(path))
|
2019-09-10 22:05:28 +02:00
|
|
|
request->send(SPIFFS, path, "application/x-msgpack");
|
2020-05-19 20:31:09 +02:00
|
|
|
else
|
2019-09-10 22:05:28 +02:00
|
|
|
request->send(404, "text/plain", "Webdav: File not found");
|
2020-05-19 20:31:09 +02:00
|
|
|
}
|
|
|
|
else if (request->method() == HTTP_DELETE)
|
|
|
|
{
|
2020-05-19 22:03:10 +02:00
|
|
|
auto stFileName = request->url().substring(prefix_.length() + 1);
|
|
|
|
Serial.print("HTTP_DELETE for ");
|
|
|
|
Serial.println(stFileName);
|
|
|
|
bool deleteSuccessful = deleteMeasurementFiles(stFileName, folder_);
|
|
|
|
if (deleteSuccessful)
|
|
|
|
request->send(204, "text/plain", "Success");
|
2020-05-19 20:31:09 +02:00
|
|
|
else
|
2020-05-16 12:33:53 +02:00
|
|
|
request->send(404, "text/plain", "Webdav: File to delete not found");
|
2020-05-19 20:31:09 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-09-10 22:05:28 +02:00
|
|
|
request->send(404, "text/plain", "Webdav: Invalid request");
|
|
|
|
}
|
|
|
|
}
|
2020-05-19 20:31:09 +02:00
|
|
|
virtual bool isRequestHandlerTrivial() override final { return false; }
|
2019-09-10 22:05:28 +02:00
|
|
|
|
|
|
|
private:
|
|
|
|
String prefix_;
|
|
|
|
String folder_;
|
|
|
|
};
|