[mpd] Add Bison/Flex parser for complex mpd commands + misc

Some of the misc:
* Initialize query_params from the start of each function
* Reduce code duplication by consolidating the handler's integer conversion
* Go back to classic int return types for handlers
* Change list grouping to respond like mpd
* Sanitize all user string output
This commit is contained in:
ejurgensen 2024-08-31 00:02:25 +02:00 committed by Alain Nussbaumer
parent 08b1b74ddd
commit d57b7565da
4 changed files with 2139 additions and 1866 deletions

View File

@ -53,8 +53,8 @@ GPERF_FILES = \
GPERF_SRC = $(GPERF_FILES:.gperf=_hash.h)
LEXER_SRC = parsers/daap_lexer.l parsers/smartpl_lexer.l parsers/rsp_lexer.l
PARSER_SRC = parsers/daap_parser.y parsers/smartpl_parser.y parsers/rsp_parser.y
LEXER_SRC = parsers/daap_lexer.l parsers/smartpl_lexer.l parsers/rsp_lexer.l parsers/mpd_lexer.l
PARSER_SRC = parsers/daap_parser.y parsers/smartpl_parser.y parsers/rsp_parser.y parsers/mpd_parser.y
# This flag is given to Bison and tells it to produce headers. Note that
# automake recognizes this flag too, and has special logic around it, so don't

3071
src/mpd.c

File diff suppressed because it is too large Load Diff

124
src/parsers/mpd_lexer.l Normal file
View File

@ -0,0 +1,124 @@
/*
* Copyright (C) 2021-2022 Espen Jürgensen <espenjurgensen@gmail.com>
*
* 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; either version 2 of the License, or
* (at your option) any later version.
*
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
/* =========================== BOILERPLATE SECTION ===========================*/
/* This is to avoid compiler warnings about unused functions. More options are
noyyalloc noyyrealloc noyyfree. */
%option noyywrap nounput noinput
/* Thread safe scanner */
%option reentrant
/* To avoid symbol name conflicts with multiple lexers */
%option prefix="mpd_"
/* Automake's ylwrap expexts the output to have this name */
%option outfile="lex.yy.c"
/* Makes a Bison-compatible yylex */
%option bison-bridge
%{
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>
#include "mpd_parser.h"
/* Unknown why this is required despite using prefix */
#define YYSTYPE MPD_STYPE
%}
/* ========================= NON-BOILERPLATE SECTION =========================*/
%option case-insensitive
singlequoted '(\\.|[^'\\])*'
doublequoted \"(\\.|[^"\\])*\"
yyyymmdd [0-9][0-9][0-9][0-9]-[0-9][0-9]-[0-9][0-9]
%%
[\n\r\t ]+ /* Ignore whitespace */
^playlistfind { return MPD_T_CMDPLAYLISTFIND; }
^playlistsearch { return MPD_T_CMDPLAYLISTSEARCH; }
^count { return MPD_T_CMDCOUNT; }
^find { return MPD_T_CMDFIND; }
^findadd { return MPD_T_CMDFINDADD; }
^list { return MPD_T_CMDLIST; }
^search { return MPD_T_CMDSEARCH; }
^searchadd { return MPD_T_CMDSEARCHADD; }
^searchcount { return MPD_T_CMDSEARCHCOUNT; }
sort { return MPD_T_SORT; }
window { return MPD_T_WINDOW; }
position { return MPD_T_POSITION; }
group { return MPD_T_GROUP; }
artist { yylval->str = strdup(yytext); return MPD_T_STRTAG; }
artistsort { yylval->str = strdup(yytext); return MPD_T_STRTAG; }
albumartist { yylval->str = strdup(yytext); return MPD_T_STRTAG; }
albumartistsort { yylval->str = strdup(yytext); return MPD_T_STRTAG; }
album { yylval->str = strdup(yytext); return MPD_T_STRTAG; }
albumsort { yylval->str = strdup(yytext); return MPD_T_STRTAG; }
title { yylval->str = strdup(yytext); return MPD_T_STRTAG; }
titlesort { yylval->str = strdup(yytext); return MPD_T_STRTAG; }
composer { yylval->str = strdup(yytext); return MPD_T_STRTAG; }
composersort { yylval->str = strdup(yytext); return MPD_T_STRTAG; }
genre { yylval->str = strdup(yytext); return MPD_T_STRTAG; }
file { yylval->str = strdup(yytext); return MPD_T_STRTAG; }
base { yylval->str = strdup(yytext); return MPD_T_BASETAG; }
track { yylval->str = strdup(yytext); return MPD_T_INTTAG; }
disc { yylval->str = strdup(yytext); return MPD_T_INTTAG; }
date { yylval->str = strdup(yytext); return MPD_T_INTTAG; }
modified-since { yylval->str = strdup(yytext); return MPD_T_SINCETAG; }
added-since { yylval->str = strdup(yytext); return MPD_T_SINCETAG; }
contains { return (yylval->ival = MPD_T_CONTAINS); }
starts_with { return (yylval->ival = MPD_T_STARTSWITH); }
ends_with { return (yylval->ival = MPD_T_ENDSWITH); }
== { return (yylval->ival = MPD_T_EQUAL); }
!= { return (yylval->ival = MPD_T_NOTEQUAL); }
\<= { return (yylval->ival = MPD_T_LESSEQUAL); }
\< { return (yylval->ival = MPD_T_LESS); }
\>= { return (yylval->ival = MPD_T_GREATEREQUAL); }
\> { return (yylval->ival = MPD_T_GREATER); }
audioformat { return MPD_T_AUDIOFORMATTAG; }
any { return MPD_T_ANYTAG; }
or { return MPD_T_OR; }
and { return MPD_T_AND; }
not { return MPD_T_NOT; }
! { return MPD_T_NOT; }
{singlequoted} { yylval->str = mpd_parser_quoted(yytext); return MPD_T_STRING; }
{doublequoted} { yylval->str = mpd_parser_quoted(yytext); return MPD_T_STRING; }
[0-9]+ { yylval->ival=atoi(yytext); return MPD_T_NUM; }
. { return yytext[0]; }
%%

