diff --git a/doc/protocol.xml b/doc/protocol.xml
index 692edaded..608a196a8 100644
--- a/doc/protocol.xml
+++ b/doc/protocol.xml
@@ -1573,6 +1573,42 @@ OK
The music database
+
+
+
+ albumart
+ URI
+ OFFSET
+
+
+
+
+ Searches the directory the file URI
+ resides in and attempts to return a chunk of an album
+ art image file at offset OFFSET.
+
+
+ Uses the filename "cover" with any of ".png, .jpg,
+ .tiff, .bmp".
+
+
+ Returns the file size and actual number
+ of bytes read at the requested offset, followed
+ by the chunk requested as raw bytes, then a
+ newline and the completion code.
+
+
+ Example:
+
+
+ albumart
+size: 1024768
+binary: 8192
+<8192 bytes>
+OK
+
+
+
diff --git a/src/command/AllCommands.cxx b/src/command/AllCommands.cxx
index ef4d34ded..3a4934dd8 100644
--- a/src/command/AllCommands.cxx
+++ b/src/command/AllCommands.cxx
@@ -84,6 +84,7 @@ static constexpr struct command commands[] = {
{ "add", PERMISSION_ADD, 1, 1, handle_add },
{ "addid", PERMISSION_ADD, 1, 2, handle_addid },
{ "addtagid", PERMISSION_ADD, 3, 3, handle_addtagid },
+ { "albumart", PERMISSION_READ, 2, 2, handle_album_art },
{ "channels", PERMISSION_READ, 0, 0, handle_channels },
{ "clear", PERMISSION_CONTROL, 0, 0, handle_clear },
{ "clearerror", PERMISSION_CONTROL, 0, 0, handle_clearerror },
diff --git a/src/command/FileCommands.cxx b/src/command/FileCommands.cxx
index b7e4bbbbf..ac8e805e3 100644
--- a/src/command/FileCommands.cxx
+++ b/src/command/FileCommands.cxx
@@ -36,8 +36,11 @@
#include "fs/AllocatedPath.hxx"
#include "fs/FileInfo.hxx"
#include "fs/DirectoryReader.hxx"
+#include "input/InputStream.hxx"
#include "LocateUri.hxx"
#include "TimePrint.hxx"
+#include "thread/Mutex.hxx"
+#include "thread/Cond.hxx"
#include
#include /* for PRIu64 */
@@ -233,3 +236,114 @@ handle_read_comments(Client &client, Request args, Response &r)
gcc_unreachable();
}
+
+/**
+ * Searches for the files listed in #artnames in the UTF8 folder
+ * URI #directory. This can be a local path or protocol-based
+ * URI that #InputStream supports. Returns the first successfully
+ * opened file or #nullptr on failure.
+ */
+static InputStreamPtr
+find_stream_art(const char *directory, Mutex &mutex, Cond &cond)
+{
+ static constexpr char const * art_names[] = {
+ "cover.png",
+ "cover.jpg",
+ "cover.tiff",
+ "cover.bmp"
+ };
+
+ for(const auto name: art_names) {
+ std::string art_file = PathTraitsUTF8::Build(directory, name);
+
+ try {
+ return InputStream::OpenReady(art_file.c_str(), mutex, cond);
+ } catch (const std::exception &e) {}
+ }
+ return nullptr;
+}
+
+static CommandResult
+read_stream_art(Response &r, const char *uri, size_t offset)
+{
+ const char *art_directory = PathTraitsUTF8::GetParent(uri).c_str();
+
+ Mutex mutex;
+ Cond cond;
+
+ InputStreamPtr is = find_stream_art(art_directory, mutex, cond);
+
+ if (is == nullptr) {
+ r.Error(ACK_ERROR_NO_EXIST, "No file exists");
+ return CommandResult::ERROR;
+ }
+ if (!is->KnownSize()) {
+ r.Error(ACK_ERROR_NO_EXIST, "Cannot get size for stream");
+ return CommandResult::ERROR;
+ }
+
+ const size_t art_file_size = is->GetSize();
+
+ constexpr size_t CHUNK_SIZE = 8192;
+ uint8_t buffer[CHUNK_SIZE];
+ size_t read_size;
+
+ is->Seek(offset);
+ read_size = is->Read(&buffer, CHUNK_SIZE);
+
+ r.Format("size: %" PRIu64 "\n"
+ "binary: %u\n",
+ art_file_size,
+ read_size
+ );
+
+ r.Write(buffer, read_size);
+ r.Write("\n");
+
+ return CommandResult::OK;
+}
+
+#ifdef ENABLE_DATABASE
+static CommandResult
+read_db_art(Client &client, Response &r, const char *uri, const uint64_t offset)
+{
+ const Storage *storage = client.GetStorage();
+ if (storage == nullptr) {
+ r.Error(ACK_ERROR_NO_EXIST, "No database");
+ return CommandResult::ERROR;
+ }
+ std::string uri2 = storage->MapUTF8(uri);
+ return read_stream_art(r, uri2.c_str(), offset);
+}
+#endif
+
+CommandResult
+handle_album_art(Client &client, Request args, Response &r)
+{
+ assert(args.size == 2);
+
+ const char *uri = args.front();
+ size_t offset = args.ParseUnsigned(1);
+
+ const auto located_uri = LocateUri(uri, &client
+#ifdef ENABLE_DATABASE
+ , nullptr
+#endif
+ );
+
+ switch (located_uri.type) {
+ case LocatedUri::Type::ABSOLUTE:
+ case LocatedUri::Type::PATH:
+ return read_stream_art(r, located_uri.canonical_uri, offset);
+ case LocatedUri::Type::RELATIVE:
+#ifdef ENABLE_DATABASE
+ return read_db_art(client, r, located_uri.canonical_uri, offset);
+#else
+ r.Error(ACK_ERROR_NO_EXIST, "Database disabled");
+ return CommandResult::ERROR;
+#endif
+ }
+ r.Error(ACK_ERROR_NO_EXIST, "No art file exists");
+ return CommandResult::ERROR;
+}
+
diff --git a/src/command/FileCommands.hxx b/src/command/FileCommands.hxx
index 74c158358..6475dd723 100644
--- a/src/command/FileCommands.hxx
+++ b/src/command/FileCommands.hxx
@@ -33,4 +33,7 @@ handle_listfiles_local(Response &response, Path path_fs);
CommandResult
handle_read_comments(Client &client, Request request, Response &response);
+CommandResult
+handle_album_art(Client &client, Request request, Response &response);
+
#endif