TidalInputPlugin.cxx 6.01 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
/*
 * Copyright 2003-2018 The Music Player Daemon Project
 * 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 "TidalInputPlugin.hxx"
#include "TidalSessionManager.hxx"
#include "TidalTrackRequest.hxx"
24
#include "TidalTagScanner.hxx"
25
#include "TidalError.hxx"
26 27 28 29 30 31 32
#include "CurlInputPlugin.hxx"
#include "PluginUnavailable.hxx"
#include "input/ProxyInputStream.hxx"
#include "input/FailingInputStream.hxx"
#include "input/InputPlugin.hxx"
#include "config/Block.hxx"
#include "thread/Mutex.hxx"
33
#include "util/Domain.hxx"
34
#include "util/Exception.hxx"
35
#include "util/StringCompare.hxx"
36
#include "Log.hxx"
37 38 39 40

#include <stdexcept>
#include <memory>

41 42
static constexpr Domain tidal_domain("tidal");

43
static TidalSessionManager *tidal_session;
44
static const char *tidal_audioquality;
45 46 47 48 49 50 51 52 53 54

class TidalInputStream final
	: public ProxyInputStream, TidalSessionHandler, TidalTrackHandler {

	const std::string track_id;

	std::unique_ptr<TidalTrackRequest> track_request;

	std::exception_ptr error;

55 56 57 58 59 60
	/**
	 * Retry to login if TidalError::IsInvalidSession() returns
	 * true?
	 */
	bool retry_login = true;

61 62
public:
	TidalInputStream(const char *_uri, const char *_track_id,
63 64
			 Mutex &_mutex) noexcept
		:ProxyInputStream(_uri, _mutex),
65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83
		 track_id(_track_id)
	{
		tidal_session->AddLoginHandler(*this);
	}

	~TidalInputStream() {
		tidal_session->RemoveLoginHandler(*this);
	}

	/* virtual methods from InputStream */

	void Check() override {
		if (error)
			std::rethrow_exception(error);
	}

private:
	void Failed(std::exception_ptr e) {
		SetInput(std::make_unique<FailingInputStream>(GetURI(), e,
84
							      mutex));
85 86 87 88 89 90
	}

	/* virtual methods from TidalSessionHandler */
	void OnTidalSession() noexcept override;

	/* virtual methods from TidalTrackHandler */
91
	void OnTidalTrackSuccess(std::string url) noexcept override;
92 93 94 95 96 97 98 99 100
	void OnTidalTrackError(std::exception_ptr error) noexcept override;
};

void
TidalInputStream::OnTidalSession() noexcept
{
	const std::lock_guard<Mutex> protect(mutex);

	try {
101
		TidalTrackHandler &h = *this;
102 103 104 105 106
		track_request = std::make_unique<TidalTrackRequest>(tidal_session->GetCurl(),
								    tidal_session->GetBaseUrl(),
								    tidal_session->GetToken(),
								    tidal_session->GetSession().c_str(),
								    track_id.c_str(),
107
								    tidal_audioquality,
108
								    h);
109
		track_request->Start();
110 111 112 113 114 115
	} catch (...) {
		Failed(std::current_exception());
	}
}

void
116
TidalInputStream::OnTidalTrackSuccess(std::string url) noexcept
117
{
118 119 120
	FormatDebug(tidal_domain, "Tidal track '%s' resolves to %s",
		    track_id.c_str(), url.c_str());

121 122
	const std::lock_guard<Mutex> protect(mutex);

123 124
	track_request.reset();

125 126
	try {
		SetInput(OpenCurlInputStream(url.c_str(), {},
127
					     mutex));
128 129 130 131 132
	} catch (...) {
		Failed(std::current_exception());
	}
}

133 134 135 136 137 138 139 140 141 142 143 144 145
gcc_pure
static bool
IsInvalidSession(std::exception_ptr e) noexcept
{
	try {
		std::rethrow_exception(e);
	} catch (const TidalError &te) {
		return te.IsInvalidSession();
	} catch (...) {
		return false;
	}
}

146 147 148 149 150
void
TidalInputStream::OnTidalTrackError(std::exception_ptr e) noexcept
{
	const std::lock_guard<Mutex> protect(mutex);

151 152 153 154 155 156 157 158 159 160 161 162
	if (retry_login && IsInvalidSession(e)) {
		/* the session has expired - obtain a new session id
		   by logging in again */

		FormatInfo(tidal_domain, "Session expired ('%s'), retrying to log in",
			   GetFullMessage(e).c_str());

		retry_login = false;
		tidal_session->AddLoginHandler(*this);
		return;
	}

163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183
	Failed(e);
}

static void
InitTidalInput(EventLoop &event_loop, const ConfigBlock &block)
{
	const char *base_url = block.GetBlockValue("base_url",
						   "https://api.tidal.com/v1");

	const char *token = block.GetBlockValue("token");
	if (token == nullptr)
		throw PluginUnavailable("No Tidal application token configured");

	const char *username = block.GetBlockValue("username");
	if (username == nullptr)
		throw PluginUnavailable("No Tidal username configured");

	const char *password = block.GetBlockValue("password");
	if (password == nullptr)
		throw PluginUnavailable("No Tidal password configured");

184
	tidal_audioquality = block.GetBlockValue("audioquality", "HIGH");
185 186 187 188 189 190 191 192 193 194 195

	tidal_session = new TidalSessionManager(event_loop, base_url, token,
						username, password);
}

static void
FinishTidalInput()
{
	delete tidal_session;
}

196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212
gcc_pure
static const char *
ExtractTidalTrackId(const char *uri)
{
	const char *track_id = StringAfterPrefix(uri, "tidal://track/");
	if (track_id == nullptr) {
		track_id = StringAfterPrefix(uri, "https://listen.tidal.com/track/");
		if (track_id == nullptr)
			return nullptr;
	}

	if (*track_id == 0)
		return nullptr;

	return track_id;
}

213
static InputStreamPtr
214
OpenTidalInput(const char *uri, Mutex &mutex)
215 216 217
{
	assert(tidal_session != nullptr);

218
	const char *track_id = ExtractTidalTrackId(uri);
219 220 221 222 223
	if (track_id == nullptr)
		return nullptr;

	// TODO: validate track_id

224
	return std::make_unique<TidalInputStream>(uri, track_id, mutex);
225 226
}

227 228 229 230 231 232 233 234 235 236 237 238 239 240 241
static std::unique_ptr<RemoteTagScanner>
ScanTidalTags(const char *uri, RemoteTagHandler &handler)
{
	assert(tidal_session != nullptr);

	const char *track_id = ExtractTidalTrackId(uri);
	if (track_id == nullptr)
		return nullptr;

	return std::make_unique<TidalTagScanner>(tidal_session->GetCurl(),
						 tidal_session->GetBaseUrl(),
						 tidal_session->GetToken(),
						 track_id, handler);
}

242 243 244 245 246
const InputPlugin tidal_input_plugin = {
	"tidal",
	InitTidalInput,
	FinishTidalInput,
	OpenTidalInput,
247
	ScanTidalTags,
248
};