mirror of
https://github.com/owntone/owntone-server.git
synced 2025-01-14 08:15:02 -05:00
9492a64846
pending further verification against iTunes. --enable-browse and --enable-query must be specified to configure to enable both options. browse support requires query support. 2. Database iteration is now sorted and the database is not kept locked as long during iteration.
701 lines
13 KiB
C
701 lines
13 KiB
C
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
|
|
#define DEBUG
|
|
|
|
#include "err.h"
|
|
#include "query.h"
|
|
|
|
static const query_field_t* find_field(const char* name,
|
|
const query_field_t* fields);
|
|
static int arith_query(query_node_t* query, void* target);
|
|
static int string_query(query_node_t* query, void* target);
|
|
|
|
static query_node_t* match_specifier(const char* query,
|
|
const char** cursor,
|
|
const query_field_t* fields);
|
|
static query_node_t* group_match(const char* query,
|
|
const char** cursor,
|
|
const query_field_t* fields);
|
|
static query_node_t* single_match(const char* query,
|
|
const char** cursor,
|
|
const query_field_t* fields);
|
|
static int get_field_name(const char** pcursor,
|
|
const char* query,
|
|
char* name,
|
|
int len);
|
|
static int get_opcode(const char** pcursor,
|
|
const char* query,
|
|
char* name,
|
|
int len);
|
|
static query_node_t* match_number(const query_field_t* field,
|
|
char not, char opcode,
|
|
const char** pcursor,
|
|
const char* query);
|
|
static query_node_t* match_string(const query_field_t* field,
|
|
char not, char opcode,
|
|
const char** pcursor,
|
|
const char* query);
|
|
char* query_unescape(const char* query);
|
|
|
|
query_node_t* query_build(const char* query, const query_field_t* fields)
|
|
{
|
|
query_node_t* left = 0;
|
|
char* raw = query_unescape(query);
|
|
const char* cursor = raw;
|
|
query_node_t* right = 0;
|
|
query_type_t join;
|
|
|
|
if(0 == (left = match_specifier(query, &cursor, fields)))
|
|
goto error;
|
|
|
|
while(*cursor)
|
|
{
|
|
query_node_t* con;
|
|
|
|
switch(*cursor)
|
|
{
|
|
case '+':
|
|
case ' ': join = qot_and; break;
|
|
case ',': join = qot_or; break;
|
|
default:
|
|
DPRINTF(ERR_LOG, "Illegal character '%c' (0%o) at index %d: %s\n",
|
|
*cursor, *cursor, cursor - raw, raw);
|
|
goto error;
|
|
}
|
|
|
|
cursor++;
|
|
|
|
if(0 == (right = match_specifier(raw, &cursor, fields)))
|
|
goto error;
|
|
|
|
con = (query_node_t*) calloc(1, sizeof(*con));
|
|
con->type = join;
|
|
con->left.node = left;
|
|
con->right.node = right;
|
|
|
|
left = con;
|
|
}
|
|
|
|
if(query != raw)
|
|
free(raw);
|
|
|
|
return left;
|
|
|
|
error:
|
|
if(left != 0)
|
|
query_free(left);
|
|
if(raw != query)
|
|
free(raw);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static query_node_t* match_specifier(const char* query,
|
|
const char** cursor,
|
|
const query_field_t* fields)
|
|
{
|
|
switch(**cursor)
|
|
{
|
|
case '\'': return single_match(query, cursor, fields);
|
|
case '(': return group_match(query, cursor, fields);
|
|
}
|
|
|
|
DPRINTF(ERR_LOG, "Illegal character '%c' (0%o) at index %d: %s\n",
|
|
**cursor, **cursor, *cursor - query, query);
|
|
return NULL;
|
|
}
|
|
|
|
static query_node_t* group_match(const char* query,
|
|
const char** pcursor,
|
|
const query_field_t* fields)
|
|
{
|
|
query_node_t* left = 0;
|
|
query_node_t* right = 0;
|
|
query_node_t* join = 0;
|
|
query_type_t opcode;
|
|
const char* cursor = *pcursor;
|
|
|
|
/* skip the opening ')' */
|
|
++cursor;
|
|
|
|
if(0 == (left = single_match(query, &cursor, fields)))
|
|
return NULL;
|
|
|
|
switch(*cursor)
|
|
{
|
|
case '+':
|
|
case ' ':
|
|
opcode = qot_and;
|
|
break;
|
|
|
|
case ',':
|
|
opcode = qot_or;
|
|
break;
|
|
|
|
default:
|
|
DPRINTF(ERR_LOG, "Illegal character '%c' (0%o) at index %d: %s\n",
|
|
*cursor, *cursor, cursor - query, query);
|
|
goto error;
|
|
}
|
|
|
|
if(0 == (right = single_match(query, &cursor, fields)))
|
|
goto error;
|
|
|
|
if(*cursor != ')')
|
|
{
|
|
DPRINTF(ERR_LOG, "Illegal character '%c' (0%o) at index %d: %s\n",
|
|
*cursor, *cursor, cursor - query, query);
|
|
goto error;
|
|
}
|
|
|
|
*pcursor = cursor + 1;
|
|
|
|
join = (query_node_t*) calloc(1, sizeof(*join));
|
|
join->type = opcode;
|
|
join->left.node = left;
|
|
join->right.node = right;
|
|
|
|
return join;
|
|
|
|
error:
|
|
if(0 != left)
|
|
query_free(left);
|
|
if(0 != right)
|
|
query_free(right);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static query_node_t* single_match(const char* query,
|
|
const char** pcursor,
|
|
const query_field_t* fields)
|
|
{
|
|
char fname[64];
|
|
const query_field_t* field;
|
|
char not = 0;
|
|
char op = 0;
|
|
query_node_t* node = 0;
|
|
|
|
/* skip opening ' */
|
|
(*pcursor)++;
|
|
|
|
/* collect the field name */
|
|
if(!get_field_name(pcursor, query, fname, sizeof(fname)))
|
|
return NULL;
|
|
|
|
if(**pcursor == '!')
|
|
{
|
|
not = '!';
|
|
++(*pcursor);
|
|
}
|
|
|
|
if(strchr(":+-", **pcursor))
|
|
{
|
|
op = **pcursor;
|
|
++(*pcursor);
|
|
}
|
|
else
|
|
{
|
|
DPRINTF(ERR_LOG, "Illegal Operator: %c (0%o) at index %d: %s\n",
|
|
**pcursor, **pcursor, *pcursor - query, query);
|
|
return NULL;
|
|
}
|
|
|
|
if(0 == (field = find_field(fname, fields)))
|
|
{
|
|
DPRINTF(ERR_LOG, "Unknown field: %s\n", fname);
|
|
return NULL;
|
|
}
|
|
|
|
switch(field->type)
|
|
{
|
|
case qft_i32:
|
|
case qft_i64:
|
|
node = match_number(field, not, op, pcursor, query);
|
|
break;
|
|
|
|
case qft_string:
|
|
node = match_string(field, not, op, pcursor, query);
|
|
break;
|
|
|
|
default:
|
|
DPRINTF(ERR_LOG, "Invalid field type: %d\n", field->type);
|
|
break;
|
|
}
|
|
|
|
if(**pcursor != '\'')
|
|
{
|
|
DPRINTF(ERR_LOG, "Illegal Character: %c (0%o) index %d: %s\n",
|
|
**pcursor, **pcursor, *pcursor - query, query);
|
|
query_free(node);
|
|
node = 0;
|
|
}
|
|
else
|
|
++(*pcursor);
|
|
|
|
return node;
|
|
}
|
|
|
|
static int get_field_name(const char** pcursor,
|
|
const char* query,
|
|
char* name,
|
|
int len)
|
|
{
|
|
const char* cursor = *pcursor;
|
|
|
|
if(!isalpha(*cursor))
|
|
return 0;
|
|
|
|
while(isalpha(*cursor) || *cursor == '.')
|
|
{
|
|
if(--len <= 0)
|
|
{
|
|
DPRINTF(ERR_LOG, "token length exceeded at offset %d: %s\n",
|
|
cursor - query, query);
|
|
return 0;
|
|
}
|
|
|
|
*name++ = *cursor++;
|
|
}
|
|
|
|
*pcursor = cursor;
|
|
|
|
*name = 0;
|
|
|
|
return 1;
|
|
}
|
|
|
|
static query_node_t* match_number(const query_field_t* field,
|
|
char not, char opcode,
|
|
const char** pcursor,
|
|
const char* query)
|
|
{
|
|
query_node_t* node = (query_node_t*) calloc(1, sizeof(node));
|
|
|
|
switch(opcode)
|
|
{
|
|
case ':':
|
|
node->type = not ? qot_ne : qot_eq;
|
|
break;
|
|
case '+':
|
|
case ' ':
|
|
node->type = not ? qot_le : qot_gt;
|
|
break;
|
|
case '-':
|
|
node->type = not ? qot_ge : qot_lt;
|
|
break;
|
|
}
|
|
|
|
node->left.field = field;
|
|
|
|
switch(field->type)
|
|
{
|
|
case qft_i32:
|
|
node->right.i32 = strtol(*pcursor, (char**) pcursor, 10);
|
|
break;
|
|
case qft_i64:
|
|
node->right.i64 = strtoll(*pcursor, (char**) pcursor, 10);
|
|
break;
|
|
}
|
|
|
|
if(**pcursor != '\'')
|
|
{
|
|
DPRINTF(ERR_LOG, "Illegal char in number '%c' (0%o) at index %d: %s\n",
|
|
**pcursor, **pcursor, *pcursor - query, query);
|
|
free(node);
|
|
return 0;
|
|
}
|
|
|
|
return node;
|
|
}
|
|
|
|
static query_node_t* match_string(const query_field_t* field,
|
|
char not, char opcode,
|
|
const char** pcursor,
|
|
const char* query)
|
|
{
|
|
char match[64];
|
|
char* dst = match;
|
|
int left = sizeof(match);
|
|
const char* cursor = *pcursor;
|
|
query_type_t op = qot_is;
|
|
query_node_t* node;
|
|
|
|
if(opcode != ':')
|
|
{
|
|
DPRINTF(ERR_LOG, "Illegal operation on string: %c at index %d: %s\n",
|
|
opcode, cursor - query - 1);
|
|
return NULL;
|
|
}
|
|
|
|
if(*cursor == '*')
|
|
{
|
|
op = qot_ends;
|
|
cursor++;
|
|
}
|
|
|
|
while(*cursor && *cursor != '\'')
|
|
{
|
|
if(--left == 0)
|
|
{
|
|
DPRINTF(ERR_LOG, "string too long at index %d: %s\n",
|
|
cursor - query, query);
|
|
return NULL;
|
|
}
|
|
|
|
if(*cursor == '\\')
|
|
{
|
|
switch(*++cursor)
|
|
{
|
|
case '*':
|
|
case '\'':
|
|
case '\\':
|
|
*dst++ = *cursor++;
|
|
break;
|
|
default:
|
|
DPRINTF(ERR_LOG, "Illegal escape: %c (0%o) at index %d: %s\n",
|
|
*cursor, *cursor, cursor - query, query);
|
|
return NULL;
|
|
}
|
|
}
|
|
else
|
|
*dst++ = *cursor++;
|
|
}
|
|
|
|
if(dst[-1] == '*')
|
|
{
|
|
op = (op == qot_is) ? qot_begins : qot_contains;
|
|
dst--;
|
|
}
|
|
|
|
*dst = 0;
|
|
|
|
node = (query_node_t*) calloc(1, sizeof(*node));
|
|
node->type = op;
|
|
node->left.field = field;
|
|
node->right.str = strdup(match);
|
|
|
|
*pcursor = cursor;
|
|
|
|
return node;
|
|
}
|
|
|
|
int query_test(query_node_t* query, void* target)
|
|
{
|
|
switch(query->type)
|
|
{
|
|
/* conjunction */
|
|
case qot_and:
|
|
return (query_test(query->left.node, target) &&
|
|
query_test(query->right.node, target));
|
|
|
|
case qot_or:
|
|
return (query_test(query->left.node, target) ||
|
|
query_test(query->right.node, target));
|
|
|
|
/* negation */
|
|
case qot_not:
|
|
return !query_test(query->left.node, target);
|
|
|
|
/* arithmetic */
|
|
case qot_eq:
|
|
case qot_ne:
|
|
case qot_le:
|
|
case qot_lt:
|
|
case qot_ge:
|
|
case qot_gt:
|
|
return arith_query(query, target);
|
|
|
|
/* string */
|
|
case qot_is:
|
|
case qot_begins:
|
|
case qot_ends:
|
|
case qot_contains:
|
|
return string_query(query, target);
|
|
break;
|
|
|
|
/* constants */
|
|
case qot_const:
|
|
return query->left.constant;
|
|
}
|
|
}
|
|
|
|
void query_free(query_node_t* query)
|
|
{
|
|
if(0 != query)
|
|
{
|
|
switch(query->type)
|
|
{
|
|
/* conjunction */
|
|
case qot_and:
|
|
case qot_or:
|
|
query_free(query->left.node);
|
|
query_free(query->right.node);
|
|
break;
|
|
|
|
/* negation */
|
|
case qot_not:
|
|
query_free(query->left.node);
|
|
break;
|
|
|
|
/* arithmetic */
|
|
case qot_eq:
|
|
case qot_ne:
|
|
case qot_le:
|
|
case qot_lt:
|
|
case qot_ge:
|
|
case qot_gt:
|
|
break;
|
|
|
|
/* string */
|
|
case qot_is:
|
|
case qot_begins:
|
|
case qot_ends:
|
|
case qot_contains:
|
|
free(query->right.str);
|
|
break;
|
|
|
|
/* constants */
|
|
case qot_const:
|
|
break;
|
|
|
|
default:
|
|
DPRINTF(ERR_LOG, "Illegal query type: %d\n", query->type);
|
|
break;
|
|
}
|
|
|
|
free(query);
|
|
}
|
|
}
|
|
|
|
static const query_field_t* find_field(const char* name, const query_field_t* fields)
|
|
{
|
|
while(fields->name && strcasecmp(fields->name, name))
|
|
fields++;
|
|
|
|
if(fields->name == 0)
|
|
{
|
|
DPRINTF(ERR_LOG, "Illegal query field: %s\n", name);
|
|
return NULL;
|
|
}
|
|
|
|
return fields;
|
|
}
|
|
|
|
static int arith_query(query_node_t* query, void* target)
|
|
{
|
|
const query_field_t* field = query->left.field;
|
|
|
|
switch(field->type)
|
|
{
|
|
case qft_i32:
|
|
{
|
|
int tv = * (int*) ((size_t) target + field->offset);
|
|
|
|
tv -= query->right.i32;
|
|
|
|
switch(query->type)
|
|
{
|
|
case qot_eq: return tv == 0;
|
|
case qot_ne: return tv != 0;
|
|
case qot_le: return tv <= 0;
|
|
case qot_lt: return tv < 0;
|
|
case qot_ge: return tv >= 0;
|
|
case qot_gt: return tv > 0;
|
|
default:
|
|
DPRINTF(ERR_LOG, "illegal query type: %d\n", query->type);
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case qft_i64:
|
|
{
|
|
long long tv = * (long long*) ((size_t) target + field->offset);
|
|
|
|
tv -= query->right.i32;
|
|
|
|
switch(query->type)
|
|
{
|
|
case qot_eq: return tv == 0;
|
|
case qot_ne: return tv != 0;
|
|
case qot_le: return tv <= 0;
|
|
case qot_lt: return tv < 0;
|
|
case qot_ge: return tv >= 0;
|
|
case qot_gt: return tv > 0;
|
|
default:
|
|
DPRINTF(ERR_LOG, "illegal query type: %d\n", query->type);
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
DPRINTF(ERR_LOG, "illegal field type: %d\n", field->type);
|
|
break;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int string_query(query_node_t* query, void* target)
|
|
{
|
|
const query_field_t* field = query->left.field;
|
|
const char* ts;
|
|
|
|
if(field->type != qft_string)
|
|
{
|
|
DPRINTF(ERR_LOG, "illegal field type: %d\n", field->type);
|
|
return 0;
|
|
}
|
|
|
|
ts = * (const char**) ((size_t) target + field->offset);
|
|
|
|
if(0 == ts)
|
|
return strlen(query->right.str) == 0;
|
|
|
|
switch(query->type)
|
|
{
|
|
case qot_is:
|
|
return !strcasecmp(query->right.str, ts);
|
|
|
|
case qot_begins:
|
|
return !strncasecmp(query->right.str, ts, strlen(query->right.str));
|
|
|
|
case qot_ends:
|
|
{
|
|
int start = strlen(ts) - strlen(query->right.str);
|
|
|
|
if(start < 0)
|
|
return 0;
|
|
|
|
return !strcasecmp(query->right.str, ts + start);
|
|
}
|
|
|
|
case qot_contains:
|
|
return !strcasestr(ts, query->right.str);
|
|
|
|
default:
|
|
DPRINTF(ERR_LOG, "Illegal query type: %d\n", query->type);
|
|
break;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void query_dump(FILE* fp, query_node_t* query, int depth)
|
|
{
|
|
static const char* labels[] = {
|
|
"NOP",
|
|
"and",
|
|
"or",
|
|
"not",
|
|
"==",
|
|
"!=",
|
|
"<=",
|
|
"<",
|
|
">=",
|
|
">",
|
|
"eq",
|
|
"beginswith",
|
|
"endwith",
|
|
"contains",
|
|
"constant"
|
|
};
|
|
|
|
switch(query->type)
|
|
{
|
|
case qot_and:
|
|
case qot_or:
|
|
fprintf(fp, "%*s(%s\n", depth, "", labels[query->type]);
|
|
query_dump(fp, query->left.node, depth + 4);
|
|
query_dump(fp, query->right.node, depth + 4);
|
|
fprintf(fp, "%*s)\n", depth, "");
|
|
break;
|
|
|
|
case qot_not:
|
|
fprintf(fp, "%*s(not\n", depth, "");
|
|
query_dump(fp, query->left.node, depth + 4);
|
|
fprintf(fp, "%*s)\n", depth, "");
|
|
break;
|
|
|
|
/* arithmetic */
|
|
case qot_eq:
|
|
case qot_ne:
|
|
case qot_le:
|
|
case qot_lt:
|
|
case qot_ge:
|
|
case qot_gt:
|
|
if(query->left.field->type == qft_i32)
|
|
fprintf(fp, "%*s(%s %s %d)\n",
|
|
depth, "", labels[query->type],
|
|
query->left.field->name, query->right.i32);
|
|
else
|
|
fprintf(fp, "%*s(%s %s %q)\n",
|
|
depth, "", labels[query->type],
|
|
query->left.field->name, query->right.i64);
|
|
break;
|
|
|
|
/* string */
|
|
case qot_is:
|
|
case qot_begins:
|
|
case qot_ends:
|
|
case qot_contains:
|
|
fprintf(fp, "%*s(%s %s \"%s\")\n",
|
|
depth, "", labels[query->type],
|
|
query->left.field->name, query->right.str);
|
|
break;
|
|
|
|
/* constants */
|
|
case qot_const:
|
|
fprintf(fp, "%*s(%s)\n", depth, "", query->left.constant ? "true" : "false");
|
|
break;
|
|
}
|
|
|
|
}
|
|
|
|
char* query_unescape(const char* src)
|
|
{
|
|
char* copy = malloc(strlen(src) + 1);
|
|
char* dst = copy;
|
|
|
|
while(*src)
|
|
{
|
|
if(*src == '%')
|
|
{
|
|
int val = 0;
|
|
|
|
if(*++src)
|
|
{
|
|
if(isdigit(*src))
|
|
val = val * 16 + *src - '0';
|
|
else
|
|
val = val * 16 + tolower(*src) - 'a' + 10;
|
|
}
|
|
|
|
if(*++src)
|
|
{
|
|
if(isdigit(*src))
|
|
val = val * 16 + *src - '0';
|
|
else
|
|
val = val * 16 + tolower(*src) - 'a' + 10;
|
|
}
|
|
|
|
src++;
|
|
*dst++ = val;
|
|
}
|
|
else
|
|
*dst++ = *src++;
|
|
}
|
|
|
|
*dst++ = 0;
|
|
|
|
return copy;
|
|
}
|