mirror of
https://github.com/daveallie/crosspoint-reader.git
synced 2026-02-06 07:37:37 +03:00
Fix OPDS browser OOM
This commit is contained in:
parent
21277e03eb
commit
d34aea6378
@ -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,14 +22,10 @@ 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);
|
||||||
XML_SetElementHandler(parser, startElement, endElement);
|
XML_SetElementHandler(parser, startElement, endElement);
|
||||||
@ -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() {
|
||||||
XML_ParserFree(parser);
|
if (XML_Parse(parser, nullptr, 0, XML_TRUE) != XML_STATUS_OK) {
|
||||||
parser = nullptr;
|
errorOccured = true;
|
||||||
|
XML_ParserFree(parser);
|
||||||
|
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() {
|
||||||
|
|||||||
@ -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;
|
||||||
};
|
};
|
||||||
|
|||||||
30
lib/OpdsParser/OpdsStream.cpp
Normal file
30
lib/OpdsParser/OpdsStream.cpp
Normal 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();
|
||||||
|
}
|
||||||
23
lib/OpdsParser/OpdsStream.h
Normal file
23
lib/OpdsParser/OpdsStream.h
Normal 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;
|
||||||
|
};
|
||||||
@ -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)) {
|
|
||||||
state = BrowserState::ERROR;
|
{
|
||||||
errorMessage = "Failed to fetch feed";
|
OpdsParserStream stream{parser};
|
||||||
updateRequired = true;
|
if (!HttpDownloader::fetchUrl(url, stream)) {
|
||||||
return;
|
state = BrowserState::ERROR;
|
||||||
|
errorMessage = "Failed to fetch feed";
|
||||||
|
updateRequired = true;
|
||||||
|
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()) {
|
||||||
|
|||||||
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -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
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user