Fix OPDS browser OOM

This commit is contained in:
Konstantin Vukolov 2026-01-17 01:32:59 +03:00
parent 21277e03eb
commit d34aea6378
7 changed files with 121 additions and 38 deletions

View File

@ -4,6 +4,14 @@
#include <cstring> #include <cstring>
OpdsParser::OpdsParser() {
parser = XML_ParserCreate(nullptr);
if (!parser) {
errorOccured = true;
Serial.printf("[%lu] [OPDS] Couldn't allocate memory for parser\n", millis());
}
}
OpdsParser::~OpdsParser() { OpdsParser::~OpdsParser() {
if (parser) { if (parser) {
XML_StopParser(parser, XML_FALSE); XML_StopParser(parser, XML_FALSE);
@ -14,13 +22,9 @@ OpdsParser::~OpdsParser() {
} }
} }
bool OpdsParser::parse(const char* xmlData, const size_t length) { void OpdsParser::push(const char* xmlData, const size_t length) {
clear(); if (errorOccured) {
return;
parser = XML_ParserCreate(nullptr);
if (!parser) {
Serial.printf("[%lu] [OPDS] Couldn't allocate memory for parser\n", millis());
return false;
} }
XML_SetUserData(parser, this); XML_SetUserData(parser, this);
@ -35,34 +39,40 @@ bool OpdsParser::parse(const char* xmlData, const size_t length) {
while (remaining > 0) { while (remaining > 0) {
void* const buf = XML_GetBuffer(parser, chunkSize); void* const buf = XML_GetBuffer(parser, chunkSize);
if (!buf) { if (!buf) {
errorOccured = true;
Serial.printf("[%lu] [OPDS] Couldn't allocate memory for buffer\n", millis()); Serial.printf("[%lu] [OPDS] Couldn't allocate memory for buffer\n", millis());
XML_ParserFree(parser); XML_ParserFree(parser);
parser = nullptr; parser = nullptr;
return false; return;
} }
const size_t toRead = remaining < chunkSize ? remaining : chunkSize; const size_t toRead = remaining < chunkSize ? remaining : chunkSize;
memcpy(buf, currentPos, toRead); memcpy(buf, currentPos, toRead);
const bool isFinal = (remaining == toRead); if (XML_ParseBuffer(parser, static_cast<int>(toRead), 0) == XML_STATUS_ERROR) {
if (XML_ParseBuffer(parser, static_cast<int>(toRead), isFinal) == XML_STATUS_ERROR) { errorOccured = true;
Serial.printf("[%lu] [OPDS] Parse error at line %lu: %s\n", millis(), XML_GetCurrentLineNumber(parser), Serial.printf("[%lu] [OPDS] Parse error at line %lu: %s\n", millis(), XML_GetCurrentLineNumber(parser),
XML_ErrorString(XML_GetErrorCode(parser))); XML_ErrorString(XML_GetErrorCode(parser)));
XML_ParserFree(parser); XML_ParserFree(parser);
parser = nullptr; parser = nullptr;
return false; return;
} }
currentPos += toRead; currentPos += toRead;
remaining -= toRead; remaining -= toRead;
} }
}
// Clean up parser void OpdsParser::finish() {
if (XML_Parse(parser, nullptr, 0, XML_TRUE) != XML_STATUS_OK) {
errorOccured = true;
XML_ParserFree(parser); XML_ParserFree(parser);
parser = nullptr; parser = nullptr;
}
}
Serial.printf("[%lu] [OPDS] Parsed %zu entries\n", millis(), entries.size()); bool OpdsParser::error() const {
return true; return errorOccured;
} }
void OpdsParser::clear() { void OpdsParser::clear() {

View File

@ -44,26 +44,23 @@ using OpdsBook = OpdsEntry;
*/ */
class OpdsParser { class OpdsParser {
public: public:
OpdsParser() = default; OpdsParser();
~OpdsParser(); ~OpdsParser();
// Disable copy // Disable copy
OpdsParser(const OpdsParser&) = delete; OpdsParser(const OpdsParser&) = delete;
OpdsParser& operator=(const OpdsParser&) = delete; OpdsParser& operator=(const OpdsParser&) = delete;
/** void push(const char* xmlData, size_t length);
* Parse an OPDS XML feed. void finish();
* @param xmlData Pointer to the XML data bool error() const;
* @param length Length of the XML data
* @return true if parsing succeeded, false on error
*/
bool parse(const char* xmlData, size_t length);
/** /**
* Get the parsed entries (both navigation and book entries). * Get the parsed entries (both navigation and book entries).
* @return Vector of OpdsEntry entries * @return Vector of OpdsEntry entries
*/ */
const std::vector<OpdsEntry>& getEntries() const { return entries; } const std::vector<OpdsEntry>& getEntries() const & { return entries; }
std::vector<OpdsEntry> getEntries() && { return std::move(entries); }
/** /**
* Get only book entries (legacy compatibility). * Get only book entries (legacy compatibility).
@ -96,4 +93,6 @@ class OpdsParser {
bool inAuthor = false; bool inAuthor = false;
bool inAuthorName = false; bool inAuthorName = false;
bool inId = false; bool inId = false;
bool errorOccured = false;
}; };

View File

@ -0,0 +1,30 @@
#include "OpdsStream.h"
OpdsParserStream::OpdsParserStream(OpdsParser& parser) : parser(parser) {}
int OpdsParserStream::available() {
return 0;
}
int OpdsParserStream::peek() {
abort();
}
int OpdsParserStream::read() {
abort();
}
size_t OpdsParserStream::write(uint8_t c) {
parser.push(reinterpret_cast<const char*>(&c), 1);
return 1;
}
size_t OpdsParserStream::write(const uint8_t *buffer, size_t size) {
parser.push(reinterpret_cast<const char*>(buffer), size);
return size;
}
OpdsParserStream::~OpdsParserStream() {
parser.finish();
}

View File

@ -0,0 +1,23 @@
#pragma once
#include "OpdsParser.h"
#include <Stream.h>
class OpdsParserStream : public Stream {
public:
OpdsParserStream(OpdsParser& parser);
// That functions are not implimented for that stream
int available() override;
int peek() override;
int read() override;
virtual size_t write(uint8_t c) override;
virtual size_t write(const uint8_t *buffer, size_t size) override;
~OpdsParserStream() override;
private:
OpdsParser& parser;
};

View File

@ -4,6 +4,8 @@
#include <HardwareSerial.h> #include <HardwareSerial.h>
#include <WiFi.h> #include <WiFi.h>
#include <OpdsStream.h>
#include "CrossPointSettings.h" #include "CrossPointSettings.h"
#include "MappedInputManager.h" #include "MappedInputManager.h"
#include "ScreenComponents.h" #include "ScreenComponents.h"
@ -264,23 +266,28 @@ void OpdsBookBrowserActivity::fetchFeed(const std::string& path) {
std::string url = UrlUtils::buildUrl(serverUrl, path); std::string url = UrlUtils::buildUrl(serverUrl, path);
Serial.printf("[%lu] [OPDS] Fetching: %s\n", millis(), url.c_str()); Serial.printf("[%lu] [OPDS] Fetching: %s\n", millis(), url.c_str());
std::string content; OpdsParser parser;
if (!HttpDownloader::fetchUrl(url, content)) {
{
OpdsParserStream stream{parser};
if (!HttpDownloader::fetchUrl(url, stream)) {
state = BrowserState::ERROR; state = BrowserState::ERROR;
errorMessage = "Failed to fetch feed"; errorMessage = "Failed to fetch feed";
updateRequired = true; updateRequired = true;
return; return;
} }
}
OpdsParser parser;
if (!parser.parse(content.c_str(), content.size())) { if (parser.error()) {
state = BrowserState::ERROR; state = BrowserState::ERROR;
errorMessage = "Failed to parse feed"; errorMessage = "Failed to parse feed";
updateRequired = true; updateRequired = true;
return; return;
} }
entries = parser.getEntries(); entries = std::move(parser).getEntries();
Serial.printf("[%lu] [OPDS] Found %d entries\n", millis(), entries.size());
selectorIndex = 0; selectorIndex = 0;
if (entries.empty()) { if (entries.empty()) {

View File

@ -5,11 +5,13 @@
#include <WiFiClient.h> #include <WiFiClient.h>
#include <WiFiClientSecure.h> #include <WiFiClientSecure.h>
#include <StreamString.h>
#include <memory> #include <memory>
#include "util/UrlUtils.h" #include "util/UrlUtils.h"
bool HttpDownloader::fetchUrl(const std::string& url, std::string& outContent) { bool HttpDownloader::fetchUrl(const std::string& url, Stream& outContent) {
// Use WiFiClientSecure for HTTPS, regular WiFiClient for HTTP // Use WiFiClientSecure for HTTPS, regular WiFiClient for HTTP
std::unique_ptr<WiFiClient> client; std::unique_ptr<WiFiClient> client;
if (UrlUtils::isHttpsUrl(url)) { if (UrlUtils::isHttpsUrl(url)) {
@ -34,10 +36,20 @@ bool HttpDownloader::fetchUrl(const std::string& url, std::string& outContent) {
return false; return false;
} }
outContent = http.getString().c_str(); http.writeToStream(&outContent);
http.end(); http.end();
Serial.printf("[%lu] [HTTP] Fetched %zu bytes\n", millis(), outContent.size()); Serial.printf("[%lu] [HTTP] Fetch success\n", millis());
return true;
}
bool HttpDownloader::fetchUrl(const std::string& url, std::string& outContent) {
StreamString stream;
if (!fetchUrl(url, stream)) {
return false;
}
outContent = stream.c_str();
return true; return true;
} }

View File

@ -27,6 +27,8 @@ class HttpDownloader {
*/ */
static bool fetchUrl(const std::string& url, std::string& outContent); static bool fetchUrl(const std::string& url, std::string& outContent);
static bool fetchUrl(const std::string& url, Stream& stream);
/** /**
* Download a file to the SD card. * Download a file to the SD card.
* @param url The URL to download * @param url The URL to download