SongLoader.cxx 3.01 KB
Newer Older
1
/*
Max Kellermann's avatar
Max Kellermann committed
2
 * Copyright (C) 2003-2015 The Music Player Daemon Project
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
 * http://www.musicpd.org
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License along
 * with this program; if not, write to the Free Software Foundation, Inc.,
 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
 */

#include "config.h"
#include "SongLoader.hxx"
22
#include "LocateUri.hxx"
23 24
#include "client/Client.hxx"
#include "db/DatabaseSong.hxx"
25
#include "storage/StorageInterface.hxx"
26 27 28 29 30 31
#include "util/Error.hxx"
#include "DetachedSong.hxx"
#include "PlaylistError.hxx"

#include <assert.h>

32 33 34
#ifdef ENABLE_DATABASE

SongLoader::SongLoader(const Client &_client)
35 36
	:client(&_client), db(_client.GetDatabase(IgnoreError())),
	 storage(_client.GetStorage()) {}
37 38 39

#endif

40 41 42 43 44 45 46 47 48 49 50 51 52 53 54
DetachedSong *
SongLoader::LoadFromDatabase(const char *uri, Error &error) const
{
#ifdef ENABLE_DATABASE
	if (db != nullptr)
		return DatabaseDetachSong(*db, *storage, uri, error);
#else
	(void)uri;
#endif

	error.Set(playlist_domain, int(PlaylistResult::NO_SUCH_SONG),
		  "No database");
	return nullptr;
}

55
DetachedSong *
56
SongLoader::LoadFile(const char *path_utf8, Path path_fs, Error &error) const
57 58
{
#ifdef ENABLE_DATABASE
59 60 61 62 63
	if (storage != nullptr) {
		const char *suffix = storage->MapToRelativeUTF8(path_utf8);
		if (suffix != nullptr)
			/* this path was relative to the music
			   directory - obtain it from the database */
64
			return LoadFromDatabase(suffix, error);
65
	}
66 67 68
#endif

	DetachedSong *song = new DetachedSong(path_utf8);
69
	if (!song->LoadFile(path_fs)) {
70 71 72 73 74 75 76 77 78
		error.Set(playlist_domain, int(PlaylistResult::NO_SUCH_SONG),
			  "No such file");
		delete song;
		return nullptr;
	}

	return song;
}

79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99
DetachedSong *
SongLoader::LoadSong(const LocatedUri &located_uri, Error &error) const
{
	switch (located_uri.type) {
	case LocatedUri::Type::UNKNOWN:
		gcc_unreachable();

	case LocatedUri::Type::ABSOLUTE:
		return new DetachedSong(located_uri.canonical_uri);

	case LocatedUri::Type::RELATIVE:
		return LoadFromDatabase(located_uri.canonical_uri, error);

	case LocatedUri::Type::PATH:
		return LoadFile(located_uri.canonical_uri, located_uri.path,
				error);
	}

	gcc_unreachable();
}

100 101 102
DetachedSong *
SongLoader::LoadSong(const char *uri_utf8, Error &error) const
{
103 104
#if !CLANG_CHECK_VERSION(3,6)
	/* disabled on clang due to -Wtautological-pointer-compare */
105
	assert(uri_utf8 != nullptr);
106
#endif
107

108 109 110 111 112 113 114 115 116
	const auto located_uri = LocateUri(uri_utf8, client,
#ifdef ENABLE_DATABASE
					   storage,
#endif
					   error);
	if (located_uri.IsUnknown())
		return nullptr;

	return LoadSong(located_uri, error);
117
}