mirror of
https://github.com/owntone/owntone-server.git
synced 2025-01-18 18:23:18 -05:00
[smartpl] Add ESCAPE to LIKE queries where the value contains % and _
Plus various minor improvements
This commit is contained in:
parent
9233c03ca8
commit
9c1f7dd380
@ -50,6 +50,7 @@
|
|||||||
|
|
||||||
%option case-insensitive
|
%option case-insensitive
|
||||||
|
|
||||||
|
/* quoted \"(\\(.|\n)|[^\\"\n])*\" */
|
||||||
quoted \"[^\"\n]*[\"\n]
|
quoted \"[^\"\n]*[\"\n]
|
||||||
yyyymmdd [0-9][0-9][0-9][0-9]-[0-9][0-9]-[0-9][0-9]
|
yyyymmdd [0-9][0-9][0-9][0-9]-[0-9][0-9]-[0-9][0-9]
|
||||||
|
|
||||||
|
@ -188,31 +188,9 @@ int smartpl_lex_parse(struct smartpl_result *result, const char *input);
|
|||||||
%code top {
|
%code top {
|
||||||
#ifndef DEBUG_PARSER_MOCK
|
#ifndef DEBUG_PARSER_MOCK
|
||||||
#include "db.h"
|
#include "db.h"
|
||||||
|
#include "misc.h"
|
||||||
#else
|
#else
|
||||||
enum data_kind {
|
#include "owntonefunctions.h"
|
||||||
DATA_KIND_FILE = 0,
|
|
||||||
DATA_KIND_HTTP = 1,
|
|
||||||
DATA_KIND_SPOTIFY = 2,
|
|
||||||
DATA_KIND_PIPE = 3,
|
|
||||||
};
|
|
||||||
|
|
||||||
enum media_kind {
|
|
||||||
MEDIA_KIND_MUSIC = 1,
|
|
||||||
MEDIA_KIND_MOVIE = 2,
|
|
||||||
MEDIA_KIND_PODCAST = 4,
|
|
||||||
MEDIA_KIND_AUDIOBOOK = 8,
|
|
||||||
MEDIA_KIND_MUSICVIDEO = 32,
|
|
||||||
MEDIA_KIND_TVSHOW = 64,
|
|
||||||
};
|
|
||||||
|
|
||||||
static char * db_escape_string(const char *str)
|
|
||||||
{
|
|
||||||
char *new = strdup(str);
|
|
||||||
char *ptr;
|
|
||||||
while ((ptr = strpbrk(new, "\\'")))
|
|
||||||
*ptr = 'X';
|
|
||||||
return new;
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -253,6 +231,27 @@ enum sql_append_type {
|
|||||||
|
|
||||||
static void sql_from_ast(struct smartpl_result *, struct result_part *, struct ast *);
|
static void sql_from_ast(struct smartpl_result *, struct result_part *, struct ast *);
|
||||||
|
|
||||||
|
// Escapes any '%' or '_' that might be in the string
|
||||||
|
static void sql_like_escape(char **value, char *escape_char)
|
||||||
|
{
|
||||||
|
char *s = *value;
|
||||||
|
size_t len = strlen(s);
|
||||||
|
char *new;
|
||||||
|
|
||||||
|
*escape_char = 0;
|
||||||
|
|
||||||
|
// Fast path, nothing to escape
|
||||||
|
if (!strpbrk(s, "_%"))
|
||||||
|
return;
|
||||||
|
|
||||||
|
len = 2 * len; // Enough for every char to be escaped
|
||||||
|
new = realloc(s, len);
|
||||||
|
safe_snreplace(new, len, "%", "\\%");
|
||||||
|
safe_snreplace(new, len, "_", "\\_");
|
||||||
|
*escape_char = '\\';
|
||||||
|
*value = new;
|
||||||
|
}
|
||||||
|
|
||||||
static void sql_str_escape(char **value)
|
static void sql_str_escape(char **value)
|
||||||
{
|
{
|
||||||
char *old = *value;
|
char *old = *value;
|
||||||
@ -285,6 +284,8 @@ static void sql_append(struct smartpl_result *result, struct result_part *part,
|
|||||||
|
|
||||||
static void sql_append_recursive(struct smartpl_result *result, struct result_part *part, struct ast *a, const char *op, const char *op_not, bool is_not, enum sql_append_type append_type)
|
static void sql_append_recursive(struct smartpl_result *result, struct result_part *part, struct ast *a, const char *op, const char *op_not, bool is_not, enum sql_append_type append_type)
|
||||||
{
|
{
|
||||||
|
char escape_char;
|
||||||
|
|
||||||
switch (append_type)
|
switch (append_type)
|
||||||
{
|
{
|
||||||
case SQL_APPEND_OPERATOR:
|
case SQL_APPEND_OPERATOR:
|
||||||
@ -301,8 +302,11 @@ static void sql_append_recursive(struct smartpl_result *result, struct result_pa
|
|||||||
case SQL_APPEND_OPERATOR_LIKE:
|
case SQL_APPEND_OPERATOR_LIKE:
|
||||||
sql_from_ast(result, part, a->l);
|
sql_from_ast(result, part, a->l);
|
||||||
sql_append(result, part, " %s '%%", is_not ? op_not : op);
|
sql_append(result, part, " %s '%%", is_not ? op_not : op);
|
||||||
|
sql_like_escape((char **)(&a->r->data), &escape_char);
|
||||||
sql_from_ast(result, part, a->r);
|
sql_from_ast(result, part, a->r);
|
||||||
sql_append(result, part, "%%'");
|
sql_append(result, part, "%%'");
|
||||||
|
if (escape_char)
|
||||||
|
sql_append(result, part, " ESCAPE '%c'", escape_char);
|
||||||
break;
|
break;
|
||||||
case SQL_APPEND_FIELD:
|
case SQL_APPEND_FIELD:
|
||||||
assert(a->l == NULL);
|
assert(a->l == NULL);
|
||||||
|
@ -90,6 +90,8 @@ smartpl_query_parse_string(struct smartpl *smartpl, const char *expression)
|
|||||||
{
|
{
|
||||||
struct smartpl_result result;
|
struct smartpl_result result;
|
||||||
|
|
||||||
|
DPRINTF(E_SPAM, L_SCAN, "Parse smartpl query input '%s'\n", expression);
|
||||||
|
|
||||||
if (smartpl_lex_parse(&result, expression) != 0)
|
if (smartpl_lex_parse(&result, expression) != 0)
|
||||||
{
|
{
|
||||||
DPRINTF(E_LOG, L_SCAN, "Could not parse '%s': %s\n", expression, result.errmsg);
|
DPRINTF(E_LOG, L_SCAN, "Could not parse '%s': %s\n", expression, result.errmsg);
|
||||||
@ -110,6 +112,9 @@ smartpl_query_parse_string(struct smartpl *smartpl, const char *expression)
|
|||||||
smartpl->order = safe_strdup(result.order);
|
smartpl->order = safe_strdup(result.order);
|
||||||
smartpl->limit = result.limit;
|
smartpl->limit = result.limit;
|
||||||
|
|
||||||
|
DPRINTF(E_SPAM, L_SCAN, "Parse smartpl query output '%s': WHERE %s HAVING %s ORDER BY %s LIMIT %d\n",
|
||||||
|
smartpl->title, smartpl->query_where, smartpl->having, smartpl->order, smartpl->limit);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user