Commit 72b6c09a authored by Max Kellermann's avatar Max Kellermann

db/simple: add an AudioFormat to each Song

parent c05bca6f
......@@ -1932,6 +1932,9 @@ test_DumpDatabase_SOURCES = test/DumpDatabase.cxx \
src/db/Selection.cxx \
src/db/PlaylistVector.cxx \
src/db/DatabaseLock.cxx \
src/AudioFormat.cxx \
src/AudioParser.cxx \
src/pcm/SampleFormat.cxx \
src/SongSave.cxx \
src/DetachedSong.cxx \
src/TagSave.cxx \
......
......@@ -5,6 +5,8 @@ ver 0.21 (not yet released)
- "outputs" prints the plugin name
- "outputset" sets runtime attributes
- close connection when client sends HTTP request
* database
- simple: scan audio formats
* player
- "one-shot" single mode
* input
......
......@@ -258,6 +258,17 @@
<listitem>
<para>
<varname>Format</varname>: the audio format of the song
(or an approximation to a format supported by MPD and the
decoder plugin being used). When playing this file, the
<varname>audio</varname> value in the <link
linkend="command_status"><command>status</command></link>
response should be the same.
</para>
</listitem>
<listitem>
<para>
<varname>Last-Modified</varname>: the time stamp of the
last modification of the underlying file in ISO 8601
format. Example:
......
......@@ -92,6 +92,9 @@ song_print_info(Response &r, const LightSong &song, bool base) noexcept
if (!IsNegative(song.mtime))
time_print(r, "Last-Modified", song.mtime);
if (song.audio_format.IsDefined())
r.Format("Format: %s\n", ToString(song.audio_format).c_str());
tag_print(r, song.tag);
}
......
......@@ -19,6 +19,7 @@
#include "config.h"
#include "SongSave.hxx"
#include "AudioParser.hxx"
#include "db/plugins/simple/Song.hxx"
#include "DetachedSong.hxx"
#include "TagSave.hxx"
......@@ -28,6 +29,8 @@
#include "tag/Tag.hxx"
#include "tag/Builder.hxx"
#include "util/ChronoUtil.hxx"
#include "util/StringAPI.hxx"
#include "util/StringBuffer.hxx"
#include "util/StringStrip.hxx"
#include "util/RuntimeError.hxx"
#include "util/NumberParser.hxx"
......@@ -56,6 +59,9 @@ song_save(BufferedOutputStream &os, const Song &song)
tag_save(os, song.tag);
if (song.audio_format.IsDefined())
os.Format("Format: %s\n", ToString(song.audio_format).c_str());
if (!IsNegative(song.mtime))
os.Format(SONG_MTIME ": %li\n",
(long)std::chrono::system_clock::to_time_t(song.mtime));
......@@ -78,7 +84,8 @@ song_save(BufferedOutputStream &os, const DetachedSong &song)
}
std::unique_ptr<DetachedSong>
song_load(TextFile &file, const char *uri)
song_load(TextFile &file, const char *uri,
AudioFormat *audio_format_r)
{
auto song = std::make_unique<DetachedSong>(uri);
......@@ -100,6 +107,15 @@ song_load(TextFile &file, const char *uri)
tag.AddItem(type, value);
} else if (strcmp(line, "Time") == 0) {
tag.SetDuration(SignedSongTime::FromS(ParseDouble(value)));
} else if (StringIsEqual(line, "Format")) {
if (audio_format_r != nullptr) {
try {
*audio_format_r =
ParseAudioFormat(value, false);
} catch (...) {
/* ignore parser errors */
}
}
} else if (strcmp(line, "Playlist") == 0) {
tag.SetHasPlaylist(strcmp(value, "yes") == 0);
} else if (strcmp(line, SONG_MTIME) == 0) {
......
......@@ -25,6 +25,7 @@
#define SONG_BEGIN "song_begin: "
struct Song;
struct AudioFormat;
class DetachedSong;
class BufferedOutputStream;
class TextFile;
......@@ -42,6 +43,7 @@ song_save(BufferedOutputStream &os, const DetachedSong &song);
* Throws #std::runtime_error on error.
*/
std::unique_ptr<DetachedSong>
song_load(TextFile &file, const char *uri);
song_load(TextFile &file, const char *uri,
AudioFormat *audio_format_r=nullptr);
#endif
......@@ -77,19 +77,23 @@ Song::UpdateFile(Storage &storage) noexcept
return false;
TagBuilder tag_builder;
auto new_audio_format = AudioFormat::Undefined();
const auto path_fs = storage.MapFS(relative_uri.c_str());
if (path_fs.IsNull()) {
const auto absolute_uri =
storage.MapUTF8(relative_uri.c_str());
if (!tag_stream_scan(absolute_uri.c_str(), tag_builder))
if (!tag_stream_scan(absolute_uri.c_str(), tag_builder,
&new_audio_format))
return false;
} else {
if (!ScanFileTagsWithGeneric(path_fs, tag_builder))
if (!ScanFileTagsWithGeneric(path_fs, tag_builder,
&new_audio_format))
return false;
}
mtime = info.mtime;
audio_format = new_audio_format;
tag_builder.Commit(tag);
return true;
}
......
......@@ -21,6 +21,7 @@
#define MPD_LIGHT_SONG_HXX
#include "Chrono.hxx"
#include "AudioFormat.hxx"
#include "Compiler.h"
#include <string>
......@@ -78,6 +79,12 @@ struct LightSong {
*/
SongTime end_time = SongTime::zero();
/**
* The audio format of the song, if given by the decoder
* plugin. May be undefined if unknown.
*/
AudioFormat audio_format = AudioFormat::Undefined();
LightSong(const char *_uri, const Tag &_tag) noexcept
:uri(_uri), tag(_tag) {}
......
......@@ -161,10 +161,15 @@ directory_load(TextFile &file, Directory &directory)
if (directory.FindSong(name) != nullptr)
throw FormatRuntimeError("Duplicate song '%s'", name);
auto song = song_load(file, name);
auto audio_format = AudioFormat::Undefined();
auto detached_song = song_load(file, name,
&audio_format);
directory.AddSong(Song::NewFrom(std::move(*song),
directory));
auto song = Song::NewFrom(std::move(*detached_song),
directory);
song->audio_format = audio_format;
directory.AddSong(song);
} else if ((p = StringAfterPrefix(line, PLAYLIST_META_BEGIN))) {
const char *name = p;
playlist_metadata_load(file, directory.playlists, name);
......
......@@ -104,5 +104,6 @@ Song::Export() const noexcept
dest.mtime = mtime;
dest.start_time = start_time;
dest.end_time = end_time;
dest.audio_format = audio_format;
return dest;
}
......@@ -23,6 +23,7 @@
#include "check.h"
#include "Chrono.hxx"
#include "tag/Tag.hxx"
#include "AudioFormat.hxx"
#include "Compiler.h"
#include <boost/intrusive/list.hpp>
......@@ -89,6 +90,12 @@ struct Song {
SongTime end_time = SongTime::zero();
/**
* The audio format of the song, if given by the decoder
* plugin. May be undefined if unknown.
*/
AudioFormat audio_format = AudioFormat::Undefined();
/**
* The file name.
*/
char uri[sizeof(int)];
......
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