forked from external/yambar
666 lines
18 KiB
C
666 lines
18 KiB
C
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <inttypes.h>
|
|
#include <time.h>
|
|
#include <threads.h>
|
|
#include <unistd.h>
|
|
#include <assert.h>
|
|
#include <errno.h>
|
|
|
|
#include <poll.h>
|
|
#include <libgen.h>
|
|
|
|
#include <sys/stat.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/un.h>
|
|
#include <sys/eventfd.h>
|
|
#include <sys/inotify.h>
|
|
|
|
#include <mpd/client.h>
|
|
|
|
#define LOG_MODULE "mpd"
|
|
#define LOG_ENABLE_DBG 0
|
|
#include "../log.h"
|
|
#include "../bar/bar.h"
|
|
#include "../config.h"
|
|
#include "../config-verify.h"
|
|
#include "../plugin.h"
|
|
|
|
struct private {
|
|
char *host;
|
|
uint16_t port;
|
|
struct particle *label;
|
|
|
|
struct mpd_connection *conn;
|
|
|
|
enum mpd_state state;
|
|
bool repeat;
|
|
bool random;
|
|
bool consume;
|
|
int volume;
|
|
char *album;
|
|
char *artist;
|
|
char *title;
|
|
char *file;
|
|
|
|
struct {
|
|
uint64_t value;
|
|
struct timespec when;
|
|
} elapsed;
|
|
uint64_t duration;
|
|
|
|
thrd_t refresh_thread_id;
|
|
int refresh_abort_fd;
|
|
};
|
|
|
|
static void
|
|
destroy(struct module *mod)
|
|
{
|
|
struct private *m = mod->private;
|
|
if (m->refresh_thread_id != 0) {
|
|
assert(m->refresh_abort_fd != -1);
|
|
if (write(m->refresh_abort_fd, &(uint64_t){1}, sizeof(uint64_t))
|
|
!= sizeof(uint64_t))
|
|
{
|
|
LOG_ERRNO("failed to signal abort to refresher thread");
|
|
} else{
|
|
int res;
|
|
thrd_join(m->refresh_thread_id, &res);
|
|
}
|
|
|
|
close(m->refresh_abort_fd);
|
|
};
|
|
|
|
free(m->host);
|
|
free(m->album);
|
|
free(m->artist);
|
|
free(m->title);
|
|
free(m->file);
|
|
assert(m->conn == NULL);
|
|
|
|
m->label->destroy(m->label);
|
|
|
|
free(m);
|
|
module_default_destroy(mod);
|
|
}
|
|
|
|
static const char *
|
|
description(const struct module *mod)
|
|
{
|
|
return "mpd";
|
|
}
|
|
|
|
static uint64_t
|
|
timespec_diff_milli_seconds(const struct timespec *a, const struct timespec *b)
|
|
{
|
|
/* TODO */
|
|
uint64_t nsecs_a = a->tv_sec * 1000000000 + a->tv_nsec;
|
|
uint64_t nsecs_b = b->tv_sec * 1000000000 + b->tv_nsec;
|
|
|
|
assert(nsecs_a >= nsecs_b);
|
|
uint64_t nsec_diff = nsecs_a - nsecs_b;
|
|
return nsec_diff / 1000000;
|
|
}
|
|
|
|
static void
|
|
secs_to_str(unsigned secs, char *s, size_t sz)
|
|
{
|
|
unsigned hours = secs / (60 * 60);
|
|
unsigned minutes = secs % (60 * 60) / 60;
|
|
secs %= 60;
|
|
|
|
if (hours > 0)
|
|
snprintf(s, sz, "%02u:%02u:%02u", hours, minutes, secs);
|
|
else
|
|
snprintf(s, sz, "%02u:%02u", minutes, secs);
|
|
}
|
|
|
|
static struct exposable *
|
|
content(struct module *mod)
|
|
{
|
|
const struct private *m = mod->private;
|
|
|
|
struct timespec now;
|
|
clock_gettime(CLOCK_MONOTONIC, &now);
|
|
|
|
mtx_lock(&mod->lock);
|
|
|
|
/* Calculate what 'elapsed' is now */
|
|
uint64_t elapsed = m->elapsed.value;
|
|
|
|
if (m->state == MPD_STATE_PLAY) {
|
|
elapsed += timespec_diff_milli_seconds(&now, &m->elapsed.when);
|
|
if (elapsed > m->duration) {
|
|
LOG_DBG(
|
|
"dynamic update of elapsed overflowed: "
|
|
"elapsed=%"PRIu64", duration=%"PRIu64, elapsed, m->duration);
|
|
elapsed = m->duration;
|
|
}
|
|
|
|
}
|
|
|
|
unsigned elapsed_secs = elapsed / 1000;
|
|
unsigned duration_secs = m->duration / 1000;
|
|
|
|
/* elapsed/duration as strings (e.g. 03:37) */
|
|
char pos[16], end[16];
|
|
secs_to_str(elapsed_secs, pos, sizeof(pos));
|
|
secs_to_str(duration_secs, end, sizeof(end));
|
|
|
|
/* State as string */
|
|
const char *state_str = NULL;
|
|
if (m->conn == NULL)
|
|
state_str = "offline";
|
|
else {
|
|
switch (m->state) {
|
|
case MPD_STATE_UNKNOWN: state_str = "unknown"; break;
|
|
case MPD_STATE_STOP: state_str = "stopped"; break;
|
|
case MPD_STATE_PAUSE: state_str = "paused"; break;
|
|
case MPD_STATE_PLAY: state_str = "playing"; break;
|
|
}
|
|
}
|
|
|
|
/* Tell particle to real-time track? */
|
|
enum tag_realtime_unit realtime = m->state == MPD_STATE_PLAY
|
|
? TAG_REALTIME_MSECS : TAG_REALTIME_NONE;
|
|
|
|
struct tag_set tags = {
|
|
.tags = (struct tag *[]){
|
|
tag_new_string(mod, "state", state_str),
|
|
tag_new_bool(mod, "repeat", m->repeat),
|
|
tag_new_bool(mod, "random", m->random),
|
|
tag_new_bool(mod, "consume", m->consume),
|
|
tag_new_int_range(mod, "volume", m->volume, 0, 100),
|
|
tag_new_string(mod, "album", m->album),
|
|
tag_new_string(mod, "artist", m->artist),
|
|
tag_new_string(mod, "title", m->title),
|
|
tag_new_string(mod, "file", m->file),
|
|
tag_new_string(mod, "pos", pos),
|
|
tag_new_string(mod, "end", end),
|
|
tag_new_int(mod, "duration", m->duration),
|
|
tag_new_int_realtime(
|
|
mod, "elapsed", elapsed, 0, m->duration, realtime),
|
|
},
|
|
.count = 13,
|
|
};
|
|
|
|
mtx_unlock(&mod->lock);
|
|
|
|
struct exposable *exposable = m->label->instantiate(m->label, &tags);
|
|
|
|
tag_set_destroy(&tags);
|
|
return exposable;
|
|
}
|
|
|
|
/* Returns true if aborted, false otherwise (regardless of whether
|
|
* socket exists or not) */
|
|
static bool
|
|
wait_for_socket_create(const struct module *mod)
|
|
{
|
|
const struct private *m = mod->private;
|
|
assert(m->port == 0);
|
|
|
|
char *copy = strdup(m->host);
|
|
const char *base = basename(copy);
|
|
const char *directory = dirname(copy);
|
|
|
|
LOG_DBG("monitoring %s for %s to be created", directory, base);
|
|
|
|
int fd = inotify_init();
|
|
if (fd == -1) {
|
|
free(copy);
|
|
return false;
|
|
}
|
|
|
|
int wd = inotify_add_watch(fd, directory, IN_CREATE);
|
|
if (wd == -1) {
|
|
close(fd);
|
|
free(copy);
|
|
return false;
|
|
}
|
|
|
|
bool have_mpd_socket = false;
|
|
|
|
/* Check if socket already exists *and* is connectable */
|
|
struct stat st;
|
|
if (stat(m->host, &st) == 0 && S_ISSOCK(st.st_mode)) {
|
|
|
|
int s = socket(AF_UNIX, SOCK_STREAM | SOCK_CLOEXEC, 0);
|
|
|
|
struct sockaddr_un addr = {.sun_family = AF_UNIX};
|
|
strncpy(addr.sun_path, m->host, sizeof(addr.sun_path) - 1);
|
|
|
|
int r = connect(s, (const struct sockaddr *)&addr, sizeof(addr));
|
|
|
|
if (r == 0) {
|
|
LOG_DBG("%s: already exists, and is connectable", m->host);
|
|
have_mpd_socket = true;
|
|
} else {
|
|
LOG_DBG("%s: already exists, but isn't connectable: %s",
|
|
m->host, strerror(errno));
|
|
}
|
|
|
|
close(s);
|
|
}
|
|
|
|
if (!have_mpd_socket)
|
|
LOG_WARN("MPD doesn't appear to be running");
|
|
|
|
bool ret = false;
|
|
while (!have_mpd_socket) {
|
|
struct pollfd fds[] = {
|
|
{.fd = mod->abort_fd, .events = POLLIN},
|
|
{.fd = fd, .events = POLLIN}
|
|
};
|
|
|
|
if (poll(fds, sizeof(fds) / sizeof(fds[0]), -1) < 0) {
|
|
if (errno == EINTR)
|
|
continue;
|
|
|
|
LOG_ERRNO("failed to poll");
|
|
break;
|
|
}
|
|
|
|
if (fds[0].revents & POLLIN) {
|
|
ret = true;
|
|
break;
|
|
}
|
|
|
|
assert(fds[1].revents & POLLIN);
|
|
|
|
char buf[1024];
|
|
ssize_t len = read(fd, buf, sizeof(buf));
|
|
|
|
for (const char *ptr = buf; ptr < buf + len; ) {
|
|
const struct inotify_event *e = (const struct inotify_event *)ptr;
|
|
LOG_DBG("inotify: CREATED: %s/%.*s", directory, e->len, e->name);
|
|
|
|
if (strncmp(base, e->name, e->len) == 0) {
|
|
LOG_DBG("MPD socket created");
|
|
have_mpd_socket = true;
|
|
break;
|
|
}
|
|
|
|
ptr += sizeof(*e) + e->len;
|
|
}
|
|
}
|
|
|
|
inotify_rm_watch(fd, wd);
|
|
close(fd);
|
|
free(copy);
|
|
return ret;
|
|
}
|
|
|
|
static struct mpd_connection *
|
|
connect_to_mpd(const struct module *mod)
|
|
{
|
|
const struct private *m = mod->private;
|
|
|
|
struct mpd_connection *conn = mpd_connection_new(m->host, m->port, 0);
|
|
if (conn == NULL) {
|
|
LOG_ERR("failed to create MPD connection");
|
|
return NULL;
|
|
}
|
|
|
|
enum mpd_error merr = mpd_connection_get_error(conn);
|
|
if (merr != MPD_ERROR_SUCCESS) {
|
|
LOG_WARN("failed to connect to MPD: %s",
|
|
mpd_connection_get_error_message(conn));
|
|
mpd_connection_free(conn);
|
|
return NULL;
|
|
}
|
|
|
|
const unsigned *version = mpd_connection_get_server_version(conn);
|
|
LOG_INFO("MPD %u.%u.%u", version[0], version[1], version[2]);
|
|
|
|
return conn;
|
|
}
|
|
|
|
static bool
|
|
update_status(struct module *mod)
|
|
{
|
|
struct private *m = mod->private;
|
|
|
|
struct mpd_status *status = mpd_run_status(m->conn);
|
|
if (status == NULL) {
|
|
LOG_ERR("failed to get status: %s",
|
|
mpd_connection_get_error_message(m->conn));
|
|
return false;
|
|
}
|
|
|
|
struct timespec now;
|
|
clock_gettime(CLOCK_MONOTONIC, &now);
|
|
|
|
mtx_lock(&mod->lock);
|
|
m->state = mpd_status_get_state(status);
|
|
m->repeat = mpd_status_get_repeat(status);
|
|
m->random = mpd_status_get_random(status);
|
|
m->consume = mpd_status_get_consume(status);
|
|
m->volume = mpd_status_get_volume(status);
|
|
m->duration = mpd_status_get_total_time(status) * 1000;
|
|
m->elapsed.value = mpd_status_get_elapsed_ms(status);
|
|
m->elapsed.when = now;
|
|
mtx_unlock(&mod->lock);
|
|
|
|
mpd_status_free(status);
|
|
|
|
struct mpd_song *song = mpd_run_current_song(m->conn);
|
|
if (song == NULL && mpd_connection_get_error(m->conn) != MPD_ERROR_SUCCESS) {
|
|
LOG_ERR("failed to get current song: %s",
|
|
mpd_connection_get_error_message(m->conn));
|
|
return false;
|
|
}
|
|
|
|
if (song == NULL) {
|
|
mtx_lock(&mod->lock);
|
|
free(m->album); m->album = NULL;
|
|
free(m->artist); m->artist = NULL;
|
|
free(m->title); m->title = NULL;
|
|
free(m->file); m->file = NULL;
|
|
mtx_unlock(&mod->lock);
|
|
} else {
|
|
const char *album = mpd_song_get_tag(song, MPD_TAG_ALBUM, 0);
|
|
const char *artist = mpd_song_get_tag(song, MPD_TAG_ARTIST, 0);
|
|
const char *title = mpd_song_get_tag(song, MPD_TAG_TITLE, 0);
|
|
const char *file = mpd_song_get_uri(song);
|
|
|
|
mtx_lock(&mod->lock);
|
|
free(m->album);
|
|
free(m->artist);
|
|
free(m->title);
|
|
free(m->file);
|
|
|
|
m->album = album != NULL ? strdup(album) : NULL;
|
|
m->artist = artist != NULL ? strdup(artist) : NULL;
|
|
m->title = title != NULL ? strdup(title) : NULL;
|
|
m->file = file != NULL ? strdup(file) : NULL;
|
|
mtx_unlock(&mod->lock);
|
|
|
|
mpd_song_free(song);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static int
|
|
run(struct module *mod)
|
|
{
|
|
const struct bar *bar = mod->bar;
|
|
struct private *m = mod->private;
|
|
|
|
bool aborted = false;
|
|
int ret = 0;
|
|
|
|
while (!aborted && ret == 0) {
|
|
|
|
if (m->conn != NULL) {
|
|
mpd_connection_free(m->conn);
|
|
m->conn = NULL;
|
|
}
|
|
|
|
/* Reset state */
|
|
mtx_lock(&mod->lock);
|
|
free(m->album); m->album = NULL;
|
|
free(m->artist); m->artist = NULL;
|
|
free(m->title); m->title = NULL;
|
|
free(m->file); m->file = NULL;
|
|
m->state = MPD_STATE_UNKNOWN;
|
|
m->elapsed.value = m->duration = 0;
|
|
m->elapsed.when.tv_sec = m->elapsed.when.tv_nsec = 0;
|
|
mtx_unlock(&mod->lock);
|
|
|
|
/* Keep trying to connect, until we succeed */
|
|
while (!aborted && ret == 0) {
|
|
if (m->port == 0) {
|
|
/* Use inotify to watch for socket creation */
|
|
aborted = wait_for_socket_create(mod);
|
|
if (aborted)
|
|
break;
|
|
}
|
|
|
|
m->conn = connect_to_mpd(mod);
|
|
if (m->conn != NULL)
|
|
break;
|
|
|
|
/*
|
|
* In case we can't use inotify to watch for socket
|
|
* creation (for example, we're connecting to a remote
|
|
* host), wait for a while until we try to re-connect
|
|
* again.
|
|
*/
|
|
while (!aborted) {
|
|
struct pollfd fds[] = {{.fd = mod->abort_fd, .events = POLLIN}};
|
|
int res = poll(fds, sizeof(fds) / sizeof(fds[0]), 10 * 1000);
|
|
|
|
if (res < 0) {
|
|
if (errno == EINTR)
|
|
continue;
|
|
|
|
LOG_ERRNO("failed to poll");
|
|
ret = 1;
|
|
break;
|
|
}
|
|
|
|
if (res == 1) {
|
|
assert(fds[0].revents & POLLIN);
|
|
aborted = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (aborted || ret != 0)
|
|
break;
|
|
|
|
/* Initial state (after establishing a connection) */
|
|
assert(m->conn != NULL);
|
|
if (!update_status(mod))
|
|
continue;
|
|
|
|
bar->refresh(bar);
|
|
|
|
/* Monitor for events from MPD */
|
|
while (true) {
|
|
struct pollfd fds[] = {
|
|
{.fd = mod->abort_fd, .events = POLLIN},
|
|
{.fd = mpd_connection_get_fd(m->conn), .events = POLLIN},
|
|
};
|
|
|
|
if (!mpd_send_idle(m->conn)) {
|
|
LOG_ERR("failed to send IDLE command: %s",
|
|
mpd_connection_get_error_message(m->conn));
|
|
break;
|
|
}
|
|
|
|
if (poll(fds, sizeof(fds) / sizeof(fds[0]), -1) < 0) {
|
|
if (errno == EINTR)
|
|
continue;
|
|
|
|
LOG_ERRNO("failed to poll");
|
|
ret = 1;
|
|
break;
|
|
}
|
|
|
|
if (fds[0].revents & POLLIN) {
|
|
aborted = true;
|
|
break;
|
|
}
|
|
|
|
if (fds[1].revents & POLLHUP) {
|
|
LOG_WARN("disconnected from MPD daemon");
|
|
break;
|
|
}
|
|
|
|
if (fds[1].revents & POLLIN) {
|
|
enum mpd_idle idle __attribute__ ((unused)) =
|
|
mpd_recv_idle(m->conn, true);
|
|
|
|
LOG_DBG("IDLE mask: %d", idle);
|
|
|
|
if (!update_status(mod))
|
|
break;
|
|
|
|
bar->refresh(bar);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (m->conn != NULL) {
|
|
mpd_connection_free(m->conn);
|
|
m->conn = NULL;
|
|
}
|
|
|
|
return aborted ? 0 : ret;
|
|
}
|
|
|
|
struct refresh_context {
|
|
struct module *mod;
|
|
int abort_fd;
|
|
long milli_seconds;
|
|
};
|
|
|
|
static int
|
|
refresh_in_thread(void *arg)
|
|
{
|
|
struct refresh_context *ctx = arg;
|
|
struct module *mod = ctx->mod;
|
|
|
|
/* Extract data from context so that we can free it */
|
|
int abort_fd = ctx->abort_fd;
|
|
long milli_seconds = ctx->milli_seconds;
|
|
free(ctx);
|
|
|
|
LOG_DBG("going to sleep for %ldms", milli_seconds);
|
|
|
|
/* Wait for timeout, or abort signal */
|
|
struct pollfd fds[] = {{.fd = abort_fd, .events = POLLIN}};
|
|
int r = poll(fds, 1, milli_seconds);
|
|
|
|
if (r < 0) {
|
|
LOG_ERRNO("failed to poll() in refresh thread");
|
|
return 1;
|
|
}
|
|
|
|
/* Aborted? */
|
|
if (r == 1) {
|
|
assert(fds[0].revents & POLLIN);
|
|
LOG_DBG("refresh thread aborted");
|
|
return 0;
|
|
}
|
|
|
|
LOG_DBG("timed refresh");
|
|
mod->bar->refresh(mod->bar);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static bool
|
|
refresh_in(struct module *mod, long milli_seconds)
|
|
{
|
|
struct private *m = mod->private;
|
|
|
|
/* Abort currently running refresh thread */
|
|
if (m->refresh_thread_id != 0) {
|
|
LOG_DBG("aborting current refresh thread");
|
|
|
|
/* Signal abort to thread */
|
|
assert(m->refresh_abort_fd != -1);
|
|
if (write(m->refresh_abort_fd, &(uint64_t){1}, sizeof(uint64_t))
|
|
!= sizeof(uint64_t))
|
|
{
|
|
LOG_ERRNO("failed to signal abort to refresher thread");
|
|
return false;
|
|
}
|
|
|
|
/* Wait for it to finish */
|
|
int res;
|
|
thrd_join(m->refresh_thread_id, &res);
|
|
|
|
/* Close and cleanup */
|
|
close(m->refresh_abort_fd);
|
|
m->refresh_abort_fd = -1;
|
|
m->refresh_thread_id = 0;
|
|
}
|
|
|
|
/* Create a new eventfd, to be able to signal abort to the thread */
|
|
int abort_fd = eventfd(0, EFD_CLOEXEC);
|
|
if (abort_fd == -1) {
|
|
LOG_ERRNO("failed to create eventfd");
|
|
return false;
|
|
}
|
|
|
|
/* Thread context */
|
|
struct refresh_context *ctx = malloc(sizeof(*ctx));
|
|
ctx->mod = mod;
|
|
ctx->abort_fd = m->refresh_abort_fd = abort_fd;
|
|
ctx->milli_seconds = milli_seconds;
|
|
|
|
/* Create thread */
|
|
int r = thrd_create(&m->refresh_thread_id, &refresh_in_thread, ctx);
|
|
|
|
if (r != thrd_success) {
|
|
LOG_ERR("failed to create refresh thread");
|
|
close(m->refresh_abort_fd);
|
|
m->refresh_abort_fd = -1;
|
|
m->refresh_thread_id = 0;
|
|
free(ctx);
|
|
}
|
|
|
|
/* Detach - we don't want to have to thrd_join() it */
|
|
//thrd_detach(tid);
|
|
return r == 0;
|
|
}
|
|
|
|
static struct module *
|
|
mpd_new(const char *host, uint16_t port, struct particle *label)
|
|
{
|
|
struct private *priv = calloc(1, sizeof(*priv));
|
|
priv->host = strdup(host);
|
|
priv->port = port;
|
|
priv->label = label;
|
|
priv->state = MPD_STATE_UNKNOWN;
|
|
priv->refresh_abort_fd = -1;
|
|
|
|
struct module *mod = module_common_new();
|
|
mod->private = priv;
|
|
mod->run = &run;
|
|
mod->destroy = &destroy;
|
|
mod->content = &content;
|
|
mod->refresh_in = &refresh_in;
|
|
mod->description = &description;
|
|
return mod;
|
|
}
|
|
|
|
static struct module *
|
|
from_conf(const struct yml_node *node, struct conf_inherit inherited)
|
|
{
|
|
const struct yml_node *host = yml_get_value(node, "host");
|
|
const struct yml_node *port = yml_get_value(node, "port");
|
|
const struct yml_node *c = yml_get_value(node, "content");
|
|
|
|
return mpd_new(
|
|
yml_value_as_string(host),
|
|
port != NULL ? yml_value_as_int(port) : 0,
|
|
conf_to_particle(c, inherited));
|
|
}
|
|
|
|
static bool
|
|
verify_conf(keychain_t *chain, const struct yml_node *node)
|
|
{
|
|
static const struct attr_info attrs[] = {
|
|
{"host", true, &conf_verify_string},
|
|
{"port", false, &conf_verify_unsigned},
|
|
MODULE_COMMON_ATTRS,
|
|
};
|
|
|
|
return conf_verify_dict(chain, node, attrs);
|
|
}
|
|
|
|
const struct module_iface module_mpd_iface = {
|
|
.verify_conf = &verify_conf,
|
|
.from_conf = &from_conf,
|
|
};
|
|
|
|
#if defined(CORE_PLUGINS_AS_SHARED_LIBRARIES)
|
|
extern const struct module_iface iface __attribute__((weak, alias("module_mpd_iface")));
|
|
#endif
|