mirror of
https://github.com/SuperBFG7/ympd
synced 2024-11-14 10:04:48 +00:00
213 lines
7.0 KiB
C
213 lines
7.0 KiB
C
/* myMPD
|
|
(c) 2018 Juergen Mang <mail@jcgames.de>
|
|
This project's homepage is: https://github.com/jcorporation/ympd
|
|
|
|
myMPD ist fork of:
|
|
|
|
ympd
|
|
(c) 2013-2014 Andrew Karpow <andy@ndyk.de>
|
|
This project's homepage is: http://www.ympd.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; version 2 of the License.
|
|
|
|
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.,
|
|
Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*/
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
#include <unistd.h>
|
|
#include <getopt.h>
|
|
#include <sys/time.h>
|
|
#include <pthread.h>
|
|
|
|
#include "mongoose/mongoose.h"
|
|
//#include "http_server.h"
|
|
#include "mpd_client.h"
|
|
#include "config.h"
|
|
|
|
extern char *optarg;
|
|
|
|
int force_exit = 0;
|
|
|
|
void bye()
|
|
{
|
|
force_exit = 1;
|
|
}
|
|
|
|
|
|
static int is_websocket(const struct mg_connection *nc) {
|
|
return nc->flags & MG_F_IS_WEBSOCKET;
|
|
}
|
|
|
|
static void server_callback(struct mg_connection *c, int ev, void *p) {
|
|
// int result = MG_EV_FALSE;
|
|
FILE *fp = NULL;
|
|
|
|
switch(ev) {
|
|
case MG_EV_CLOSE:
|
|
mpd_close_handler(c);
|
|
break;
|
|
// return MG_TRUE;
|
|
case MG_EV_HTTP_REQUEST:
|
|
if (is_websocket(c)) {
|
|
c->content[c->content_len] = '\0';
|
|
if(c->content_len)
|
|
callback_mpd(c);
|
|
break;
|
|
} else
|
|
break;
|
|
case MG_EV_AUTH:
|
|
// no auth for websockets since mobile safari does not support it
|
|
if ( (mpd.gpass == NULL) || (c->is_websocket) || ((mpd.local_port > 0) && (c->local_port == mpd.local_port)) )
|
|
break;
|
|
else {
|
|
if ( (fp = fopen(mpd.gpass, "r")) != NULL ) {
|
|
mg_authorize_digest(c, fp);
|
|
fclose(fp);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
int main(int argc, char **argv)
|
|
{
|
|
int n, option_index = 0;
|
|
struct mg_server *server = mg_create_server(NULL, server_callback);
|
|
unsigned int current_timer = 0, last_timer = 0;
|
|
char *run_as_user = NULL;
|
|
char const *error_msg = NULL;
|
|
char *webport = "8080";
|
|
|
|
atexit(bye);
|
|
mg_set_option(server, "document_root", SRC_PATH);
|
|
|
|
mg_set_option(server, "auth_domain", "mympd");
|
|
mpd.port = 6600;
|
|
mpd.local_port = 0;
|
|
mpd.gpass = NULL;
|
|
strcpy(mpd.host, "127.0.0.1");
|
|
streamport = 8000;
|
|
strcpy(coverimage, "folder.jpg");
|
|
|
|
static struct option long_options[] = {
|
|
{"digest", required_argument, 0, 'D'},
|
|
{"host", required_argument, 0, 'h'},
|
|
{"port", required_argument, 0, 'p'},
|
|
{"localport", required_argument, 0, 'l'},
|
|
{"webport", required_argument, 0, 'w'},
|
|
{"user", required_argument, 0, 'u'},
|
|
{"version", no_argument, 0, 'v'},
|
|
{"help", no_argument, 0, 0 },
|
|
{"mpdpass", required_argument, 0, 'm'},
|
|
{"streamport", required_argument, 0, 's'},
|
|
{"coverimage", required_argument, 0, 'i'},
|
|
{0, 0, 0, 0 }
|
|
};
|
|
|
|
while((n = getopt_long(argc, argv, "D:h:p:l:w:u:d:vm:s:i:",
|
|
long_options, &option_index)) != -1) {
|
|
switch (n) {
|
|
case 'D':
|
|
mpd.gpass = strdup(optarg);
|
|
break;
|
|
case 'h':
|
|
strncpy(mpd.host, optarg, sizeof(mpd.host));
|
|
break;
|
|
case 'p':
|
|
mpd.port = atoi(optarg);
|
|
break;
|
|
case 'l':
|
|
mpd.local_port = atoi(optarg);
|
|
break;
|
|
case 'w':
|
|
webport = strdup(optarg);
|
|
break;
|
|
case 'u':
|
|
run_as_user = strdup(optarg);
|
|
break;
|
|
case 'm':
|
|
if (strlen(optarg) > 0)
|
|
mpd.password = strdup(optarg);
|
|
break;
|
|
case 's':
|
|
streamport = atoi(optarg);
|
|
break;
|
|
case 'i':
|
|
strncpy(coverimage, optarg, sizeof(coverimage));
|
|
break;
|
|
case 'v':
|
|
fprintf(stdout, "myMPD %d.%d.%d\n"
|
|
"Copyright (C) 2018 Juergen Mang <mail@jcgames.de>\n"
|
|
"Built " __DATE__ " "__TIME__ "\n",
|
|
MYMPD_VERSION_MAJOR, MYMPD_VERSION_MINOR, MYMPD_VERSION_PATCH);
|
|
return EXIT_SUCCESS;
|
|
break;
|
|
default:
|
|
fprintf(stderr, "Usage: %s [OPTION]...\n\n"
|
|
" -D, --digest <htdigest>\tpath to htdigest file for authorization\n"
|
|
" \t(realm ympd) [no authorization]\n"
|
|
" -h, --host <host>\t\tconnect to mpd at host [localhost]\n"
|
|
" -p, --port <port>\t\tconnect to mpd at port [6600]\n"
|
|
" -l, --localport <port>\t\tskip authorization for local port\n"
|
|
" -w, --webport [ip:]<port>\tlisten interface/port for webserver [8080]\n"
|
|
" -u, --user <username>\t\tdrop priviliges to user after socket bind\n"
|
|
" -v, --version\t\t\tget version\n"
|
|
" -m, --mpdpass <password>\tspecifies the password to use when connecting to mpd\n"
|
|
" -s, --streamport <port>\tconnect to mpd http stream at port [8000]\n"
|
|
" -i, --coverimage <filename>\tfilename for coverimage [folder.jpg]\n"
|
|
" --help\t\t\t\tthis help\n"
|
|
, argv[0]);
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
if(error_msg)
|
|
{
|
|
fprintf(stderr, "Mongoose error: %s\n", error_msg);
|
|
return EXIT_FAILURE;
|
|
}
|
|
}
|
|
|
|
error_msg = mg_set_option(server, "listening_port", webport);
|
|
if(error_msg) {
|
|
fprintf(stderr, "Mongoose error: %s\n", error_msg);
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
/* drop privilges at last to ensure proper port binding */
|
|
if(run_as_user != NULL) {
|
|
error_msg = mg_set_option(server, "run_as_user", run_as_user);
|
|
free(run_as_user);
|
|
if(error_msg)
|
|
{
|
|
fprintf(stderr, "Mongoose error: %s\n", error_msg);
|
|
return EXIT_FAILURE;
|
|
}
|
|
}
|
|
|
|
while (!force_exit) {
|
|
mg_poll_server(server, 200);
|
|
current_timer = time(NULL);
|
|
if(current_timer - last_timer)
|
|
{
|
|
last_timer = current_timer;
|
|
mpd_poll(server);
|
|
}
|
|
}
|
|
|
|
mpd_disconnect();
|
|
mg_destroy_server(&server);
|
|
|
|
return EXIT_SUCCESS;
|
|
}
|