mpd/src/output/plugins/httpd/HttpdOutputPlugin.cxx

446 lines
9.2 KiB
C++
Raw Normal View History

/*
2018-10-31 17:54:59 +01:00
* 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"
2013-01-15 18:22:17 +01:00
#include "HttpdOutputPlugin.hxx"
#include "HttpdInternal.hxx"
#include "HttpdClient.hxx"
#include "output/OutputAPI.hxx"
#include "encoder/EncoderInterface.hxx"
#include "encoder/Configured.hxx"
#include "net/UniqueSocketDescriptor.hxx"
#include "net/SocketAddress.hxx"
#include "net/ToString.hxx"
2013-01-30 09:18:52 +01:00
#include "Page.hxx"
2013-01-30 09:08:50 +01:00
#include "IcyMetaDataServer.hxx"
#include "event/Call.hxx"
#include "util/Domain.hxx"
#include "util/DeleteDisposer.hxx"
#include "Log.hxx"
#include "config/Net.hxx"
#include <assert.h>
2013-07-30 20:11:57 +02:00
#include <string.h>
#include <errno.h>
#ifdef HAVE_LIBWRAP
#include <sys/socket.h> /* needed for AF_LOCAL */
#include <tcpd.h>
#endif
const Domain httpd_output_domain("httpd_output");
inline
HttpdOutput::HttpdOutput(EventLoop &_loop, const ConfigBlock &block)
:AudioOutput(FLAG_ENABLE_DISABLE|FLAG_PAUSE),
ServerSocket(_loop),
prepared_encoder(CreateConfiguredEncoder(block)),
defer_broadcast(_loop, BIND_THIS_METHOD(OnDeferredBroadcast))
{
/* read configuration */
name = block.GetBlockValue("name", "Set name in config");
genre = block.GetBlockValue("genre", "Set genre in config");
website = block.GetBlockValue("website", "Set website in config");
clients_max = block.GetBlockValue("max_clients", 0u);
/* set up bind_to_address */
ServerSocketAddGeneric(*this, block.GetBlockValue("bind_to_address"), block.GetBlockValue("port", 8000u));
/* determine content type */
content_type = prepared_encoder->GetMimeType();
if (content_type == nullptr)
content_type = "application/octet-stream";
}
inline void
HttpdOutput::Bind()
{
open = false;
BlockingCall(GetEventLoop(), [this](){
ServerSocket::Open();
});
}
inline void
HttpdOutput::Unbind()
{
assert(!open);
BlockingCall(GetEventLoop(), [this](){
ServerSocket::Close();
});
}
/**
2013-01-15 18:22:17 +01:00
* Creates a new #HttpdClient object and adds it into the
* HttpdOutput.clients linked list.
*/
inline void
HttpdOutput::AddClient(UniqueSocketDescriptor fd)
{
auto *client = new HttpdClient(*this, std::move(fd), GetEventLoop(),
!encoder->ImplementsTag());
clients.push_front(*client);
/* pass metadata to client */
if (metadata != nullptr)
clients.front().PushMetaData(metadata);
}
void
HttpdOutput::OnDeferredBroadcast() noexcept
{
/* this method runs in the IOThread; it broadcasts pages from
our own queue to all clients */
const std::lock_guard<Mutex> protect(mutex);
while (!pages.empty()) {
PagePtr page = std::move(pages.front());
pages.pop();
for (auto &client : clients)
client.PushPage(page);
}
/* wake up the client that may be waiting for the queue to be
flushed */
cond.broadcast();
}
void
HttpdOutput::OnAccept(UniqueSocketDescriptor fd,
SocketAddress address, gcc_unused int uid) noexcept
{
/* the listener socket has become readable - a client has
connected */
#ifdef HAVE_LIBWRAP
if (address.GetFamily() != AF_LOCAL) {
const auto hostaddr = ToString(address);
// TODO: shall we obtain the program name from argv[0]?
const char *progname = "mpd";
struct request_info req;
request_init(&req, RQ_FILE, fd.Get(), RQ_DAEMON, progname, 0);
fromhost(&req);
if (!hosts_access(&req)) {
/* tcp wrappers says no */
FormatWarning(httpd_output_domain,
"libwrap refused connection (libwrap=%s) from %s",
progname, hostaddr.c_str());
return;
}
}
#else
(void)address;
#endif /* HAVE_WRAP */
const std::lock_guard<Mutex> protect(mutex);
/* can we allow additional client */
if (open && (clients_max == 0 || clients.size() < clients_max))
AddClient(std::move(fd));
}
PagePtr
HttpdOutput::ReadPage()
{
if (unflushed_input >= 65536) {
/* we have fed a lot of input into the encoder, but it
didn't give anything back yet - flush now to avoid
buffer underruns */
try {
encoder->Flush();
} catch (...) {
/* ignore */
}
unflushed_input = 0;
}
size_t size = 0;
do {
size_t nbytes = encoder->Read(buffer + size,
sizeof(buffer) - size);
if (nbytes == 0)
break;
unflushed_input = 0;
size += nbytes;
} while (size < sizeof(buffer));
if (size == 0)
2013-10-28 23:58:17 +01:00
return nullptr;
return std::make_shared<Page>(buffer, size);
}
inline void
HttpdOutput::OpenEncoder(AudioFormat &audio_format)
{
encoder = prepared_encoder->Open(audio_format);
/* we have to remember the encoder header, i.e. the first
bytes of encoder output after opening it, because it has to
be sent to every new client */
header = ReadPage();
unflushed_input = 0;
}
void
HttpdOutput::Open(AudioFormat &audio_format)
{
assert(!open);
assert(clients.empty());
2017-01-25 10:03:17 +01:00
const std::lock_guard<Mutex> protect(mutex);
OpenEncoder(audio_format);
/* initialize other attributes */
2013-08-03 21:00:50 +02:00
timer = new Timer(audio_format);
open = true;
pause = false;
}
void
HttpdOutput::Close() noexcept
{
2017-01-25 10:03:17 +01:00
assert(open);
2013-05-12 15:03:42 +02:00
delete timer;
BlockingCall(GetEventLoop(), [this](){
defer_broadcast.Cancel();
const std::lock_guard<Mutex> protect(mutex);
open = false;
clients.clear_and_dispose(DeleteDisposer());
});
header.reset();
delete encoder;
}
void
HttpdOutput::RemoveClient(HttpdClient &client)
{
assert(!clients.empty());
clients.erase_and_dispose(clients.iterator_to(client),
DeleteDisposer());
}
void
HttpdOutput::SendHeader(HttpdClient &client) const
{
2013-10-28 23:58:17 +01:00
if (header != nullptr)
client.PushPage(header);
}
std::chrono::steady_clock::duration
HttpdOutput::Delay() const noexcept
2010-11-05 09:42:14 +01:00
{
if (!LockHasClients() && pause) {
/* if there's no client and this output is paused,
then httpd_output_pause() will not do anything, it
will not fill the buffer and it will not update the
timer; therefore, we reset the timer here */
timer->Reset();
/* some arbitrary delay that is long enough to avoid
consuming too much CPU, and short enough to notice
new clients quickly enough */
return std::chrono::seconds(1);
}
return timer->IsStarted()
2016-12-28 10:11:07 +01:00
? timer->GetDelay()
: std::chrono::steady_clock::duration::zero();
2010-11-05 09:42:14 +01:00
}
void
HttpdOutput::BroadcastPage(PagePtr page)
{
2013-10-28 23:58:17 +01:00
assert(page != nullptr);
2017-02-19 20:32:21 +01:00
{
const std::lock_guard<Mutex> lock(mutex);
pages.emplace(std::move(page));
}
defer_broadcast.Schedule();
}
void
HttpdOutput::BroadcastFromEncoder()
{
/* synchronize with the IOThread */
2017-02-19 20:32:21 +01:00
{
const std::lock_guard<Mutex> lock(mutex);
while (!pages.empty())
cond.wait(mutex);
}
2017-02-19 20:32:21 +01:00
bool empty = true;
PagePtr page;
while ((page = ReadPage()) != nullptr) {
const std::lock_guard<Mutex> lock(mutex);
pages.emplace(std::move(page));
empty = false;
2017-02-19 20:32:21 +01:00
}
if (!empty)
defer_broadcast.Schedule();
}
inline void
HttpdOutput::EncodeAndPlay(const void *chunk, size_t size)
{
encoder->Write(chunk, size);
unflushed_input += size;
BroadcastFromEncoder();
}
size_t
HttpdOutput::Play(const void *chunk, size_t size)
{
pause = false;
if (LockHasClients())
EncodeAndPlay(chunk, size);
if (!timer->IsStarted())
timer->Start();
timer->Add(size);
return size;
}
2017-01-25 10:03:17 +01:00
bool
HttpdOutput::Pause()
{
pause = true;
2017-01-25 10:03:17 +01:00
if (LockHasClients()) {
2013-01-15 18:22:17 +01:00
static const char silence[1020] = { 0 };
2017-01-25 10:03:17 +01:00
Play(silence, sizeof(silence));
}
return true;
}
void
HttpdOutput::SendTag(const Tag &tag)
{
if (encoder->ImplementsTag()) {
/* embed encoder tags */
/* flush the current stream, and end it */
try {
encoder->PreTag();
} catch (...) {
/* ignore */
}
BroadcastFromEncoder();
/* send the tag to the encoder - which starts a new
stream now */
try {
encoder->SendTag(tag);
encoder->Flush();
} catch (...) {
/* ignore */
}
/* the first page generated by the encoder will now be
used as the new "header" page, which is sent to all
new clients */
auto page = ReadPage();
2013-10-28 23:58:17 +01:00
if (page != nullptr) {
header = page;
BroadcastPage(page);
}
} else {
/* use Icy-Metadata */
static constexpr TagType types[] = {
TAG_ALBUM, TAG_ARTIST, TAG_TITLE,
TAG_NUM_OF_ITEM_TYPES
};
metadata = icy_server_metadata_page(tag, &types[0]);
2013-10-28 23:58:17 +01:00
if (metadata != nullptr) {
const std::lock_guard<Mutex> protect(mutex);
for (auto &client : clients)
client.PushMetaData(metadata);
}
}
}
inline void
HttpdOutput::CancelAllClients()
{
const std::lock_guard<Mutex> protect(mutex);
while (!pages.empty()) {
PagePtr page = std::move(pages.front());
pages.pop();
}
for (auto &client : clients)
client.CancelQueue();
cond.broadcast();
}
2017-01-25 10:03:17 +01:00
void
HttpdOutput::Cancel() noexcept
{
BlockingCall(GetEventLoop(), [this](){
2017-01-25 10:03:17 +01:00
CancelAllClients();
});
}
const struct AudioOutputPlugin httpd_output_plugin = {
2013-01-15 18:22:17 +01:00
"httpd",
nullptr,
&HttpdOutput::Create,
2013-01-15 18:22:17 +01:00
nullptr,
};