806
src/parsers/mpd_parser.y Normal file
View File

@ -0,0 +1,806 @@
/*
* Copyright (C) 2021-2022 Espen Jürgensen <espenjurgensen@gmail.com>
*
* 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; either version 2 of the License, or
* (at your option) any later version.
*
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
/* =========================== BOILERPLATE SECTION ===========================*/
/* No global variables and yylex has scanner as argument */
%define api.pure true
/* Change prefix of symbols from yy to avoid clashes with any other parsers we
may want to link */
%define api.prefix {mpd_}
/* Gives better errors than "syntax error" */
%define parse.error verbose
/* Enables debug mode */
%define parse.trace
/* Adds output parameter to the parser */
%parse-param {struct mpd_result *result}
/* Adds "scanner" as argument to the parses calls to yylex, which is required
when the lexer is in reentrant mode. The type is void because caller caller
shouldn't need to know about yyscan_t */
%param {void *scanner}
%code provides {
/* Convenience functions for caller to use instead of interfacing with lexer and
parser directly */
int mpd_lex_cb(char *input, void (*cb)(int, const char *));
int mpd_lex_parse(struct mpd_result *result, const char *input);
}
/* Implementation of the convenience function and the parsing error function
required by Bison */
%code {
#include "mpd_lexer.h"
int mpd_lex_cb(char *input, void (*cb)(int, const char *))
{
int ret;
yyscan_t scanner;
YY_BUFFER_STATE buf;
YYSTYPE val;
if ((ret = mpd_lex_init(&scanner)) != 0)
return ret;
buf = mpd__scan_string(input, scanner);
while ((ret = mpd_lex(&val, scanner)) > 0)
cb(ret, mpd_get_text(scanner));
mpd__delete_buffer(buf, scanner);
mpd_lex_destroy(scanner);
return 0;
}
int mpd_lex_parse(struct mpd_result *result, const char *input)
{
YY_BUFFER_STATE buffer;
yyscan_t scanner;
int retval = -1;
int ret;
result->errmsg[0] = '\0'; // For safety
ret = mpd_lex_init(&scanner);
if (ret != 0)
goto error_init;
buffer = mpd__scan_string(input, scanner);
if (!buffer)
goto error_buffer;
ret = mpd_parse(result, scanner);
if (ret != 0)
goto error_parse;
retval = 0;
error_parse:
mpd__delete_buffer(buffer, scanner);
error_buffer:
mpd_lex_destroy(scanner);
error_init:
return retval;
}
void mpd_error(struct mpd_result *result, yyscan_t scanner, const char *msg)
{
snprintf(result->errmsg, sizeof(result->errmsg), "%s", msg);
}
}
/* ============ ABSTRACT SYNTAX TREE (AST) BOILERPLATE SECTION ===============*/
%code {
struct ast
{
int type;
struct ast *l;
struct ast *r;
void *data;
int ival;
};
__attribute__((unused)) static struct ast * ast_new(int type, struct ast *l, struct ast *r)
{
struct ast *a = calloc(1, sizeof(struct ast));
a->type = type;
a->l = l;
a->r = r;
return a;
}
/* Note *data is expected to be freeable with regular free() */
__attribute__((unused)) static struct ast * ast_data(int type, void *data)
{
struct ast *a = calloc(1, sizeof(struct ast));
a->type = type;
a->data = data;
return a;
}
__attribute__((unused)) static struct ast * ast_int(int type, int ival)
{
struct ast *a = calloc(1, sizeof(struct ast));
a->type = type;
a->ival = ival;
return a;
}
__attribute__((unused)) static void ast_free(struct ast *a)
{
if (!a)
return;
ast_free(a->l);
ast_free(a->r);
free(a->data);
free(a);
}
}
%destructor { free($$); } <str>
%destructor { ast_free($$); } <ast>
/* ========================= NON-BOILERPLATE SECTION =========================*/
/* Includes required by the parser rules */
%code top {
#ifndef _GNU_SOURCE
#define _GNU_SOURCE // For asprintf
#endif
#include <stdio.h>
#include <stdlib.h>
#include <stdbool.h>
#include <stdarg.h> // For vsnprintf
#include <string.h>
#include <time.h>
#include <assert.h>
#define INVERT_MASK 0x80000000
}
/* Dependencies, mocked or real */
%code top {
#ifndef DEBUG_PARSER_MOCK
#include "db.h"
#include "misc.h"
#else
#include "owntonefunctions.h"
#endif
}
/* Definition of struct that will hold the parsing result
* Some users have sizeable smart playlists, e.g. listing many artist names,
* which translate to sizeable sql queries.
*/
%code requires {
struct mpd_result_part {
char str[8192];
int offset;
};
struct mpd_result {
struct mpd_result_part where_part;
struct mpd_result_part order_part;
struct mpd_result_part group_part;
char tagtype_buf[64];
char position_buf[64];
// Pointers to the strings in mpd_result_part
const char *where;
const char *order;
const char *group;
const char *tagtype;
const char *position;
// Set to 0 if not found
int offset;
int limit;
int err;
char errmsg[128];
};
enum mpd_type {
MPD_TYPE_INT,
MPD_TYPE_STRING,
MPD_TYPE_SPECIAL,
};
struct mpd_tag_map {
const char *name;
const char *dbcol;
enum mpd_type type;
int dbmfi_offset;
};
char *mpd_parser_quoted(const char *str);
struct mpd_tag_map *mpd_parser_tag_from_dbcol(const char *dbcol);
void mpd_parser_enum_tagtypes(void (*func)(struct mpd_tag_map *, void *), void *arg);
}
%code {
enum sql_append_type {
SQL_APPEND_OPERATOR,
SQL_APPEND_OPERATOR_STR,
SQL_APPEND_OPERATOR_LIKE,
SQL_APPEND_FIELD,
SQL_APPEND_STR,
SQL_APPEND_INT,
SQL_APPEND_TIME,
SQL_APPEND_ORDER,
SQL_APPEND_PARENS,
};
static struct mpd_tag_map mpd_tag_map[] =
{
{ "Artist", "f.artist", MPD_TYPE_STRING, dbmfi_offsetof(artist), },
{ "ArtistSort", "f.artist_sort", MPD_TYPE_STRING, dbmfi_offsetof(artist_sort), },
{ "AlbumArtist", "f.album_artist", MPD_TYPE_STRING, dbmfi_offsetof(album_artist), },
{ "AlbumArtistSort", "f.album_artist_sort", MPD_TYPE_STRING, dbmfi_offsetof(album_artist_sort), },
{ "Album", "f.album", MPD_TYPE_STRING, dbmfi_offsetof(album), },
{ "AlbumSort", "f.album_sort", MPD_TYPE_STRING, dbmfi_offsetof(album_sort), },
{ "Title", "f.title", MPD_TYPE_STRING, dbmfi_offsetof(title), },
{ "TitleSort", "f.title_sort", MPD_TYPE_STRING, dbmfi_offsetof(title_sort), },
{ "Genre", "f.genre", MPD_TYPE_STRING, dbmfi_offsetof(genre), },
{ "Composer", "f.composer", MPD_TYPE_STRING, dbmfi_offsetof(composer), },
{ "ComposerSort", "f.composer_sort", MPD_TYPE_STRING, dbmfi_offsetof(composer_sort), },
{ "file", "f.virtual_path", MPD_TYPE_SPECIAL, dbmfi_offsetof(virtual_path), },
{ "base", "f.virtual_path", MPD_TYPE_SPECIAL, dbmfi_offsetof(virtual_path), },
{ "Track", "f.track", MPD_TYPE_INT, dbmfi_offsetof(track), },
{ "Disc", "f.disc", MPD_TYPE_INT, dbmfi_offsetof(disc), },
{ "Date", "f.year", MPD_TYPE_INT, dbmfi_offsetof(year), },
{ "modified-since", "f.time_modified", MPD_TYPE_SPECIAL, dbmfi_offsetof(time_modified), },
{ "added-since", "f.time_added", MPD_TYPE_SPECIAL, dbmfi_offsetof(time_added), },
// AudioFormat tag
{ "samplerate", "f.samplerate", MPD_TYPE_INT, dbmfi_offsetof(samplerate), },
{ "bits_per_sample", "f.bits_per_sample", MPD_TYPE_INT, dbmfi_offsetof(bits_per_sample), },
{ "channels", "f.channels", MPD_TYPE_INT, dbmfi_offsetof(channels), },
{ NULL },
};
static const char *
tag_to_dbcol(const char *tag)
{
struct mpd_tag_map *mapptr;
for (mapptr = mpd_tag_map; mapptr->name; mapptr++)
{
if (strcasecmp(tag, mapptr->name) == 0)
return mapptr->dbcol;
}
return "error"; // Should never happen, means tag_to_db_map is out of sync with lexer
}
struct mpd_tag_map *
mpd_parser_tag_from_dbcol(const char *dbcol)
{
struct mpd_tag_map *mapptr;
if (!dbcol)
return NULL;
for (mapptr = mpd_tag_map; mapptr->name; mapptr++)
{
if (strcasecmp(dbcol, mapptr->dbcol) == 0)
return mapptr;
}
return NULL;
}
void
mpd_parser_enum_tagtypes(void (*func)(struct mpd_tag_map *, void *), void *arg)
{
struct mpd_tag_map *mapptr;
for (mapptr = mpd_tag_map; mapptr->name; mapptr++)
{
func(mapptr, arg);
}
}
// Remove any backslash that was used to escape single or double quotes
char *
mpd_parser_quoted(const char *str)
{
char *out = strdup(str + 1); // Copy from after the first quote
size_t len = strlen(out);
const char *src;
char *dst;
out[len - 1] = '\0'; // Remove terminating quote
// Remove escaping backslashes
for (src = dst = out; *src != '\0'; src++, dst++)
{
if (*src == '\\')
src++;
if (*src == '\0')
break;
*dst = *src;
}
*dst = '\0';
return out;
}
static void sql_from_ast(struct mpd_result *, struct mpd_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)
{
char *old = *value;
*value = db_escape_string(old);
free(old);
}
static void sql_append(struct mpd_result *result, struct mpd_result_part *part, const char *fmt, ...)
{
va_list ap;
int remaining = sizeof(part->str) - part->offset;
int ret;
if (remaining <= 0)
goto nospace;
va_start(ap, fmt);
ret = vsnprintf(part->str + part->offset, remaining, fmt, ap);
va_end(ap);
if (ret < 0 || ret >= remaining)
goto nospace;
part->offset += ret;
return;
nospace:
snprintf(result->errmsg, sizeof(result->errmsg), "Parser output buffer too small (%zu bytes)", sizeof(part->str));
result->err = -2;
}
static void sql_append_recursive(struct mpd_result *result, struct mpd_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)
{
case SQL_APPEND_OPERATOR:
sql_from_ast(result, part, a->l);
sql_append(result, part, " %s ", is_not ? op_not : op);
sql_from_ast(result, part, a->r);
break;
case SQL_APPEND_OPERATOR_STR:
sql_from_ast(result, part, a->l);
sql_append(result, part, " %s '", is_not ? op_not : op);
sql_from_ast(result, part, a->r);
sql_append(result, part, "'");
break;
case SQL_APPEND_OPERATOR_LIKE:
sql_from_ast(result, part, a->l);
sql_append(result, part, " %s '%s", is_not ? op_not : op, a->type == MPD_T_STARTSWITH ? "" : "%");
sql_like_escape((char **)(&a->r->data), &escape_char);
sql_from_ast(result, part, a->r);
sql_append(result, part, "%s'", a->type == MPD_T_ENDSWITH ? "" : "%");
if (escape_char)
sql_append(result, part, " ESCAPE '%c'", escape_char);
break;
case SQL_APPEND_FIELD:
assert(a->l == NULL);
assert(a->r == NULL);
sql_append(result, part, "%s", tag_to_dbcol((char *)a->data));
break;
case SQL_APPEND_STR:
assert(a->l == NULL);
assert(a->r == NULL);
sql_str_escape((char **)&a->data);
sql_append(result, part, "%s", (char *)a->data);
break;
case SQL_APPEND_INT:
assert(a->l == NULL);
assert(a->r == NULL);
sql_append(result, part, "%d", a->ival);
break;
case SQL_APPEND_TIME:
assert(a->l == NULL);
assert(a->r == NULL);
sql_str_escape((char **)&a->data);
// MPD docs say the value can be a unix timestamp or ISO 8601
sql_append(result, part, "strftime('%%s', datetime('%s', '%s'))", (char *)a->data, strchr((char *)a->data, '-') ? "utc" : "unixepoch");
break;
case SQL_APPEND_ORDER:
assert(a->l == NULL);
assert(a->r == NULL);
if (a->data)
sql_append(result, part, "%s ", tag_to_dbcol((char *)a->data));
sql_append(result, part, "%s", is_not ? op_not : op);
break;
case SQL_APPEND_PARENS:
assert(a->r == NULL);
if (is_not ? op_not : op)
sql_append(result, part, "%s ", is_not ? op_not : op);
sql_append(result, part, "(");
sql_from_ast(result, part, a->l);
sql_append(result, part, ")");
break;
}
}
/* Creates the parsing result from the AST. Errors are set via result->err. */
static void sql_from_ast(struct mpd_result *result, struct mpd_result_part *part, struct ast *a) {
if (!a || result->err < 0)
return;
bool is_not = (a->type & INVERT_MASK);
a->type &= ~INVERT_MASK;
switch (a->type)
{
case MPD_T_LESS:
sql_append_recursive(result, part, a, "<", ">=", is_not, SQL_APPEND_OPERATOR); break;
case MPD_T_LESSEQUAL:
sql_append_recursive(result, part, a, "<=", ">", is_not, SQL_APPEND_OPERATOR); break;
case MPD_T_GREATER:
sql_append_recursive(result, part, a, ">", ">=", is_not, SQL_APPEND_OPERATOR); break;
case MPD_T_GREATEREQUAL:
sql_append_recursive(result, part, a, ">=", "<", is_not, SQL_APPEND_OPERATOR); break;
case MPD_T_EQUAL:
sql_append_recursive(result, part, a, "=", "!=", is_not, SQL_APPEND_OPERATOR_STR); break;
case MPD_T_NOTEQUAL:
sql_append_recursive(result, part, a, "!=", "=", is_not, SQL_APPEND_OPERATOR_STR); break;
case MPD_T_CONTAINS:
case MPD_T_STARTSWITH:
case MPD_T_ENDSWITH:
sql_append_recursive(result, part, a, "LIKE", "NOT LIKE", is_not, SQL_APPEND_OPERATOR_LIKE); break;
case MPD_T_AND:
sql_append_recursive(result, part, a, "AND", "AND NOT", is_not, SQL_APPEND_OPERATOR); break;
case MPD_T_OR:
sql_append_recursive(result, part, a, "OR", "OR NOT", is_not, SQL_APPEND_OPERATOR); break;
case MPD_T_STRING:
sql_append_recursive(result, part, a, NULL, NULL, 0, SQL_APPEND_STR); break;
case MPD_T_STRTAG:
case MPD_T_INTTAG:
sql_append_recursive(result, part, a, NULL, NULL, 0, SQL_APPEND_FIELD); break;
case MPD_T_NUM:
sql_append_recursive(result, part, a, NULL, NULL, 0, SQL_APPEND_INT); break;
case MPD_T_TIME:
sql_append_recursive(result, part, a, NULL, NULL, 0, SQL_APPEND_TIME); break;
case MPD_T_SORT:
sql_append_recursive(result, part, a, "ASC", "DESC", is_not, SQL_APPEND_ORDER); break;
case MPD_T_GROUP:
sql_append_recursive(result, part, a, ",", ",", is_not, SQL_APPEND_OPERATOR); break;
case MPD_T_PARENS:
sql_append_recursive(result, part, a, NULL, "NOT", is_not, SQL_APPEND_PARENS); break;
default:
snprintf(result->errmsg, sizeof(result->errmsg), "Parser produced unrecognized AST type %d", a->type);
result->err = -1;
}
}
static int result_set(struct mpd_result *result, char *tagtype, struct ast *filter, struct ast *sort, struct ast *window, char *position, struct ast *group)
{
memset(result, 0, sizeof(struct mpd_result));
if (tagtype)
{
snprintf(result->tagtype_buf, sizeof(result->tagtype_buf), "%s", tag_to_dbcol(tagtype));
result->tagtype = result->tagtype_buf;
}
sql_from_ast(result, &result->where_part, filter);
if (result->where_part.offset)
result->where = result->where_part.str;
sql_from_ast(result, &result->order_part, sort);
if (result->order_part.offset)
result->order = result->order_part.str;
sql_from_ast(result, &result->group_part, group);
if (tagtype)
sql_append(result, &result->group_part, result->group_part.offset ? " , %s" : "%s", tag_to_dbcol(tagtype));
if (result->group_part.offset)
result->group = result->group_part.str;
if (position)
{
snprintf(result->position_buf, sizeof(result->position_buf), "%s", position);
result->position = result->position_buf;
}
if (window && window->l->ival <= window->r->ival)
{
result->offset = window->l->ival;
result->limit = window->r->ival - window->l->ival + 1;
}
free(tagtype);
free(position);
ast_free(filter);
ast_free(sort);
ast_free(group);
ast_free(window);
return result->err;
}
static struct ast * ast_new_strnode(int type, const char *tag, const char *value)
{
return ast_new(type, ast_data(MPD_T_STRTAG, strdup(tag)), ast_data(MPD_T_STRING, strdup(value)));
}
/* This creates an OR ast tree with each tag in the tags array */
static struct ast * ast_new_any(int type, const char *value)
{
const char *tags[] = { "albumartist", "artist", "album", "title", NULL, };
const char **tagptr = tags;
struct ast *a;
int op = (type == MPD_T_NOTEQUAL) ? MPD_T_AND : MPD_T_OR;
a = ast_new_strnode(type, *tagptr, value);
for (tagptr++; *tagptr; tagptr++)
a = ast_new(op, a, ast_new_strnode(type, *tagptr, value));
// In case the expression will be negated, e.g. !(any contains 'cat'), we must
// group the result
return ast_new(MPD_T_PARENS, a, NULL);
}
static struct ast * ast_new_audioformat(int type, const char *value)
{
const char *tags[] = { "samplerate", "bits_per_sample", "channels", NULL, };
const char **tagptr = tags;
char *value_copy = strdup(value);
char *saveptr;
char *token;
struct ast *a;
token = strtok_r(value_copy, ":", &saveptr);
a = ast_new_strnode(type, "samplerate", value_copy);
if (!token)
goto end; // If there was no ':' we assume we just have a samplerate
for (tagptr++, token = strtok_r(NULL, ":", &saveptr); *tagptr && token; tagptr++, token = strtok_r(NULL, ":", &saveptr))
a = ast_new(MPD_T_AND, a, ast_new_strnode(type, *tagptr, token));
end:
free(value_copy);
// In case the expression will be negated we group the result
return ast_new(MPD_T_PARENS, a, NULL);
}
}
%union {
unsigned int ival;
char *str;
struct ast *ast;
}
/* mpd commands */
%token MPD_T_CMDSEARCH
%token MPD_T_CMDSEARCHADD
%token MPD_T_CMDFIND
%token MPD_T_CMDFINDADD
%token MPD_T_CMDCOUNT
%token MPD_T_CMDSEARCHCOUNT
%token MPD_T_CMDPLAYLISTFIND
%token MPD_T_CMDPLAYLISTSEARCH
%token MPD_T_CMDLIST
%token MPD_T_SORT
%token MPD_T_WINDOW
%token MPD_T_POSITION
%token MPD_T_GROUP
/* A string that was quoted. Quotes were stripped by lexer. */
%token <str> MPD_T_STRING
/* Numbers (integers) */
%token <ival> MPD_T_NUM
/* Since time is a quoted string the lexer will just use a MPD_T_STRING token.
The parser will recognize it as time based on the keyword and use MPD_T_TIME
for the AST tree. */
%token MPD_T_TIME
/* The semantic value holds the actual name of the field */
%token <str> MPD_T_STRTAG
%token <str> MPD_T_INTTAG
%token <str> MPD_T_SINCETAG
%token <str> MPD_T_BASETAG
%token MPD_T_ANYTAG
%token MPD_T_AUDIOFORMATTAG
%token MPD_T_PARENS
%token MPD_T_OR
%token MPD_T_AND
%token MPD_T_NOT
/* The below are only ival so we can set intbool, datebool and strbool via the
default rule for semantic values, i.e. $$ = $1. The semantic value (ival) is
set to the token value by the lexer. */
%token <ival> MPD_T_CONTAINS
%token <ival> MPD_T_STARTSWITH
%token <ival> MPD_T_ENDSWITH
%token <ival> MPD_T_EQUAL
%token <ival> MPD_T_NOTEQUAL
%token <ival> MPD_T_LESS
%token <ival> MPD_T_LESSEQUAL
%token <ival> MPD_T_GREATER
%token <ival> MPD_T_GREATEREQUAL
%left MPD_T_OR
%left MPD_T_AND
%left MPD_T_NOT
%type <str> tagtype
%type <ast> filter
%type <ast> sort
%type <ast> window
%type <str> position
%type <ast> group
%type <ast> groups
%type <ast> predicate
%type <ival> strbool
%type <ival> intbool
%%
/*
* Version 0.24:
* Type cmd_fsw (filter sort window):
* playlistfind {FILTER} [sort {TYPE}] [window {START:END}]
* playlistsearch {FILTER} [sort {TYPE}] [window {START:END}]
* find {FILTER} [sort {TYPE}] [window {START:END}]
* search {FILTER} [sort {TYPE}] [window {START:END}]
* Type fswp (filter sort window position):
* findadd {FILTER} [sort {TYPE}] [window {START:END}] [position POS]
* searchadd {FILTER} [sort {TYPE}] [window {START:END}] [position POS]
* Type fg (filter group):
* count {FILTER} [group {GROUPTYPE} group {GROUPTYPE} ...]
* searchcount {FILTER} [group {GROUPTYPE} group {GROUPTYPE} ...]
* Type tfg (type filter group):
* list {TYPE} {FILTER} [group {GROUPTYPE} group {GROUPTYPE} ...]
* Not implemented:
* searchaddpl {NAME} {FILTER} [sort {TYPE}] [window {START:END}] [position POS]
* searchplaylist {NAME} {FILTER} [{START:END}]
* case sensitivity for find
*/
command: cmd_fsw filter sort window { if (result_set(result, NULL, $2, $3, $4, NULL, NULL) < 0) YYABORT; }
| cmd_fswp filter sort window position { if (result_set(result, NULL, $2, $3, $4, $5, NULL) < 0) YYABORT; }
| cmd_fg filter groups { if (result_set(result, NULL, $2, NULL, NULL, NULL, $3) < 0) YYABORT; }
| cmd_fg groups { if (result_set(result, NULL, NULL, NULL, NULL, NULL, $2) < 0) YYABORT; }
| cmd_tfg tagtype filter groups { if (result_set(result, $2, $3, NULL, NULL, NULL, $4) < 0) YYABORT; }
| cmd_tfg tagtype groups { if (result_set(result, $2, NULL, NULL, NULL, NULL, $3) < 0) YYABORT; }
;
tagtype: MPD_T_STRTAG { if (asprintf(&($$), "%s", $1) < 0) YYABORT; }
;
filter: filter MPD_T_AND filter { $$ = ast_new(MPD_T_AND, $1, $3); }
| filter MPD_T_OR filter { $$ = ast_new(MPD_T_OR, $1, $3); }
| '(' filter ')' { $$ = ast_new(MPD_T_PARENS, $2, NULL); }
| MPD_T_NOT filter { struct ast *a = $2; a->type |= INVERT_MASK; $$ = $2; }
| predicate
;
sort: MPD_T_SORT MPD_T_STRTAG { $$ = ast_data(MPD_T_SORT, $2); }
| MPD_T_SORT '-' MPD_T_STRTAG { $$ = ast_data(MPD_T_SORT | INVERT_MASK, $3); }
| %empty { $$ = NULL; }
;
window: MPD_T_WINDOW MPD_T_NUM ':' MPD_T_NUM { $$ = ast_new(MPD_T_WINDOW, ast_int(MPD_T_NUM, $2), ast_int(MPD_T_NUM, $4)); }
| %empty { $$ = NULL; }
;
position: MPD_T_POSITION MPD_T_NUM { if (asprintf(&($$), "%d", $2) < 0) YYABORT; }
| MPD_T_POSITION '+' MPD_T_NUM { if (asprintf(&($$), "+%d", $3) < 0) YYABORT; }
| MPD_T_POSITION '-' MPD_T_NUM { if (asprintf(&($$), "-%d", $3) < 0) YYABORT; }
| %empty { $$ = NULL; }
;
groups: groups group { $$ = $1 ? ast_new(MPD_T_GROUP, $2, $1) : $2; }
| %empty { $$ = NULL; }
;
group: MPD_T_GROUP MPD_T_STRTAG { $$ = ast_data(MPD_T_STRTAG, $2); }
| MPD_T_GROUP MPD_T_INTTAG { $$ = ast_data(MPD_T_INTTAG, $2); }
;
// We accept inttags with numeric and string values, so both date == 2007 and date == '2007'
predicate: '(' MPD_T_STRTAG strbool MPD_T_STRING ')' { $$ = ast_new($3, ast_data(MPD_T_STRTAG, $2), ast_data(MPD_T_STRING, $4)); }
| '(' MPD_T_INTTAG strbool MPD_T_STRING ')' { $$ = ast_new($3, ast_data(MPD_T_STRTAG, $2), ast_data(MPD_T_STRING, $4)); }
| '(' MPD_T_INTTAG intbool MPD_T_NUM ')' { $$ = ast_new($3, ast_data(MPD_T_INTTAG, $2), ast_int(MPD_T_NUM, $4)); }
| '(' MPD_T_ANYTAG strbool MPD_T_STRING ')' { $$ = ast_new_any($3, $4); }
| '(' MPD_T_AUDIOFORMATTAG strbool MPD_T_STRING ')' { $$ = ast_new_audioformat($3, $4); }
| '(' MPD_T_SINCETAG MPD_T_STRING ')' { $$ = ast_new(MPD_T_GREATEREQUAL, ast_data(MPD_T_STRTAG, $2), ast_data(MPD_T_TIME, $3)); }
| '(' MPD_T_BASETAG MPD_T_STRING ')' { $$ = ast_new(MPD_T_STARTSWITH, ast_data(MPD_T_STRTAG, $2), ast_data(MPD_T_STRING, $3)); }
;
strbool: MPD_T_EQUAL
| MPD_T_NOTEQUAL
| MPD_T_CONTAINS
| MPD_T_STARTSWITH
| MPD_T_ENDSWITH
;
intbool: MPD_T_LESS
| MPD_T_LESSEQUAL
| MPD_T_GREATER
| MPD_T_GREATEREQUAL
;
cmd_fsw: MPD_T_CMDPLAYLISTFIND
| MPD_T_CMDPLAYLISTSEARCH
| MPD_T_CMDSEARCH
| MPD_T_CMDFIND
;
cmd_fswp: MPD_T_CMDFINDADD
| MPD_T_CMDSEARCHADD
;
cmd_fg: MPD_T_CMDCOUNT
| MPD_T_CMDSEARCHCOUNT
;
cmd_tfg: MPD_T_CMDLIST
;
%%