directory: simplify list update handling logic
Now the "update" command can be issued multiple times regardless of whether the client is in list mode or not. We serialize the update tasks to prevent updates from trampling over each other and will spawn another update task once the current one is finished updating and reaped. Right now we cap the queue size to 32 which is probably enough (I bet most people usually run update with no argument anyways); but we can make it grow/shrink dynamically if needed. There'll still be a hard-coded limit to prevent DoS attacks, though.
This commit is contained in:
parent
700f18eee5
commit
37a8239f44
@ -36,6 +36,7 @@
|
|||||||
#include "tag.h"
|
#include "tag.h"
|
||||||
#include "client.h"
|
#include "client.h"
|
||||||
#include "tag_print.h"
|
#include "tag_print.h"
|
||||||
|
#include "path.h"
|
||||||
#include "os_compat.h"
|
#include "os_compat.h"
|
||||||
|
|
||||||
#define COMMAND_PLAY "play"
|
#define COMMAND_PLAY "play"
|
||||||
@ -153,8 +154,6 @@ static const char check_non_negative[] = "\"%s\" is not an integer >= 0";
|
|||||||
static const char *current_command;
|
static const char *current_command;
|
||||||
static int command_listNum;
|
static int command_listNum;
|
||||||
|
|
||||||
static CommandEntry *getCommandEntryFromString(char *string, int permission);
|
|
||||||
|
|
||||||
static List *commandList;
|
static List *commandList;
|
||||||
|
|
||||||
static CommandEntry *newCommandEntry(void)
|
static CommandEntry *newCommandEntry(void)
|
||||||
@ -814,55 +813,31 @@ static int handlePlaylistMove(struct client *client,
|
|||||||
return print_playlist_result(client, result);
|
return print_playlist_result(client, result);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int listHandleUpdate(struct client *client,
|
static int print_update_result(struct client *client, int ret)
|
||||||
mpd_unused int argc,
|
|
||||||
char *argv[],
|
|
||||||
struct strnode *cmdnode, CommandEntry * cmd)
|
|
||||||
{
|
{
|
||||||
List *pathList = makeList(NULL, 1);
|
if (ret >= 0) {
|
||||||
CommandEntry *nextCmd = NULL;
|
client_printf(client, "updating_db: %i\n", ret);
|
||||||
struct strnode *next = cmdnode->next;
|
return 0;
|
||||||
|
|
||||||
if (argc == 2)
|
|
||||||
insertInList(pathList, argv[1], NULL);
|
|
||||||
else
|
|
||||||
insertInList(pathList, "", NULL);
|
|
||||||
|
|
||||||
if (next)
|
|
||||||
nextCmd = getCommandEntryFromString(next->data,
|
|
||||||
client_get_permission(client));
|
|
||||||
|
|
||||||
if (cmd != nextCmd) {
|
|
||||||
int ret = updateInit(pathList);
|
|
||||||
if (ret == -1)
|
|
||||||
command_error(client, ACK_ERROR_UPDATE_ALREADY,
|
|
||||||
"already updating");
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
}
|
||||||
|
if (ret == -2)
|
||||||
return 0;
|
command_error(client, ACK_ERROR_ARG, "invalid path");
|
||||||
|
else
|
||||||
|
command_error(client, ACK_ERROR_UPDATE_ALREADY,
|
||||||
|
"already updating");
|
||||||
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int handleUpdate(struct client *client,
|
static int handleUpdate(struct client *client,
|
||||||
mpd_unused int argc, char *argv[])
|
mpd_unused int argc, char *argv[])
|
||||||
{
|
{
|
||||||
int ret;
|
char *path = NULL;
|
||||||
|
|
||||||
if (argc == 2) {
|
assert(argc <= 2);
|
||||||
List *pathList = makeList(NULL, 1);
|
if (argc == 2 && !(path = sanitizePathDup(argv[1]))) {
|
||||||
insertInList(pathList, argv[1], NULL);
|
command_error(client, ACK_ERROR_ARG, "invalid path");
|
||||||
ret = updateInit(pathList);
|
return -1;
|
||||||
} else {
|
|
||||||
ret = updateInit(NULL);
|
|
||||||
}
|
}
|
||||||
|
return print_update_result(client, directory_update_init(path));
|
||||||
if (ret == -1)
|
|
||||||
command_error(client, ACK_ERROR_UPDATE_ALREADY,
|
|
||||||
"already updating");
|
|
||||||
|
|
||||||
client_printf(client, "updating_db: %i\n", ret);
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int handleNext(mpd_unused struct client *client,
|
static int handleNext(mpd_unused struct client *client,
|
||||||
@ -1307,7 +1282,7 @@ void initCommands(void)
|
|||||||
addCommand(COMMAND_PLAYLISTINFO, PERMISSION_READ, 0, 1, handlePlaylistInfo, NULL);
|
addCommand(COMMAND_PLAYLISTINFO, PERMISSION_READ, 0, 1, handlePlaylistInfo, NULL);
|
||||||
addCommand(COMMAND_FIND, PERMISSION_READ, 2, -1, handleFind, NULL);
|
addCommand(COMMAND_FIND, PERMISSION_READ, 2, -1, handleFind, NULL);
|
||||||
addCommand(COMMAND_SEARCH, PERMISSION_READ, 2, -1, handleSearch, NULL);
|
addCommand(COMMAND_SEARCH, PERMISSION_READ, 2, -1, handleSearch, NULL);
|
||||||
addCommand(COMMAND_UPDATE, PERMISSION_ADMIN, 0, 1, handleUpdate, listHandleUpdate);
|
addCommand(COMMAND_UPDATE, PERMISSION_ADMIN, 0, 1, handleUpdate, NULL);
|
||||||
addCommand(COMMAND_NEXT, PERMISSION_CONTROL, 0, 0, handleNext, NULL);
|
addCommand(COMMAND_NEXT, PERMISSION_CONTROL, 0, 0, handleNext, NULL);
|
||||||
addCommand(COMMAND_PREVIOUS, PERMISSION_CONTROL, 0, 0, handlePrevious, NULL);
|
addCommand(COMMAND_PREVIOUS, PERMISSION_CONTROL, 0, 0, handlePrevious, NULL);
|
||||||
addCommand(COMMAND_LISTALL, PERMISSION_READ, 0, 1, handleListAll, NULL);
|
addCommand(COMMAND_LISTALL, PERMISSION_READ, 0, 1, handleListAll, NULL);
|
||||||
@ -1420,24 +1395,6 @@ static CommandEntry *getCommandEntryAndCheckArgcAndPermission(struct client *cli
|
|||||||
return cmd;
|
return cmd;
|
||||||
}
|
}
|
||||||
|
|
||||||
static CommandEntry *getCommandEntryFromString(char *string, int permission)
|
|
||||||
{
|
|
||||||
CommandEntry *cmd = NULL;
|
|
||||||
char *argv[COMMAND_ARGV_MAX] = { NULL };
|
|
||||||
char *duplicated = xstrdup(string);
|
|
||||||
int argc = buffer2array(duplicated, argv, COMMAND_ARGV_MAX);
|
|
||||||
|
|
||||||
if (0 == argc)
|
|
||||||
goto out;
|
|
||||||
|
|
||||||
cmd = getCommandEntryAndCheckArgcAndPermission(0, permission,
|
|
||||||
argc, argv);
|
|
||||||
|
|
||||||
out:
|
|
||||||
free(duplicated);
|
|
||||||
return cmd;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int processCommandInternal(struct client *client,
|
static int processCommandInternal(struct client *client,
|
||||||
char *commandString, struct strnode *cmdnode)
|
char *commandString, struct strnode *cmdnode)
|
||||||
{
|
{
|
||||||
|
111
src/directory.c
111
src/directory.c
@ -53,13 +53,18 @@ enum update_progress {
|
|||||||
UPDATE_PROGRESS_DONE = 2
|
UPDATE_PROGRESS_DONE = 2
|
||||||
} progress;
|
} progress;
|
||||||
|
|
||||||
|
/* make this dynamic?, or maybe this is big enough... */
|
||||||
|
static char *update_paths[32];
|
||||||
|
static size_t update_paths_nr;
|
||||||
|
|
||||||
static Directory *music_root;
|
static Directory *music_root;
|
||||||
|
|
||||||
static time_t directory_dbModTime;
|
static time_t directory_dbModTime;
|
||||||
|
|
||||||
static pthread_t update_thr;
|
static pthread_t update_thr;
|
||||||
|
|
||||||
static int directory_updateJobId;
|
static const int update_task_id_max = 1 << 15;
|
||||||
|
static int update_task_id;
|
||||||
|
|
||||||
static enum update_return
|
static enum update_return
|
||||||
addToDirectory(Directory * directory, const char *name);
|
addToDirectory(Directory * directory, const char *name);
|
||||||
@ -99,74 +104,84 @@ static char *getDbFile(void)
|
|||||||
|
|
||||||
int isUpdatingDB(void)
|
int isUpdatingDB(void)
|
||||||
{
|
{
|
||||||
return (progress != UPDATE_PROGRESS_IDLE) ? directory_updateJobId : 0;
|
return (progress != UPDATE_PROGRESS_IDLE) ? update_task_id : 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void * update_task(void *_path)
|
||||||
|
{
|
||||||
|
enum update_return ret = UPDATE_RETURN_NOUPDATE;
|
||||||
|
|
||||||
|
if (_path) {
|
||||||
|
ret = updatePath((char *)_path);
|
||||||
|
free(_path);
|
||||||
|
} else {
|
||||||
|
ret = updateDirectory(music_root);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ret == UPDATE_RETURN_UPDATED && writeDirectoryDB() < 0)
|
||||||
|
ret = UPDATE_RETURN_ERROR;
|
||||||
|
progress = UPDATE_PROGRESS_DONE;
|
||||||
|
wakeup_main_task();
|
||||||
|
return (void *)ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void spawn_update_task(char *path)
|
||||||
|
{
|
||||||
|
pthread_attr_t attr;
|
||||||
|
|
||||||
|
assert(pthread_equal(pthread_self(), main_task));
|
||||||
|
|
||||||
|
progress = UPDATE_PROGRESS_RUNNING;
|
||||||
|
pthread_attr_init(&attr);
|
||||||
|
if (pthread_create(&update_thr, &attr, update_task, path))
|
||||||
|
FATAL("Failed to spawn update task: %s\n", strerror(errno));
|
||||||
|
if (++update_task_id > update_task_id_max)
|
||||||
|
update_task_id = 1;
|
||||||
|
DEBUG("spawned thread for update job id %i\n", update_task_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
void reap_update_task(void)
|
void reap_update_task(void)
|
||||||
{
|
{
|
||||||
enum update_return ret;
|
enum update_return ret;
|
||||||
|
|
||||||
|
assert(pthread_equal(pthread_self(), main_task));
|
||||||
|
|
||||||
if (progress != UPDATE_PROGRESS_DONE)
|
if (progress != UPDATE_PROGRESS_DONE)
|
||||||
return;
|
return;
|
||||||
if (pthread_join(update_thr, (void **)&ret))
|
if (pthread_join(update_thr, (void **)&ret))
|
||||||
FATAL("error joining update thread: %s\n", strerror(errno));
|
FATAL("error joining update thread: %s\n", strerror(errno));
|
||||||
if (ret == UPDATE_RETURN_UPDATED)
|
if (ret == UPDATE_RETURN_UPDATED)
|
||||||
playlistVersionChange();
|
playlistVersionChange();
|
||||||
progress = UPDATE_PROGRESS_IDLE;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void * update_task(void *arg)
|
if (update_paths_nr) {
|
||||||
{
|
char *path = update_paths[0];
|
||||||
List *path_list = (List *)arg;
|
memmove(&update_paths[0], &update_paths[1],
|
||||||
enum update_return ret = UPDATE_RETURN_NOUPDATE;
|
--update_paths_nr * sizeof(char *));
|
||||||
|
spawn_update_task(path);
|
||||||
if (path_list) {
|
|
||||||
ListNode *node = path_list->firstNode;
|
|
||||||
|
|
||||||
while (node) {
|
|
||||||
switch (updatePath(node->key)) {
|
|
||||||
case UPDATE_RETURN_ERROR:
|
|
||||||
ret = UPDATE_RETURN_ERROR;
|
|
||||||
goto out;
|
|
||||||
case UPDATE_RETURN_NOUPDATE:
|
|
||||||
break;
|
|
||||||
case UPDATE_RETURN_UPDATED:
|
|
||||||
ret = UPDATE_RETURN_UPDATED;
|
|
||||||
}
|
|
||||||
node = node->nextNode;
|
|
||||||
}
|
|
||||||
freeList(path_list);
|
|
||||||
} else {
|
} else {
|
||||||
ret = updateDirectory(music_root);
|
progress = UPDATE_PROGRESS_IDLE;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ret == UPDATE_RETURN_UPDATED && writeDirectoryDB() < 0)
|
|
||||||
ret = UPDATE_RETURN_ERROR;
|
|
||||||
out:
|
|
||||||
progress = UPDATE_PROGRESS_DONE;
|
|
||||||
wakeup_main_task();
|
|
||||||
return (void *)ret;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int updateInit(List * path_list)
|
int directory_update_init(char *path)
|
||||||
{
|
{
|
||||||
pthread_attr_t attr;
|
assert(pthread_equal(pthread_self(), main_task));
|
||||||
|
|
||||||
if (progress != UPDATE_PROGRESS_IDLE)
|
if (progress != UPDATE_PROGRESS_IDLE) {
|
||||||
return -1;
|
int next_task_id;
|
||||||
|
|
||||||
progress = UPDATE_PROGRESS_RUNNING;
|
if (!path)
|
||||||
|
return -1;
|
||||||
|
if (update_paths_nr == ARRAY_SIZE(update_paths))
|
||||||
|
return -1;
|
||||||
|
assert(update_paths_nr < ARRAY_SIZE(update_paths));
|
||||||
|
update_paths[update_paths_nr++] = path;
|
||||||
|
next_task_id = update_task_id + update_paths_nr;
|
||||||
|
|
||||||
pthread_attr_init(&attr);
|
return next_task_id > update_task_id_max ? 1 : next_task_id;
|
||||||
if (pthread_create(&update_thr, &attr, update_task, path_list))
|
}
|
||||||
FATAL("Failed to spawn update task: %s\n", strerror(errno));
|
spawn_update_task(path);
|
||||||
directory_updateJobId++;
|
return update_task_id;
|
||||||
if (directory_updateJobId > 1 << 15)
|
|
||||||
directory_updateJobId = 1;
|
|
||||||
DEBUG("updateInit: spawned update thread for update job id %i\n",
|
|
||||||
(int)directory_updateJobId);
|
|
||||||
|
|
||||||
return (int)directory_updateJobId;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void directory_set_stat(Directory * dir, const struct stat *st)
|
static void directory_set_stat(Directory * dir, const struct stat *st)
|
||||||
|
@ -42,12 +42,12 @@ void reap_update_task(void);
|
|||||||
|
|
||||||
int isUpdatingDB(void);
|
int isUpdatingDB(void);
|
||||||
|
|
||||||
/**
|
/*
|
||||||
* Starts the tag cache update in the specified location(s). Returns
|
* returns the non-negative update job ID on success,
|
||||||
* the job id on success, -1 on error or 0 if an update is already
|
* returns -1 if busy
|
||||||
* running.
|
* @path will be freed by this function and should not be reused
|
||||||
*/
|
*/
|
||||||
int updateInit(List * pathList);
|
int directory_update_init(char *path);
|
||||||
|
|
||||||
void directory_init(void);
|
void directory_init(void);
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user