mirror of
https://github.com/CTCaer/RetroArch.git
synced 2025-01-23 03:25:27 +00:00
1026 lines
24 KiB
C
1026 lines
24 KiB
C
/* Copyright (C) 2010-2017 The RetroArch team
|
|
*
|
|
* ---------------------------------------------------------------------------------------
|
|
* The following license statement only applies to this file (query.c).
|
|
* ---------------------------------------------------------------------------------------
|
|
*
|
|
* Permission is hereby granted, free of charge,
|
|
* to any person obtaining a copy of this software and associated documentation files (the "Software"),
|
|
* to deal in the Software without restriction, including without limitation the rights to
|
|
* use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software,
|
|
* and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED,
|
|
* INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
|
* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
|
* WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
*/
|
|
|
|
#ifdef _WIN32
|
|
#include <direct.h>
|
|
#else
|
|
#include <unistd.h>
|
|
#endif
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <ctype.h>
|
|
#include <string.h>
|
|
|
|
#include <compat/fnmatch.h>
|
|
#include <compat/strl.h>
|
|
|
|
#include "libretrodb.h"
|
|
#include "query.h"
|
|
#include "rmsgpack_dom.h"
|
|
|
|
#define MAX_ERROR_LEN 256
|
|
#define QUERY_MAX_ARGS 50
|
|
|
|
struct buffer
|
|
{
|
|
const char *data;
|
|
size_t len;
|
|
ssize_t offset;
|
|
};
|
|
|
|
enum argument_type
|
|
{
|
|
AT_FUNCTION,
|
|
AT_VALUE
|
|
};
|
|
|
|
struct argument;
|
|
|
|
typedef struct rmsgpack_dom_value (*rarch_query_func)(
|
|
struct rmsgpack_dom_value input,
|
|
unsigned argc,
|
|
const struct argument *argv
|
|
);
|
|
|
|
struct invocation
|
|
{
|
|
rarch_query_func func;
|
|
unsigned argc;
|
|
struct argument *argv;
|
|
};
|
|
|
|
struct argument
|
|
{
|
|
enum argument_type type;
|
|
union
|
|
{
|
|
struct rmsgpack_dom_value value;
|
|
struct invocation invocation;
|
|
} a;
|
|
};
|
|
|
|
struct query
|
|
{
|
|
unsigned ref_count;
|
|
struct invocation root;
|
|
};
|
|
|
|
struct registered_func
|
|
{
|
|
const char *name;
|
|
rarch_query_func func;
|
|
};
|
|
|
|
static char tmp_error_buff [MAX_ERROR_LEN] = {0};
|
|
|
|
/* Forward declarations */
|
|
static struct buffer query_parse_method_call(struct buffer buff,
|
|
struct invocation *invocation, const char **error);
|
|
static struct buffer query_parse_table(struct buffer buff,
|
|
struct invocation *invocation, const char **error);
|
|
|
|
/* Errors */
|
|
static void query_raise_too_many_arguments(const char **error)
|
|
{
|
|
strlcpy(tmp_error_buff,
|
|
"Too many arguments in function call.", sizeof(tmp_error_buff));
|
|
*error = tmp_error_buff;
|
|
}
|
|
|
|
static struct rmsgpack_dom_value query_func_is_true(
|
|
struct rmsgpack_dom_value input,
|
|
unsigned argc, const struct argument *argv)
|
|
{
|
|
struct rmsgpack_dom_value res;
|
|
|
|
res.type = RDT_BOOL;
|
|
res.val.bool_ = 0;
|
|
|
|
if (argc > 0 || input.type != RDT_BOOL)
|
|
res.val.bool_ = 0;
|
|
else
|
|
res.val.bool_ = input.val.bool_;
|
|
|
|
return res;
|
|
}
|
|
|
|
static struct rmsgpack_dom_value func_equals(
|
|
struct rmsgpack_dom_value input,
|
|
unsigned argc, const struct argument * argv)
|
|
{
|
|
struct argument arg;
|
|
struct rmsgpack_dom_value res;
|
|
|
|
res.type = RDT_BOOL;
|
|
res.val.bool_ = 0;
|
|
|
|
if (argc == 1)
|
|
{
|
|
arg = argv[0];
|
|
|
|
if (arg.type != AT_VALUE)
|
|
res.val.bool_ = 0;
|
|
else
|
|
{
|
|
if (input.type == RDT_UINT && arg.a.value.type == RDT_INT)
|
|
{
|
|
arg.a.value.type = RDT_UINT;
|
|
arg.a.value.val.uint_ = arg.a.value.val.int_;
|
|
}
|
|
res.val.bool_ = (rmsgpack_dom_value_cmp(&input, &arg.a.value) == 0);
|
|
}
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
static struct rmsgpack_dom_value query_func_operator_or(
|
|
struct rmsgpack_dom_value input,
|
|
unsigned argc, const struct argument * argv)
|
|
{
|
|
unsigned i;
|
|
struct rmsgpack_dom_value res;
|
|
|
|
res.type = RDT_BOOL;
|
|
res.val.bool_ = 0;
|
|
|
|
for (i = 0; i < argc; i++)
|
|
{
|
|
if (argv[i].type == AT_VALUE)
|
|
res = func_equals(input, 1, &argv[i]);
|
|
else
|
|
{
|
|
res = query_func_is_true(
|
|
argv[i].a.invocation.func(input,
|
|
argv[i].a.invocation.argc,
|
|
argv[i].a.invocation.argv
|
|
), 0, NULL);
|
|
}
|
|
|
|
if (res.val.bool_)
|
|
return res;
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
static struct rmsgpack_dom_value query_func_operator_and(
|
|
struct rmsgpack_dom_value input,
|
|
unsigned argc, const struct argument * argv)
|
|
{
|
|
unsigned i;
|
|
struct rmsgpack_dom_value res;
|
|
|
|
res.type = RDT_BOOL;
|
|
res.val.bool_ = 0;
|
|
|
|
for (i = 0; i < argc; i++)
|
|
{
|
|
if (argv[i].type == AT_VALUE)
|
|
res = func_equals(input, 1, &argv[i]);
|
|
else
|
|
{
|
|
res = query_func_is_true(
|
|
argv[i].a.invocation.func(input,
|
|
argv[i].a.invocation.argc,
|
|
argv[i].a.invocation.argv
|
|
),
|
|
0, NULL);
|
|
}
|
|
|
|
if (!res.val.bool_)
|
|
return res;
|
|
}
|
|
return res;
|
|
}
|
|
|
|
static struct rmsgpack_dom_value query_func_between(
|
|
struct rmsgpack_dom_value input,
|
|
unsigned argc, const struct argument * argv)
|
|
{
|
|
struct rmsgpack_dom_value res;
|
|
unsigned i = 0;
|
|
|
|
res.type = RDT_BOOL;
|
|
res.val.bool_ = 0;
|
|
|
|
(void)i;
|
|
|
|
if (argc != 2)
|
|
return res;
|
|
if (argv[0].type != AT_VALUE || argv[1].type != AT_VALUE)
|
|
return res;
|
|
if (argv[0].a.value.type != RDT_INT || argv[1].a.value.type != RDT_INT)
|
|
return res;
|
|
|
|
switch (input.type)
|
|
{
|
|
case RDT_INT:
|
|
res.val.bool_ = (
|
|
(input.val.int_ >= argv[0].a.value.val.int_)
|
|
&& (input.val.int_ <= argv[1].a.value.val.int_));
|
|
break;
|
|
case RDT_UINT:
|
|
res.val.bool_ = (
|
|
((unsigned)input.val.int_ >= argv[0].a.value.val.uint_)
|
|
&& (input.val.int_ <= argv[1].a.value.val.int_));
|
|
break;
|
|
default:
|
|
return res;
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
static struct rmsgpack_dom_value query_func_glob(
|
|
struct rmsgpack_dom_value input,
|
|
unsigned argc, const struct argument * argv)
|
|
{
|
|
struct rmsgpack_dom_value res;
|
|
unsigned i = 0;
|
|
|
|
res.type = RDT_BOOL;
|
|
res.val.bool_ = 0;
|
|
|
|
(void)i;
|
|
|
|
if (argc != 1)
|
|
return res;
|
|
if (argv[0].type != AT_VALUE || argv[0].a.value.type != RDT_STRING)
|
|
return res;
|
|
if (input.type != RDT_STRING)
|
|
return res;
|
|
res.val.bool_ = rl_fnmatch(
|
|
argv[0].a.value.val.string.buff,
|
|
input.val.string.buff,
|
|
0
|
|
) == 0;
|
|
return res;
|
|
}
|
|
|
|
struct registered_func registered_functions[100] = {
|
|
{"is_true", query_func_is_true},
|
|
{"or", query_func_operator_or},
|
|
{"and", query_func_operator_and},
|
|
{"between", query_func_between},
|
|
{"glob", query_func_glob},
|
|
{NULL, NULL}
|
|
};
|
|
|
|
static void query_raise_expected_number(ssize_t where, const char **error)
|
|
{
|
|
#ifdef _WIN32
|
|
snprintf(tmp_error_buff, MAX_ERROR_LEN,
|
|
"%I64u::Expected number",
|
|
(unsigned long long)where);
|
|
#else
|
|
snprintf(tmp_error_buff, MAX_ERROR_LEN,
|
|
"%llu::Expected number",
|
|
(unsigned long long)where);
|
|
#endif
|
|
*error = tmp_error_buff;
|
|
}
|
|
|
|
static void query_raise_expected_string(ssize_t where, const char ** error)
|
|
{
|
|
#ifdef _WIN32
|
|
snprintf(tmp_error_buff, MAX_ERROR_LEN,
|
|
"%I64u::Expected string",
|
|
(unsigned long long)where);
|
|
#else
|
|
snprintf(tmp_error_buff, MAX_ERROR_LEN,
|
|
"%llu::Expected string",
|
|
(unsigned long long)where);
|
|
#endif
|
|
*error = tmp_error_buff;
|
|
}
|
|
|
|
static void query_raise_unexpected_eof(ssize_t where, const char ** error)
|
|
{
|
|
#ifdef _WIN32
|
|
snprintf(tmp_error_buff, MAX_ERROR_LEN,
|
|
"%I64u::Unexpected EOF",
|
|
(unsigned long long)where
|
|
);
|
|
#else
|
|
snprintf(tmp_error_buff, MAX_ERROR_LEN,
|
|
"%llu::Unexpected EOF",
|
|
(unsigned long long)where
|
|
);
|
|
#endif
|
|
*error = tmp_error_buff;
|
|
}
|
|
|
|
static void query_raise_enomem(const char **error)
|
|
{
|
|
strlcpy(tmp_error_buff, "Out of memory", sizeof(tmp_error_buff));
|
|
*error = tmp_error_buff;
|
|
}
|
|
|
|
static void query_raise_unknown_function(ssize_t where, const char *name,
|
|
ssize_t len, const char **error)
|
|
{
|
|
#ifdef _WIN32
|
|
int n = snprintf(tmp_error_buff, MAX_ERROR_LEN,
|
|
"%I64u::Unknown function '",
|
|
(unsigned long long)where
|
|
);
|
|
#else
|
|
int n = snprintf(tmp_error_buff, MAX_ERROR_LEN,
|
|
"%llu::Unknown function '",
|
|
(unsigned long long)where
|
|
);
|
|
#endif
|
|
|
|
if (len < (MAX_ERROR_LEN - n - 3))
|
|
strncpy(tmp_error_buff + n, name, len);
|
|
|
|
strlcpy(tmp_error_buff + n + len, "'", sizeof(tmp_error_buff));
|
|
*error = tmp_error_buff;
|
|
}
|
|
|
|
static void query_raise_expected_eof(
|
|
ssize_t where, char found, const char **error)
|
|
{
|
|
#ifdef _WIN32
|
|
snprintf(tmp_error_buff, MAX_ERROR_LEN,
|
|
"%I64u::Expected EOF found '%c'",
|
|
(unsigned long long)where,
|
|
found
|
|
);
|
|
#else
|
|
snprintf(tmp_error_buff, MAX_ERROR_LEN,
|
|
"%llu::Expected EOF found '%c'",
|
|
(unsigned long long)where,
|
|
found
|
|
);
|
|
#endif
|
|
*error = tmp_error_buff;
|
|
}
|
|
|
|
static void query_raise_unexpected_char(
|
|
ssize_t where, char expected, char found,
|
|
const char **error)
|
|
{
|
|
#ifdef _WIN32
|
|
snprintf(tmp_error_buff, MAX_ERROR_LEN,
|
|
"%I64u::Expected '%c' found '%c'",
|
|
(unsigned long long)where, expected, found);
|
|
#else
|
|
snprintf(tmp_error_buff, MAX_ERROR_LEN,
|
|
"%llu::Expected '%c' found '%c'",
|
|
(unsigned long long)where, expected, found);
|
|
#endif
|
|
*error = tmp_error_buff;
|
|
}
|
|
|
|
|
|
static void query_argument_free(struct argument *arg)
|
|
{
|
|
unsigned i;
|
|
|
|
if (arg->type != AT_FUNCTION)
|
|
{
|
|
rmsgpack_dom_value_free(&arg->a.value);
|
|
return;
|
|
}
|
|
|
|
for (i = 0; i < arg->a.invocation.argc; i++)
|
|
query_argument_free(&arg->a.invocation.argv[i]);
|
|
|
|
free((void*)arg->a.invocation.argv);
|
|
}
|
|
|
|
static struct buffer query_parse_integer(struct buffer buff,
|
|
struct rmsgpack_dom_value *value, const char **error)
|
|
{
|
|
bool test = false;
|
|
|
|
value->type = RDT_INT;
|
|
|
|
#ifdef _WIN32
|
|
test = (sscanf(buff.data + buff.offset,
|
|
"%I64d",
|
|
(signed long long*)&value->val.int_) == 0);
|
|
#else
|
|
test = (sscanf(buff.data + buff.offset,
|
|
"%lld",
|
|
(signed long long*)&value->val.int_) == 0);
|
|
#endif
|
|
|
|
if (test)
|
|
query_raise_expected_number(buff.offset, error);
|
|
else
|
|
{
|
|
while (isdigit((int)buff.data[buff.offset]))
|
|
buff.offset++;
|
|
}
|
|
|
|
return buff;
|
|
}
|
|
|
|
static struct buffer query_chomp(struct buffer buff)
|
|
{
|
|
for (; (unsigned)buff.offset < buff.len
|
|
&& isspace((int)buff.data[buff.offset]); buff.offset++);
|
|
return buff;
|
|
}
|
|
|
|
|
|
static struct buffer query_expect_eof(struct buffer buff, const char ** error)
|
|
{
|
|
buff = query_chomp(buff);
|
|
if ((unsigned)buff.offset < buff.len)
|
|
query_raise_expected_eof(buff.offset, buff.data[buff.offset], error);
|
|
return buff;
|
|
}
|
|
|
|
static int query_peek(struct buffer buff, const char * data)
|
|
{
|
|
size_t remain = buff.len - buff.offset;
|
|
|
|
if (remain < strlen(data))
|
|
return 0;
|
|
|
|
return (strncmp(buff.data + buff.offset,
|
|
data, strlen(data)) == 0);
|
|
}
|
|
|
|
static int query_is_eot(struct buffer buff)
|
|
{
|
|
return ((unsigned)buff.offset >= buff.len);
|
|
}
|
|
|
|
static struct buffer query_get_char(
|
|
struct buffer buff, char * c,
|
|
const char ** error)
|
|
{
|
|
if (query_is_eot(buff))
|
|
{
|
|
query_raise_unexpected_eof(buff.offset, error);
|
|
return buff;
|
|
}
|
|
|
|
*c = buff.data[buff.offset];
|
|
buff.offset++;
|
|
return buff;
|
|
}
|
|
|
|
static struct buffer query_parse_string(struct buffer buff,
|
|
struct rmsgpack_dom_value *value, const char **error)
|
|
{
|
|
const char * str_start = NULL;
|
|
char terminator = '\0';
|
|
char c = '\0';
|
|
int is_binstr = 0;
|
|
|
|
(void)c;
|
|
|
|
buff = query_get_char(buff, &terminator, error);
|
|
|
|
if (*error)
|
|
return buff;
|
|
|
|
if (terminator == 'b')
|
|
{
|
|
is_binstr = 1;
|
|
buff = query_get_char(buff, &terminator, error);
|
|
}
|
|
|
|
if (terminator != '"' && terminator != '\'')
|
|
{
|
|
buff.offset--;
|
|
query_raise_expected_string(buff.offset, error);
|
|
}
|
|
|
|
str_start = buff.data + buff.offset;
|
|
buff = query_get_char(buff, &c, error);
|
|
|
|
while (!*error)
|
|
{
|
|
if (c == terminator)
|
|
break;
|
|
buff = query_get_char(buff, &c, error);
|
|
}
|
|
|
|
if (!*error)
|
|
{
|
|
size_t count;
|
|
value->type = is_binstr ? RDT_BINARY : RDT_STRING;
|
|
value->val.string.len = (uint32_t)((buff.data + buff.offset) - str_start - 1);
|
|
|
|
count = is_binstr ? (value->val.string.len + 1) / 2
|
|
: (value->val.string.len + 1);
|
|
value->val.string.buff = (char*)calloc(count, sizeof(char));
|
|
|
|
if (!value->val.string.buff)
|
|
query_raise_enomem(error);
|
|
else if (is_binstr)
|
|
{
|
|
unsigned i;
|
|
const char *tok = str_start;
|
|
unsigned j = 0;
|
|
|
|
for (i = 0; i < value->val.string.len; i += 2)
|
|
{
|
|
uint8_t hi, lo;
|
|
char hic = tok[i];
|
|
char loc = tok[i + 1];
|
|
|
|
if (hic <= '9')
|
|
hi = hic - '0';
|
|
else
|
|
hi = (hic - 'A') + 10;
|
|
|
|
if (loc <= '9')
|
|
lo = loc - '0';
|
|
else
|
|
lo = (loc - 'A') + 10;
|
|
|
|
value->val.string.buff[j++] = hi * 16 + lo;
|
|
}
|
|
|
|
value->val.string.len = j;
|
|
}
|
|
else
|
|
memcpy(value->val.string.buff, str_start, value->val.string.len);
|
|
}
|
|
return buff;
|
|
}
|
|
|
|
static struct buffer query_parse_value(struct buffer buff,
|
|
struct rmsgpack_dom_value *value, const char **error)
|
|
{
|
|
buff = query_chomp(buff);
|
|
|
|
if (query_peek(buff, "nil"))
|
|
{
|
|
buff.offset += strlen("nil");
|
|
value->type = RDT_NULL;
|
|
}
|
|
else if (query_peek(buff, "true"))
|
|
{
|
|
buff.offset += strlen("true");
|
|
value->type = RDT_BOOL;
|
|
value->val.bool_ = 1;
|
|
}
|
|
else if (query_peek(buff, "false"))
|
|
{
|
|
buff.offset += strlen("false");
|
|
value->type = RDT_BOOL;
|
|
value->val.bool_ = 0;
|
|
}
|
|
else if (query_peek(buff, "b") || query_peek(buff, "\"") || query_peek(buff, "'"))
|
|
buff = query_parse_string(buff, value, error);
|
|
else if (isdigit((int)buff.data[buff.offset]))
|
|
buff = query_parse_integer(buff, value, error);
|
|
return buff;
|
|
}
|
|
|
|
static void query_peek_char(struct buffer buff, char *c,
|
|
const char **error)
|
|
{
|
|
if (query_is_eot(buff))
|
|
{
|
|
query_raise_unexpected_eof(buff.offset, error);
|
|
return;
|
|
}
|
|
|
|
*c = buff.data[buff.offset];
|
|
}
|
|
|
|
static struct buffer query_get_ident(struct buffer buff,
|
|
const char **ident,
|
|
size_t *len, const char **error)
|
|
{
|
|
char c = '\0';
|
|
|
|
if (query_is_eot(buff))
|
|
{
|
|
query_raise_unexpected_eof(buff.offset, error);
|
|
return buff;
|
|
}
|
|
|
|
*ident = buff.data + buff.offset;
|
|
*len = 0;
|
|
query_peek_char(buff, &c, error);
|
|
|
|
if (*error)
|
|
goto clean;
|
|
if (!isalpha((int)c))
|
|
return buff;
|
|
|
|
buff.offset++;
|
|
*len = *len + 1;
|
|
query_peek_char(buff, &c, error);
|
|
|
|
while (!*error)
|
|
{
|
|
if (!(isalpha((int)c) || isdigit((int)c) || c == '_'))
|
|
break;
|
|
buff.offset++;
|
|
*len = *len + 1;
|
|
query_peek_char(buff, &c, error);
|
|
}
|
|
|
|
clean:
|
|
return buff;
|
|
}
|
|
|
|
static struct buffer query_expect_char(struct buffer buff,
|
|
char c, const char ** error)
|
|
{
|
|
if ((unsigned)buff.offset >= buff.len)
|
|
query_raise_unexpected_eof(buff.offset, error);
|
|
else if (buff.data[buff.offset] != c)
|
|
query_raise_unexpected_char(
|
|
buff.offset, c, buff.data[buff.offset], error);
|
|
else
|
|
buff.offset++;
|
|
return buff;
|
|
}
|
|
|
|
|
|
static struct buffer query_parse_argument(struct buffer buff,
|
|
struct argument *arg, const char **error)
|
|
{
|
|
buff = query_chomp(buff);
|
|
|
|
if (
|
|
isalpha((int)buff.data[buff.offset])
|
|
&& !(
|
|
query_peek(buff, "nil")
|
|
|| query_peek(buff, "true")
|
|
|| query_peek(buff, "false")
|
|
|| query_peek(buff, "b\"")
|
|
|| query_peek(buff, "b'") /* bin string prefix*/
|
|
)
|
|
)
|
|
{
|
|
arg->type = AT_FUNCTION;
|
|
buff = query_parse_method_call(buff,
|
|
&arg->a.invocation, error);
|
|
}
|
|
else if (query_peek(buff, "{"))
|
|
{
|
|
arg->type = AT_FUNCTION;
|
|
buff = query_parse_table(buff, &arg->a.invocation, error);
|
|
}
|
|
else
|
|
{
|
|
arg->type = AT_VALUE;
|
|
buff = query_parse_value(buff, &arg->a.value, error);
|
|
}
|
|
return buff;
|
|
}
|
|
|
|
static struct buffer query_parse_method_call(struct buffer buff,
|
|
struct invocation *invocation, const char **error)
|
|
{
|
|
size_t func_name_len;
|
|
unsigned i;
|
|
struct argument args[QUERY_MAX_ARGS];
|
|
unsigned argi = 0;
|
|
const char *func_name = NULL;
|
|
struct registered_func *rf = registered_functions;
|
|
|
|
invocation->func = NULL;
|
|
|
|
buff = query_get_ident(buff, &func_name, &func_name_len, error);
|
|
if (*error)
|
|
goto clean;
|
|
|
|
buff = query_chomp(buff);
|
|
buff = query_expect_char(buff, '(', error);
|
|
if (*error)
|
|
goto clean;
|
|
|
|
while (rf->name)
|
|
{
|
|
if (strncmp(rf->name, func_name, func_name_len) == 0)
|
|
{
|
|
invocation->func = rf->func;
|
|
break;
|
|
}
|
|
rf++;
|
|
}
|
|
|
|
if (!invocation->func)
|
|
{
|
|
query_raise_unknown_function(buff.offset, func_name,
|
|
func_name_len, error);
|
|
goto clean;
|
|
}
|
|
|
|
buff = query_chomp(buff);
|
|
while (!query_peek(buff, ")"))
|
|
{
|
|
if (argi >= QUERY_MAX_ARGS)
|
|
{
|
|
query_raise_too_many_arguments(error);
|
|
goto clean;
|
|
}
|
|
|
|
buff = query_parse_argument(buff, &args[argi], error);
|
|
|
|
if (*error)
|
|
goto clean;
|
|
|
|
argi++;
|
|
buff = query_chomp(buff);
|
|
buff = query_expect_char(buff, ',', error);
|
|
|
|
if (*error)
|
|
{
|
|
*error = NULL;
|
|
break;
|
|
}
|
|
buff = query_chomp(buff);
|
|
}
|
|
buff = query_expect_char(buff, ')', error);
|
|
|
|
if (*error)
|
|
goto clean;
|
|
|
|
invocation->argc = argi;
|
|
invocation->argv = (struct argument*)
|
|
malloc(sizeof(struct argument) * argi);
|
|
|
|
if (!invocation->argv)
|
|
{
|
|
query_raise_enomem(error);
|
|
goto clean;
|
|
}
|
|
memcpy(invocation->argv, args,
|
|
sizeof(struct argument) * argi);
|
|
|
|
goto success;
|
|
clean:
|
|
for (i = 0; i < argi; i++)
|
|
query_argument_free(&args[i]);
|
|
success:
|
|
return buff;
|
|
}
|
|
|
|
static struct rmsgpack_dom_value query_func_all_map(
|
|
struct rmsgpack_dom_value input,
|
|
unsigned argc, const struct argument *argv)
|
|
{
|
|
unsigned i;
|
|
struct argument arg;
|
|
struct rmsgpack_dom_value res;
|
|
struct rmsgpack_dom_value nil_value;
|
|
struct rmsgpack_dom_value *value = NULL;
|
|
|
|
res.type = RDT_BOOL;
|
|
res.val.bool_ = 1;
|
|
|
|
nil_value.type = RDT_NULL;
|
|
|
|
if (argc % 2 != 0)
|
|
{
|
|
res.val.bool_ = 0;
|
|
return res;
|
|
}
|
|
|
|
if (input.type != RDT_MAP)
|
|
return res;
|
|
|
|
for (i = 0; i < argc; i += 2)
|
|
{
|
|
arg = argv[i];
|
|
if (arg.type != AT_VALUE)
|
|
{
|
|
res.val.bool_ = 0;
|
|
goto clean;
|
|
}
|
|
value = rmsgpack_dom_value_map_value(&input, &arg.a.value);
|
|
if (!value) /* All missing fields are nil */
|
|
value = &nil_value;
|
|
arg = argv[i + 1];
|
|
if (arg.type == AT_VALUE)
|
|
res = func_equals(*value, 1, &arg);
|
|
else
|
|
{
|
|
res = query_func_is_true(arg.a.invocation.func(
|
|
*value,
|
|
arg.a.invocation.argc,
|
|
arg.a.invocation.argv
|
|
), 0, NULL);
|
|
value = NULL;
|
|
}
|
|
if (!res.val.bool_)
|
|
break;
|
|
}
|
|
clean:
|
|
return res;
|
|
}
|
|
|
|
static struct buffer query_parse_table(struct buffer buff,
|
|
struct invocation *invocation, const char **error)
|
|
{
|
|
unsigned i;
|
|
size_t ident_len;
|
|
struct argument args[QUERY_MAX_ARGS];
|
|
const char *ident_name = NULL;
|
|
unsigned argi = 0;
|
|
|
|
buff = query_chomp(buff);
|
|
buff = query_expect_char(buff, '{', error);
|
|
|
|
if (*error)
|
|
goto clean;
|
|
|
|
buff = query_chomp(buff);
|
|
|
|
while (!query_peek(buff, "}"))
|
|
{
|
|
if (argi >= QUERY_MAX_ARGS)
|
|
{
|
|
query_raise_too_many_arguments(error);
|
|
goto clean;
|
|
}
|
|
|
|
if (isalpha((int)buff.data[buff.offset]))
|
|
{
|
|
buff = query_get_ident(buff, &ident_name, &ident_len, error);
|
|
|
|
if (!*error)
|
|
{
|
|
args[argi].a.value.type = RDT_STRING;
|
|
args[argi].a.value.val.string.len = (uint32_t)ident_len;
|
|
args[argi].a.value.val.string.buff = (char*)calloc(
|
|
ident_len + 1,
|
|
sizeof(char)
|
|
);
|
|
|
|
if (!args[argi].a.value.val.string.buff)
|
|
goto clean;
|
|
|
|
strncpy(
|
|
args[argi].a.value.val.string.buff,
|
|
ident_name,
|
|
ident_len
|
|
);
|
|
}
|
|
}
|
|
else
|
|
buff = query_parse_string(buff, &args[argi].a.value, error);
|
|
|
|
if (*error)
|
|
goto clean;
|
|
|
|
args[argi].type = AT_VALUE;
|
|
buff = query_chomp(buff);
|
|
argi++;
|
|
buff = query_expect_char(buff, ':', error);
|
|
|
|
if (*error)
|
|
goto clean;
|
|
|
|
buff = query_chomp(buff);
|
|
|
|
if (argi >= QUERY_MAX_ARGS)
|
|
{
|
|
query_raise_too_many_arguments(error);
|
|
goto clean;
|
|
}
|
|
|
|
buff = query_parse_argument(buff, &args[argi], error);
|
|
|
|
if (*error)
|
|
goto clean;
|
|
argi++;
|
|
buff = query_chomp(buff);
|
|
buff = query_expect_char(buff, ',', error);
|
|
|
|
if (*error)
|
|
{
|
|
*error = NULL;
|
|
break;
|
|
}
|
|
buff = query_chomp(buff);
|
|
}
|
|
|
|
buff = query_expect_char(buff, '}', error);
|
|
|
|
if (*error)
|
|
goto clean;
|
|
|
|
invocation->func = query_func_all_map;
|
|
invocation->argc = argi;
|
|
invocation->argv = (struct argument*)
|
|
malloc(sizeof(struct argument) * argi);
|
|
|
|
if (!invocation->argv)
|
|
{
|
|
query_raise_enomem(error);
|
|
goto clean;
|
|
}
|
|
memcpy(invocation->argv, args,
|
|
sizeof(struct argument) * argi);
|
|
|
|
goto success;
|
|
clean:
|
|
for (i = 0; i < argi; i++)
|
|
query_argument_free(&args[i]);
|
|
success:
|
|
return buff;
|
|
}
|
|
|
|
void libretrodb_query_free(void *q)
|
|
{
|
|
unsigned i;
|
|
struct query *real_q = (struct query*)q;
|
|
|
|
real_q->ref_count--;
|
|
if (real_q->ref_count > 0)
|
|
return;
|
|
|
|
for (i = 0; i < real_q->root.argc; i++)
|
|
query_argument_free(&real_q->root.argv[i]);
|
|
|
|
free(real_q->root.argv);
|
|
real_q->root.argv = NULL;
|
|
real_q->root.argc = 0;
|
|
free(real_q);
|
|
}
|
|
|
|
void *libretrodb_query_compile(libretrodb_t *db,
|
|
const char *query, size_t buff_len, const char **error_string)
|
|
{
|
|
struct buffer buff;
|
|
struct query *q = (struct query*)calloc(1, sizeof(*q));
|
|
|
|
if (!q)
|
|
goto error;
|
|
|
|
q->ref_count = 1;
|
|
buff.data = query;
|
|
buff.len = buff_len;
|
|
buff.offset = 0;
|
|
*error_string = NULL;
|
|
|
|
buff = query_chomp(buff);
|
|
|
|
if (query_peek(buff, "{"))
|
|
{
|
|
buff = query_parse_table(buff, &q->root, error_string);
|
|
if (*error_string)
|
|
goto error;
|
|
}
|
|
else if (isalpha((int)buff.data[buff.offset]))
|
|
buff = query_parse_method_call(buff, &q->root, error_string);
|
|
|
|
buff = query_expect_eof(buff, error_string);
|
|
if (*error_string)
|
|
goto error;
|
|
|
|
if (!q->root.func)
|
|
{
|
|
query_raise_unexpected_eof(buff.offset, error_string);
|
|
goto error;
|
|
}
|
|
|
|
return q;
|
|
|
|
error:
|
|
if (q)
|
|
libretrodb_query_free(q);
|
|
return NULL;
|
|
}
|
|
|
|
void libretrodb_query_inc_ref(libretrodb_query_t *q)
|
|
{
|
|
struct query *rq = (struct query*)q;
|
|
if (rq)
|
|
rq->ref_count += 1;
|
|
}
|
|
|
|
int libretrodb_query_filter(libretrodb_query_t *q,
|
|
struct rmsgpack_dom_value *v)
|
|
{
|
|
struct invocation inv = ((struct query *)q)->root;
|
|
struct rmsgpack_dom_value res = inv.func(*v, inv.argc, inv.argv);
|
|
return (res.type == RDT_BOOL && res.val.bool_);
|
|
}
|