You need to sign in or sign up before continuing.
Commit 9df8b32f authored by Ryan Walklin's avatar Ryan Walklin

Add albumart command

Add API documentation Support 64 bit offsets Use InputStream for all reads
parent 6f37f575
...@@ -1573,6 +1573,42 @@ OK ...@@ -1573,6 +1573,42 @@ OK
<title>The music database</title> <title>The music database</title>
<variablelist> <variablelist>
<varlistentry id="album_art">
<term>
<cmdsynopsis>
<command>albumart</command>
<arg choice="req"><replaceable>URI</replaceable></arg>
<arg choice="req"><replaceable>OFFSET</replaceable></arg>
</cmdsynopsis>
</term>
<listitem>
<para>
Searches the directory the file <varname>URI</varname>
resides in and attempts to return a chunk of an album
art image file at offset <varname>OFFSET</varname>.
</para>
<para>
Uses the filename "cover" with any of ".png, .jpg,
.tiff, .bmp".
</para>
<para>
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.
</para>
<para>
Example:
</para>
<programlisting>albumart
size: 1024768
binary: 8192
&lt;8192 bytes&gt;
OK
</programlisting>
</listitem>
</varlistentry>
<varlistentry id="command_count"> <varlistentry id="command_count">
<term> <term>
......
...@@ -84,6 +84,7 @@ static constexpr struct command commands[] = { ...@@ -84,6 +84,7 @@ static constexpr struct command commands[] = {
{ "add", PERMISSION_ADD, 1, 1, handle_add }, { "add", PERMISSION_ADD, 1, 1, handle_add },
{ "addid", PERMISSION_ADD, 1, 2, handle_addid }, { "addid", PERMISSION_ADD, 1, 2, handle_addid },
{ "addtagid", PERMISSION_ADD, 3, 3, handle_addtagid }, { "addtagid", PERMISSION_ADD, 3, 3, handle_addtagid },
{ "albumart", PERMISSION_READ, 2, 2, handle_album_art },
{ "channels", PERMISSION_READ, 0, 0, handle_channels }, { "channels", PERMISSION_READ, 0, 0, handle_channels },
{ "clear", PERMISSION_CONTROL, 0, 0, handle_clear }, { "clear", PERMISSION_CONTROL, 0, 0, handle_clear },
{ "clearerror", PERMISSION_CONTROL, 0, 0, handle_clearerror }, { "clearerror", PERMISSION_CONTROL, 0, 0, handle_clearerror },
......
...@@ -36,8 +36,11 @@ ...@@ -36,8 +36,11 @@
#include "fs/AllocatedPath.hxx" #include "fs/AllocatedPath.hxx"
#include "fs/FileInfo.hxx" #include "fs/FileInfo.hxx"
#include "fs/DirectoryReader.hxx" #include "fs/DirectoryReader.hxx"
#include "input/InputStream.hxx"
#include "LocateUri.hxx" #include "LocateUri.hxx"
#include "TimePrint.hxx" #include "TimePrint.hxx"
#include "thread/Mutex.hxx"
#include "thread/Cond.hxx"
#include <assert.h> #include <assert.h>
#include <inttypes.h> /* for PRIu64 */ #include <inttypes.h> /* for PRIu64 */
...@@ -233,3 +236,114 @@ handle_read_comments(Client &client, Request args, Response &r) ...@@ -233,3 +236,114 @@ handle_read_comments(Client &client, Request args, Response &r)
gcc_unreachable(); 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;
}
...@@ -33,4 +33,7 @@ handle_listfiles_local(Response &response, Path path_fs); ...@@ -33,4 +33,7 @@ handle_listfiles_local(Response &response, Path path_fs);
CommandResult CommandResult
handle_read_comments(Client &client, Request request, Response &response); handle_read_comments(Client &client, Request request, Response &response);
CommandResult
handle_album_art(Client &client, Request request, Response &response);
#endif #endif
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment