mariadb/storage/mroonga/vendor/groonga/lib/ctx.c
2019-05-13 17:54:04 +03:00

1873 lines
50 KiB
C

/* -*- c-basic-offset: 2 -*- */
/*
Copyright(C) 2009-2017 Brazil
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License version 2.1 as published by the Free Software Foundation.
This library 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
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with this library; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1335 USA
*/
#include "grn.h"
#include <string.h>
#include "grn_request_canceler.h"
#include "grn_request_timer.h"
#include "grn_tokenizers.h"
#include "grn_ctx_impl.h"
#include "grn_ii.h"
#include "grn_pat.h"
#include "grn_index_column.h"
#include "grn_proc.h"
#include "grn_plugin.h"
#include "grn_snip.h"
#include "grn_output.h"
#include "grn_normalizer.h"
#include "grn_mrb.h"
#include "grn_ctx_impl_mrb.h"
#include "grn_logger.h"
#include "grn_cache.h"
#include "grn_expr.h"
#include <stdio.h>
#include <stdarg.h>
#include <time.h>
#ifdef GRN_WITH_ONIGMO
# define GRN_SUPPORT_REGEXP
#endif /* GRN_WITH_ONIGMO */
#ifdef GRN_SUPPORT_REGEXP
# include <onigmo.h>
#endif /* GRN_SUPPORT_REGEXP */
#ifdef WIN32
# include <share.h>
#else /* WIN32 */
# include <netinet/in.h>
#endif /* WIN32 */
#define GRN_CTX_INITIALIZER(enc) \
{ GRN_SUCCESS, 0, enc, 0, GRN_LOG_NOTICE,\
GRN_CTX_FIN, 0, 0, 0, 0, {0}, NULL, NULL, NULL, NULL, NULL, \
{NULL, NULL,NULL, NULL,NULL, NULL,NULL, NULL,NULL, NULL,NULL, NULL,NULL, NULL,NULL, NULL}, ""}
#define GRN_CTX_CLOSED(ctx) ((ctx)->stat == GRN_CTX_FIN)
grn_ctx grn_gctx = GRN_CTX_INITIALIZER(GRN_ENC_DEFAULT);
int grn_pagesize;
grn_critical_section grn_glock;
uint32_t grn_gtick;
int grn_lock_timeout = GRN_LOCK_TIMEOUT;
#ifdef USE_UYIELD
int grn_uyield_count = 0;
#endif
static grn_bool grn_ctx_per_db = GRN_FALSE;
static void
grn_init_from_env(void)
{
{
char grn_ctx_per_db_env[GRN_ENV_BUFFER_SIZE];
grn_getenv("GRN_CTX_PER_DB",
grn_ctx_per_db_env,
GRN_ENV_BUFFER_SIZE);
if (grn_ctx_per_db_env[0] && strcmp(grn_ctx_per_db_env, "yes") == 0) {
grn_ctx_per_db = GRN_TRUE;
}
}
grn_alloc_init_from_env();
grn_mrb_init_from_env();
grn_ctx_impl_mrb_init_from_env();
grn_io_init_from_env();
grn_ii_init_from_env();
grn_db_init_from_env();
grn_expr_init_from_env();
grn_index_column_init_from_env();
grn_proc_init_from_env();
grn_plugin_init_from_env();
}
static void
grn_init_external_libraries(void)
{
#ifdef GRN_SUPPORT_REGEXP
onig_init();
#endif /* GRN_SUPPORT_REGEXP */
}
static void
grn_fin_external_libraries(void)
{
#ifdef GRN_SUPPORT_REGEXP
onig_end();
#endif /* GRN_SUPPORT_REGEXP */
}
void
grn_sleep(uint32_t seconds)
{
#ifdef WIN32
Sleep(seconds * 1000);
#else // WIN32
sleep(seconds);
#endif // WIN32
}
void
grn_nanosleep(uint64_t nanoseconds)
{
#ifdef WIN32
Sleep((DWORD)(nanoseconds / 1000000));
#else // WIN32
struct timespec interval;
interval.tv_sec = (time_t)(nanoseconds / 1000000000);
interval.tv_nsec = (long)(nanoseconds % 1000000000);
nanosleep(&interval, NULL);
#endif // WIN32
}
const char *
grn_get_global_error_message(void)
{
return grn_gctx.errbuf;
}
static void
grn_loader_init(grn_loader *loader)
{
GRN_TEXT_INIT(&loader->values, 0);
GRN_UINT32_INIT(&loader->level, GRN_OBJ_VECTOR);
GRN_PTR_INIT(&loader->columns, GRN_OBJ_VECTOR, GRN_ID_NIL);
GRN_UINT32_INIT(&loader->ids, GRN_OBJ_VECTOR);
GRN_INT32_INIT(&loader->return_codes, GRN_OBJ_VECTOR);
GRN_TEXT_INIT(&loader->error_messages, GRN_OBJ_VECTOR);
loader->id_offset = -1;
loader->key_offset = -1;
loader->table = NULL;
loader->last = NULL;
loader->ifexists = NULL;
loader->each = NULL;
loader->values_size = 0;
loader->nrecords = 0;
loader->stat = GRN_LOADER_BEGIN;
loader->columns_status = GRN_LOADER_COLUMNS_UNSET;
loader->rc = GRN_SUCCESS;
loader->errbuf[0] = '\0';
loader->output_ids = GRN_FALSE;
loader->output_errors = GRN_FALSE;
}
void
grn_ctx_loader_clear(grn_ctx *ctx)
{
grn_loader *loader = &ctx->impl->loader;
grn_obj *v = (grn_obj *)(GRN_BULK_HEAD(&loader->values));
grn_obj *ve = (grn_obj *)(GRN_BULK_CURR(&loader->values));
grn_obj **p = (grn_obj **)GRN_BULK_HEAD(&loader->columns);
uint32_t i = GRN_BULK_VSIZE(&loader->columns) / sizeof(grn_obj *);
if (ctx->impl->db) { while (i--) { grn_obj_unlink(ctx, *p++); } }
if (loader->ifexists) { grn_obj_unlink(ctx, loader->ifexists); }
if (loader->each) { grn_obj_unlink(ctx, loader->each); }
while (v < ve) { GRN_OBJ_FIN(ctx, v++); }
GRN_OBJ_FIN(ctx, &loader->values);
GRN_OBJ_FIN(ctx, &loader->level);
GRN_OBJ_FIN(ctx, &loader->columns);
GRN_OBJ_FIN(ctx, &loader->ids);
GRN_OBJ_FIN(ctx, &loader->return_codes);
GRN_OBJ_FIN(ctx, &loader->error_messages);
grn_loader_init(loader);
}
#define IMPL_SIZE ((sizeof(struct _grn_ctx_impl) + (grn_pagesize - 1)) & ~(grn_pagesize - 1))
#ifdef GRN_WITH_MESSAGE_PACK
static int
grn_msgpack_buffer_write(void *data, const char *buf, msgpack_size_t len)
{
grn_ctx *ctx = (grn_ctx *)data;
return grn_bulk_write(ctx, ctx->impl->output.buf, buf, len);
}
#endif
static grn_rc
grn_ctx_impl_init(grn_ctx *ctx)
{
grn_io_mapinfo mi;
if (!(ctx->impl = grn_io_anon_map(ctx, &mi, IMPL_SIZE))) {
return ctx->rc;
}
grn_alloc_init_ctx_impl(ctx);
ctx->impl->encoding = ctx->encoding;
ctx->impl->lifoseg = -1;
ctx->impl->currseg = -1;
CRITICAL_SECTION_INIT(ctx->impl->lock);
if (!(ctx->impl->values = grn_array_create(ctx, NULL, sizeof(grn_db_obj *),
GRN_ARRAY_TINY))) {
CRITICAL_SECTION_FIN(ctx->impl->lock);
grn_io_anon_unmap(ctx, &mi, IMPL_SIZE);
ctx->impl = NULL;
return ctx->rc;
}
if (!(ctx->impl->temporary_columns = grn_pat_create(ctx, NULL,
GRN_TABLE_MAX_KEY_SIZE,
sizeof(grn_obj *),
0))) {
grn_array_close(ctx, ctx->impl->values);
CRITICAL_SECTION_FIN(ctx->impl->lock);
grn_io_anon_unmap(ctx, &mi, IMPL_SIZE);
ctx->impl = NULL;
return ctx->rc;
}
if (!(ctx->impl->ios = grn_hash_create(ctx, NULL, GRN_TABLE_MAX_KEY_SIZE,
sizeof(grn_io *),
GRN_OBJ_KEY_VAR_SIZE|GRN_HASH_TINY))) {
grn_array_close(ctx, ctx->impl->values);
grn_pat_close(ctx, ctx->impl->temporary_columns);
CRITICAL_SECTION_FIN(ctx->impl->lock);
grn_io_anon_unmap(ctx, &mi, IMPL_SIZE);
ctx->impl = NULL;
return ctx->rc;
}
ctx->impl->db = NULL;
ctx->impl->expr_vars = grn_hash_create(ctx, NULL, sizeof(grn_id), sizeof(grn_obj *), 0);
ctx->impl->stack_curr = 0;
ctx->impl->curr_expr = NULL;
GRN_TEXT_INIT(&ctx->impl->current_request_id, 0);
ctx->impl->current_request_timer_id = NULL;
ctx->impl->parser = NULL;
GRN_TEXT_INIT(&ctx->impl->output.names, GRN_OBJ_VECTOR);
GRN_UINT32_INIT(&ctx->impl->output.levels, GRN_OBJ_VECTOR);
ctx->impl->command.flags = 0;
if (ctx == &grn_gctx) {
ctx->impl->command.version = GRN_COMMAND_VERSION_STABLE;
} else {
ctx->impl->command.version = grn_get_default_command_version();
}
ctx->impl->command.keep.command = NULL;
ctx->impl->command.keep.version = ctx->impl->command.version;
if (ctx == &grn_gctx) {
ctx->impl->match_escalation_threshold =
GRN_DEFAULT_MATCH_ESCALATION_THRESHOLD;
} else {
ctx->impl->match_escalation_threshold =
grn_get_default_match_escalation_threshold();
}
ctx->impl->finalizer = NULL;
ctx->impl->com = NULL;
ctx->impl->output.buf = grn_obj_open(ctx, GRN_BULK, 0, GRN_DB_TEXT);
ctx->impl->output.func = NULL;
ctx->impl->output.data.ptr = NULL;
#ifdef GRN_WITH_MESSAGE_PACK
msgpack_packer_init(&ctx->impl->output.msgpacker,
ctx, grn_msgpack_buffer_write);
#endif
ctx->impl->tv.tv_sec = 0;
ctx->impl->tv.tv_nsec = 0;
ctx->impl->edge = NULL;
grn_loader_init(&ctx->impl->loader);
ctx->impl->plugin_path = NULL;
GRN_TEXT_INIT(&ctx->impl->query_log_buf, 0);
ctx->impl->previous_errbuf[0] = '\0';
ctx->impl->n_same_error_messages = 0;
grn_ctx_impl_mrb_init(ctx);
GRN_TEXT_INIT(&(ctx->impl->temporary_open_spaces.stack), 0);
ctx->impl->temporary_open_spaces.current = NULL;
return ctx->rc;
}
void
grn_ctx_set_keep_command(grn_ctx *ctx, grn_obj *command)
{
ctx->impl->command.keep.command = command;
ctx->impl->command.keep.version = ctx->impl->command.version;
}
static void
grn_ctx_impl_clear_n_same_error_messagges(grn_ctx *ctx)
{
if (ctx->impl->n_same_error_messages == 0) {
return;
}
GRN_LOG(ctx, GRN_LOG_NOTICE, "(%u same messages are truncated)",
ctx->impl->n_same_error_messages);
ctx->impl->n_same_error_messages = 0;
}
grn_bool
grn_ctx_impl_should_log(grn_ctx *ctx)
{
if (!ctx->impl) {
return GRN_TRUE;
}
if (strcmp(ctx->errbuf, ctx->impl->previous_errbuf) == 0) {
ctx->impl->n_same_error_messages++;
return GRN_FALSE;
}
return GRN_TRUE;
}
void
grn_ctx_impl_set_current_error_message(grn_ctx *ctx)
{
if (!ctx->impl) {
return;
}
grn_ctx_impl_clear_n_same_error_messagges(ctx);
grn_strcpy(ctx->impl->previous_errbuf, GRN_CTX_MSGSIZE, ctx->errbuf);
}
static grn_rc
grn_ctx_init_internal(grn_ctx *ctx, int flags)
{
if (!ctx) { return GRN_INVALID_ARGUMENT; }
// if (ctx->stat != GRN_CTX_FIN) { return GRN_INVALID_ARGUMENT; }
ctx->rc = GRN_SUCCESS;
ERRCLR(ctx);
ctx->flags = flags;
if (grn_ctx_per_db) {
ctx->flags |= GRN_CTX_PER_DB;
}
ctx->stat = GRN_CTX_INITED;
ctx->encoding = grn_gctx.encoding;
ctx->seqno = 0;
ctx->seqno2 = 0;
ctx->subno = 0;
ctx->impl = NULL;
ctx->user_data.ptr = NULL;
CRITICAL_SECTION_ENTER(grn_glock);
ctx->next = grn_gctx.next;
ctx->prev = &grn_gctx;
grn_gctx.next->prev = ctx;
grn_gctx.next = ctx;
CRITICAL_SECTION_LEAVE(grn_glock);
ctx->errline = 0;
ctx->errfile = "";
ctx->errfunc = "";
ctx->trace[0] = NULL;
ctx->errbuf[0] = '\0';
return GRN_SUCCESS;
}
grn_rc
grn_ctx_init(grn_ctx *ctx, int flags)
{
grn_rc rc;
rc = grn_ctx_init_internal(ctx, flags);
if (rc == GRN_SUCCESS) {
grn_ctx_impl_init(ctx);
rc = ctx->rc;
if (rc != GRN_SUCCESS) {
grn_ctx_fin(ctx);
if (flags & GRN_CTX_ALLOCATED) {
CRITICAL_SECTION_ENTER(grn_glock);
ctx->next->prev = ctx->prev;
ctx->prev->next = ctx->next;
CRITICAL_SECTION_LEAVE(grn_glock);
}
}
}
return rc;
}
grn_ctx *
grn_ctx_open(int flags)
{
grn_ctx *ctx = GRN_GMALLOCN(grn_ctx, 1);
if (ctx) {
grn_ctx_init(ctx, flags|GRN_CTX_ALLOCATED);
if (ERRP(ctx, GRN_ERROR)) {
GRN_GFREE(ctx);
ctx = NULL;
}
}
return ctx;
}
grn_rc
grn_ctx_fin(grn_ctx *ctx)
{
grn_rc rc = GRN_SUCCESS;
if (!ctx) { return GRN_INVALID_ARGUMENT; }
if (ctx->stat == GRN_CTX_FIN) { return GRN_INVALID_ARGUMENT; }
if (!(ctx->flags & GRN_CTX_ALLOCATED)) {
CRITICAL_SECTION_ENTER(grn_glock);
ctx->next->prev = ctx->prev;
ctx->prev->next = ctx->next;
CRITICAL_SECTION_LEAVE(grn_glock);
}
if (ctx->impl) {
grn_ctx_impl_clear_n_same_error_messagges(ctx);
if (ctx->impl->finalizer) {
ctx->impl->finalizer(ctx, 0, NULL, &(ctx->user_data));
}
{
grn_obj *stack;
grn_obj *spaces;
unsigned int i, n_spaces;
stack = &(ctx->impl->temporary_open_spaces.stack);
spaces = (grn_obj *)GRN_BULK_HEAD(stack);
n_spaces = GRN_BULK_VSIZE(stack) / sizeof(grn_obj);
for (i = 0; i < n_spaces; i++) {
grn_obj *space = spaces + (n_spaces - i - 1);
GRN_OBJ_FIN(ctx, space);
}
GRN_OBJ_FIN(ctx, stack);
}
grn_ctx_impl_mrb_fin(ctx);
grn_ctx_loader_clear(ctx);
if (ctx->impl->parser) {
grn_expr_parser_close(ctx);
}
GRN_OBJ_FIN(ctx, &ctx->impl->current_request_id);
if (ctx->impl->values) {
#ifndef USE_MEMORY_DEBUG
grn_db_obj *o;
GRN_ARRAY_EACH(ctx, ctx->impl->values, 0, 0, id, &o, {
grn_obj_close(ctx, *((grn_obj **)o));
});
#endif
grn_array_close(ctx, ctx->impl->values);
}
if (ctx->impl->temporary_columns) {
#ifndef USE_MEMORY_DEBUG
grn_obj *value;
GRN_PAT_EACH(ctx, ctx->impl->temporary_columns, id, NULL, NULL, &value, {
grn_obj_close(ctx, *((grn_obj **)value));
});
#endif
grn_pat_close(ctx, ctx->impl->temporary_columns);
}
if (ctx->impl->ios) {
grn_hash_close(ctx, ctx->impl->ios);
}
if (ctx->impl->com) {
if (ctx->stat != GRN_CTX_QUIT) {
int flags;
char *str;
unsigned int str_len;
grn_ctx_send(ctx, "quit", 4, GRN_CTX_HEAD);
grn_ctx_recv(ctx, &str, &str_len, &flags);
}
grn_ctx_send(ctx, "ACK", 3, GRN_CTX_HEAD);
rc = grn_com_close(ctx, ctx->impl->com);
}
GRN_OBJ_FIN(ctx, &ctx->impl->query_log_buf);
GRN_OBJ_FIN(ctx, &ctx->impl->output.names);
GRN_OBJ_FIN(ctx, &ctx->impl->output.levels);
rc = grn_obj_close(ctx, ctx->impl->output.buf);
{
grn_hash **vp;
grn_obj *value;
GRN_HASH_EACH(ctx, ctx->impl->expr_vars, eid, NULL, NULL, &vp, {
if (*vp) {
GRN_HASH_EACH(ctx, *vp, id, NULL, NULL, &value, {
GRN_OBJ_FIN(ctx, value);
});
}
grn_hash_close(ctx, *vp);
});
}
grn_hash_close(ctx, ctx->impl->expr_vars);
if (ctx->impl->db && ctx->flags & GRN_CTX_PER_DB) {
grn_obj *db = ctx->impl->db;
ctx->impl->db = NULL;
grn_obj_close(ctx, db);
}
grn_alloc_fin_ctx_impl(ctx);
grn_alloc_info_dump(ctx);
grn_alloc_info_free(ctx);
CRITICAL_SECTION_FIN(ctx->impl->lock);
{
grn_io_mapinfo mi;
mi.map = (void *)ctx->impl;
grn_io_anon_unmap(ctx, &mi, IMPL_SIZE);
}
ctx->impl = NULL;
}
ctx->stat = GRN_CTX_FIN;
return rc;
}
grn_rc
grn_ctx_set_finalizer(grn_ctx *ctx, grn_proc_func *finalizer)
{
if (!ctx) { return GRN_INVALID_ARGUMENT; }
if (!ctx->impl) {
if (ERRP(ctx, GRN_ERROR)) { return ctx->rc; }
}
ctx->impl->finalizer = finalizer;
return GRN_SUCCESS;
}
grn_timeval grn_starttime;
static void
check_overcommit_memory(grn_ctx *ctx)
{
FILE *file;
int value;
file = grn_fopen("/proc/sys/vm/overcommit_memory", "r");
if (!file) { return; }
value = fgetc(file);
if (value != '1') {
GRN_LOG(ctx, GRN_LOG_NOTICE,
"vm.overcommit_memory kernel parameter should be 1: <%c>: "
"See INFO level log to resolve this",
value);
GRN_LOG(ctx, GRN_LOG_INFO,
"Some processings with vm.overcommit_memory != 1 "
"may break DB under low memory condition.");
GRN_LOG(ctx, GRN_LOG_INFO,
"To set vm.overcommit_memory to 1");
GRN_LOG(ctx, GRN_LOG_INFO,
"add 'vm.overcommit_memory = 1' to /etc/sysctl.conf and "
"restart your system or");
GRN_LOG(ctx, GRN_LOG_INFO,
"run 'sudo /sbin/sysctl vm.overcommit_memory=1' command.");
}
fclose(file);
}
grn_rc
grn_init(void)
{
grn_rc rc;
grn_ctx *ctx = &grn_gctx;
grn_init_from_env();
grn_init_external_libraries();
grn_alloc_info_init();
grn_logger_init();
grn_query_logger_init();
CRITICAL_SECTION_INIT(grn_glock);
grn_gtick = 0;
ctx->next = ctx;
ctx->prev = ctx;
rc = grn_ctx_init_internal(ctx, 0);
if (rc) {
goto fail_ctx_init_internal;
}
ctx->encoding = grn_encoding_parse(GRN_DEFAULT_ENCODING);
rc = grn_timeval_now(ctx, &grn_starttime);
if (rc) {
goto fail_start_time;
}
#ifdef WIN32
{
SYSTEM_INFO si;
GetSystemInfo(&si);
grn_pagesize = si.dwAllocationGranularity;
}
#else /* WIN32 */
if ((grn_pagesize = sysconf(_SC_PAGESIZE)) == -1) {
SERR("_SC_PAGESIZE");
rc = ctx->rc;
goto fail_page_size;
}
#endif /* WIN32 */
if (grn_pagesize & (grn_pagesize - 1)) {
GRN_LOG(ctx, GRN_LOG_CRIT, "pagesize=%x", grn_pagesize);
}
// expand_stack();
if ((rc = grn_com_init())) {
GRN_LOG(ctx, GRN_LOG_ALERT, "grn_com_init failed (%d)", rc);
goto fail_com;
}
if ((rc = grn_ctx_impl_init(ctx))) {
GRN_LOG(ctx, GRN_LOG_ALERT, "grn_ctx_impl_init failed (%d)", rc);
goto fail_ctx_impl;
}
if ((rc = grn_plugins_init())) {
GRN_LOG(ctx, GRN_LOG_ALERT, "grn_plugins_init failed (%d)", rc);
goto fail_plugins;
}
if ((rc = grn_normalizer_init())) {
GRN_LOG(ctx, GRN_LOG_ALERT, "grn_normalizer_init failed (%d)", rc);
goto fail_normalizer;
}
if ((rc = grn_tokenizers_init())) {
GRN_LOG(ctx, GRN_LOG_ALERT, "grn_tokenizers_init failed (%d)", rc);
goto fail_tokenizer;
}
grn_cache_init();
if (!grn_request_canceler_init()) {
rc = ctx->rc;
GRN_LOG(ctx, GRN_LOG_ALERT,
"failed to initialize request canceler (%d)", rc);
goto fail_request_canceler;
}
if (!grn_request_timer_init()) {
rc = ctx->rc;
GRN_LOG(ctx, GRN_LOG_ALERT,
"failed to initialize request timer (%d)", rc);
goto fail_request_timer;
}
GRN_LOG(ctx, GRN_LOG_NOTICE, "grn_init: <%s>", grn_get_version());
check_overcommit_memory(ctx);
return rc;
fail_request_timer:
grn_request_canceler_fin();
fail_request_canceler:
grn_cache_fin();
fail_tokenizer:
grn_normalizer_fin();
fail_normalizer:
grn_plugins_fin();
fail_plugins:
grn_ctx_fin(ctx);
fail_ctx_impl:
grn_com_fin();
fail_com:
#ifndef WIN32
fail_page_size:
#endif /* WIN32 */
fail_start_time:
fail_ctx_init_internal:
GRN_LOG(ctx, GRN_LOG_NOTICE, "grn_init: <%s>: failed", grn_get_version());
grn_query_logger_fin(ctx);
grn_logger_fin(ctx);
CRITICAL_SECTION_FIN(grn_glock);
grn_alloc_info_fin();
grn_fin_external_libraries();
return rc;
}
grn_encoding
grn_get_default_encoding(void)
{
return grn_gctx.encoding;
}
grn_rc
grn_set_default_encoding(grn_encoding encoding)
{
switch (encoding) {
case GRN_ENC_DEFAULT :
grn_gctx.encoding = grn_encoding_parse(GRN_DEFAULT_ENCODING);
return GRN_SUCCESS;
case GRN_ENC_NONE :
case GRN_ENC_EUC_JP :
case GRN_ENC_UTF8 :
case GRN_ENC_SJIS :
case GRN_ENC_LATIN1 :
case GRN_ENC_KOI8R :
grn_gctx.encoding = encoding;
return GRN_SUCCESS;
default :
return GRN_INVALID_ARGUMENT;
}
}
grn_command_version
grn_get_default_command_version(void)
{
return grn_ctx_get_command_version(&grn_gctx);
}
grn_rc
grn_set_default_command_version(grn_command_version version)
{
return grn_ctx_set_command_version(&grn_gctx, version);
}
long long int
grn_get_default_match_escalation_threshold(void)
{
return grn_ctx_get_match_escalation_threshold(&grn_gctx);
}
grn_rc
grn_set_default_match_escalation_threshold(long long int threshold)
{
return grn_ctx_set_match_escalation_threshold(&grn_gctx, threshold);
}
int
grn_get_lock_timeout(void)
{
return grn_lock_timeout;
}
grn_rc
grn_set_lock_timeout(int timeout)
{
grn_lock_timeout = timeout;
return GRN_SUCCESS;
}
grn_rc
grn_fin(void)
{
grn_ctx *ctx, *ctx_;
if (grn_gctx.stat == GRN_CTX_FIN) { return GRN_INVALID_ARGUMENT; }
for (ctx = grn_gctx.next; ctx != &grn_gctx; ctx = ctx_) {
ctx_ = ctx->next;
if (ctx->stat != GRN_CTX_FIN) { grn_ctx_fin(ctx); }
if (ctx->flags & GRN_CTX_ALLOCATED) {
ctx->next->prev = ctx->prev;
ctx->prev->next = ctx->next;
GRN_GFREE(ctx);
}
}
grn_query_logger_fin(ctx);
grn_request_timer_fin();
grn_request_canceler_fin();
grn_cache_fin();
grn_tokenizers_fin();
grn_normalizer_fin();
grn_plugins_fin();
grn_ctx_fin(ctx);
grn_com_fin();
GRN_LOG(ctx, GRN_LOG_NOTICE, "grn_fin (%d)", grn_alloc_count());
grn_logger_fin(ctx);
CRITICAL_SECTION_FIN(grn_glock);
grn_alloc_info_fin();
grn_fin_external_libraries();
return GRN_SUCCESS;
}
grn_rc
grn_ctx_connect(grn_ctx *ctx, const char *host, int port, int flags)
{
GRN_API_ENTER;
if (!ctx->impl) { goto exit; }
{
grn_com *com = grn_com_copen(ctx, NULL, host, port);
if (com) {
ctx->impl->com = com;
}
}
exit :
GRN_API_RETURN(ctx->rc);
}
grn_rc
grn_ctx_close(grn_ctx *ctx)
{
grn_rc rc = grn_ctx_fin(ctx);
CRITICAL_SECTION_ENTER(grn_glock);
ctx->next->prev = ctx->prev;
ctx->prev->next = ctx->next;
CRITICAL_SECTION_LEAVE(grn_glock);
GRN_GFREE(ctx);
return rc;
}
grn_command_version
grn_ctx_get_command_version(grn_ctx *ctx)
{
if (ctx->impl) {
return ctx->impl->command.version;
} else {
return GRN_COMMAND_VERSION_STABLE;
}
}
grn_rc
grn_ctx_set_command_version(grn_ctx *ctx, grn_command_version version)
{
switch (version) {
case GRN_COMMAND_VERSION_DEFAULT :
ctx->impl->command.version = GRN_COMMAND_VERSION_STABLE;
return GRN_SUCCESS;
default :
if (GRN_COMMAND_VERSION_MIN <= version &&
version <= GRN_COMMAND_VERSION_MAX) {
ctx->impl->command.version = version;
return GRN_SUCCESS;
} else {
return GRN_UNSUPPORTED_COMMAND_VERSION;
}
}
}
grn_content_type
grn_ctx_get_output_type(grn_ctx *ctx)
{
if (ctx->impl) {
return ctx->impl->output.type;
} else {
return GRN_CONTENT_NONE;
}
}
grn_rc
grn_ctx_set_output_type(grn_ctx *ctx, grn_content_type type)
{
grn_rc rc = GRN_SUCCESS;
if (ctx->impl) {
ctx->impl->output.type = type;
switch (ctx->impl->output.type) {
case GRN_CONTENT_NONE :
ctx->impl->output.mime_type = "application/octet-stream";
break;
case GRN_CONTENT_TSV :
ctx->impl->output.mime_type = "text/tab-separated-values";
break;
case GRN_CONTENT_JSON :
ctx->impl->output.mime_type = "application/json";
break;
case GRN_CONTENT_XML :
ctx->impl->output.mime_type = "text/xml";
break;
case GRN_CONTENT_MSGPACK :
ctx->impl->output.mime_type = "application/x-msgpack";
break;
case GRN_CONTENT_GROONGA_COMMAND_LIST :
ctx->impl->output.mime_type = "text/x-groonga-command-list";
break;
}
} else {
rc = GRN_INVALID_ARGUMENT;
}
return rc;
}
const char *
grn_ctx_get_mime_type(grn_ctx *ctx)
{
if (ctx->impl) {
return ctx->impl->output.mime_type;
} else {
return NULL;
}
}
long long int
grn_ctx_get_match_escalation_threshold(grn_ctx *ctx)
{
if (ctx->impl) {
return ctx->impl->match_escalation_threshold;
} else {
return GRN_DEFAULT_MATCH_ESCALATION_THRESHOLD;
}
}
grn_rc
grn_ctx_set_match_escalation_threshold(grn_ctx *ctx, long long int threshold)
{
ctx->impl->match_escalation_threshold = threshold;
return GRN_SUCCESS;
}
grn_content_type
grn_get_ctype(grn_obj *var)
{
return grn_content_type_parse(NULL, var, GRN_CONTENT_JSON);
}
grn_content_type
grn_content_type_parse(grn_ctx *ctx,
grn_obj *var,
grn_content_type default_value)
{
grn_content_type ct = default_value;
if (var->header.domain == GRN_DB_INT32) {
ct = GRN_INT32_VALUE(var);
} else if (GRN_TEXT_LEN(var)) {
switch (*(GRN_TEXT_VALUE(var))) {
case 't' :
case 'T' :
ct = GRN_CONTENT_TSV;
break;
case 'j' :
case 'J' :
ct = GRN_CONTENT_JSON;
break;
case 'x' :
case 'X' :
ct = GRN_CONTENT_XML;
break;
}
}
return ct;
}
static void
get_content_mime_type(grn_ctx *ctx, const char *p, const char *pe)
{
ctx->impl->output.type = GRN_CONTENT_NONE;
ctx->impl->output.mime_type = "application/octet-stream";
if (p + 2 <= pe) {
switch (*p) {
case 'c' :
if (p + 3 == pe && !memcmp(p, "css", 3)) {
ctx->impl->output.type = GRN_CONTENT_NONE;
ctx->impl->output.mime_type = "text/css";
}
break;
case 'g' :
if (p + 3 == pe && !memcmp(p, "gif", 3)) {
ctx->impl->output.type = GRN_CONTENT_NONE;
ctx->impl->output.mime_type = "image/gif";
}
break;
case 'h' :
if (p + 4 == pe && !memcmp(p, "html", 4)) {
ctx->impl->output.type = GRN_CONTENT_NONE;
ctx->impl->output.mime_type = "text/html";
}
break;
case 'j' :
if (!memcmp(p, "js", 2)) {
if (p + 2 == pe) {
ctx->impl->output.type = GRN_CONTENT_NONE;
ctx->impl->output.mime_type = "text/javascript";
} else if (p + 4 == pe && !memcmp(p + 2, "on", 2)) {
ctx->impl->output.type = GRN_CONTENT_JSON;
ctx->impl->output.mime_type = "application/json";
}
} else if (p + 3 == pe && !memcmp(p, "jpg", 3)) {
ctx->impl->output.type = GRN_CONTENT_NONE;
ctx->impl->output.mime_type = "image/jpeg";
}
break;
#ifdef GRN_WITH_MESSAGE_PACK
case 'm' :
if (p + 7 == pe && !memcmp(p, "msgpack", 7)) {
ctx->impl->output.type = GRN_CONTENT_MSGPACK;
ctx->impl->output.mime_type = "application/x-msgpack";
}
break;
#endif
case 'p' :
if (p + 3 == pe && !memcmp(p, "png", 3)) {
ctx->impl->output.type = GRN_CONTENT_NONE;
ctx->impl->output.mime_type = "image/png";
}
break;
case 't' :
if (p + 3 == pe && !memcmp(p, "txt", 3)) {
ctx->impl->output.type = GRN_CONTENT_NONE;
ctx->impl->output.mime_type = "text/plain";
} else if (p + 3 == pe && !memcmp(p, "tsv", 3)) {
ctx->impl->output.type = GRN_CONTENT_TSV;
ctx->impl->output.mime_type = "text/tab-separated-values";
}
break;
case 'x':
if (p + 3 == pe && !memcmp(p, "xml", 3)) {
ctx->impl->output.type = GRN_CONTENT_XML;
ctx->impl->output.mime_type = "text/xml";
}
break;
}
}
}
static void
grn_str_get_mime_type(grn_ctx *ctx, const char *p, const char *pe,
const char **key_end, const char **filename_end)
{
const char *pd = NULL;
for (; p < pe && *p != '?' && *p != '#'; p++) {
if (*p == '.') { pd = p; }
}
*filename_end = p;
if (pd && pd < p) {
get_content_mime_type(ctx, pd + 1, p);
*key_end = pd;
} else {
*key_end = pe;
}
}
static void
get_command_version(grn_ctx *ctx, const char *p, const char *pe)
{
grn_command_version version;
const char *rest;
version = grn_atoui(p, pe, &rest);
if (pe == rest) {
grn_rc rc;
rc = grn_ctx_set_command_version(ctx, version);
if (rc == GRN_UNSUPPORTED_COMMAND_VERSION) {
ERR(rc,
"unsupported command version is specified: %d: "
"stable command version: %d: "
"available command versions: %d-%d",
version,
GRN_COMMAND_VERSION_STABLE,
GRN_COMMAND_VERSION_MIN, GRN_COMMAND_VERSION_MAX);
}
}
}
#define INDEX_HTML "index.html"
#define OUTPUT_TYPE "output_type"
#define COMMAND_VERSION "command_version"
#define REQUEST_ID "request_id"
#define REQUEST_TIMEOUT "request_timeout"
#define OUTPUT_PRETTY "output_pretty"
#define EXPR_MISSING "expr_missing"
#define OUTPUT_TYPE_LEN (sizeof(OUTPUT_TYPE) - 1)
#define COMMAND_VERSION_LEN (sizeof(COMMAND_VERSION) - 1)
#define REQUEST_ID_LEN (sizeof(REQUEST_ID) - 1)
#define REQUEST_TIMEOUT_LEN (sizeof(REQUEST_TIMEOUT) - 1)
#define OUTPUT_PRETTY_LEN (sizeof(OUTPUT_PRETTY) - 1)
#define HTTP_QUERY_PAIR_DELIMITER "="
#define HTTP_QUERY_PAIRS_DELIMITERS "&;"
static inline int
command_proc_p(grn_obj *expr)
{
return (expr->header.type == GRN_PROC &&
((grn_proc *)expr)->type == GRN_PROC_COMMAND);
}
grn_obj *
grn_ctx_qe_exec_uri(grn_ctx *ctx, const char *path, uint32_t path_len)
{
grn_obj buf, *expr, *val;
grn_obj request_id;
double request_timeout;
const char *p = path, *e = path + path_len, *v, *key_end, *filename_end;
request_timeout = grn_get_default_request_timeout();
GRN_TEXT_INIT(&buf, 0);
GRN_TEXT_INIT(&request_id, 0);
p = grn_text_urldec(ctx, &buf, p, e, '?');
if (!GRN_TEXT_LEN(&buf)) { GRN_TEXT_SETS(ctx, &buf, INDEX_HTML); }
v = GRN_TEXT_VALUE(&buf);
grn_str_get_mime_type(ctx, v, GRN_BULK_CURR(&buf), &key_end, &filename_end);
if ((GRN_TEXT_LEN(&buf) >= 2 && v[0] == 'd' && v[1] == '/')) {
const char *command_name = v + 2;
int command_name_size = key_end - command_name;
expr = grn_ctx_get(ctx, command_name, command_name_size);
if (expr && command_proc_p(expr)) {
while (p < e) {
int l;
GRN_BULK_REWIND(&buf);
p = grn_text_cgidec(ctx, &buf, p, e, HTTP_QUERY_PAIR_DELIMITER);
v = GRN_TEXT_VALUE(&buf);
l = GRN_TEXT_LEN(&buf);
if (l == OUTPUT_TYPE_LEN && !memcmp(v, OUTPUT_TYPE, OUTPUT_TYPE_LEN)) {
GRN_BULK_REWIND(&buf);
p = grn_text_cgidec(ctx, &buf, p, e, HTTP_QUERY_PAIRS_DELIMITERS);
v = GRN_TEXT_VALUE(&buf);
get_content_mime_type(ctx, v, GRN_BULK_CURR(&buf));
} else if (l == COMMAND_VERSION_LEN &&
!memcmp(v, COMMAND_VERSION, COMMAND_VERSION_LEN)) {
GRN_BULK_REWIND(&buf);
p = grn_text_cgidec(ctx, &buf, p, e, HTTP_QUERY_PAIRS_DELIMITERS);
get_command_version(ctx, GRN_TEXT_VALUE(&buf), GRN_BULK_CURR(&buf));
if (ctx->rc) { goto exit; }
} else if (l == REQUEST_ID_LEN &&
!memcmp(v, REQUEST_ID, REQUEST_ID_LEN)) {
GRN_BULK_REWIND(&request_id);
p = grn_text_cgidec(ctx, &request_id, p, e,
HTTP_QUERY_PAIRS_DELIMITERS);
} else if (l == REQUEST_TIMEOUT_LEN &&
!memcmp(v, REQUEST_TIMEOUT, REQUEST_TIMEOUT_LEN)) {
GRN_BULK_REWIND(&buf);
p = grn_text_cgidec(ctx, &buf, p, e, HTTP_QUERY_PAIRS_DELIMITERS);
GRN_TEXT_PUTC(ctx, &buf, '\0');
request_timeout = strtod(GRN_TEXT_VALUE(&buf), NULL);
} else if (l == OUTPUT_PRETTY_LEN &&
!memcmp(v, OUTPUT_PRETTY, OUTPUT_PRETTY_LEN)) {
GRN_BULK_REWIND(&buf);
p = grn_text_cgidec(ctx, &buf, p, e, HTTP_QUERY_PAIRS_DELIMITERS);
if (GRN_TEXT_LEN(&buf) == strlen("yes") &&
!memcmp(GRN_TEXT_VALUE(&buf), "yes", GRN_TEXT_LEN(&buf))) {
ctx->impl->output.is_pretty = GRN_TRUE;
} else {
ctx->impl->output.is_pretty = GRN_FALSE;
}
} else {
if (!(val = grn_expr_get_or_add_var(ctx, expr, v, l))) {
val = &buf;
}
grn_obj_reinit(ctx, val, GRN_DB_TEXT, 0);
p = grn_text_cgidec(ctx, val, p, e, HTTP_QUERY_PAIRS_DELIMITERS);
}
}
if (request_timeout > 0 && GRN_TEXT_LEN(&request_id) == 0) {
grn_text_printf(ctx, &request_id, "%p", ctx);
}
if (GRN_TEXT_LEN(&request_id) > 0) {
GRN_TEXT_SET(ctx, &ctx->impl->current_request_id,
GRN_TEXT_VALUE(&request_id),
GRN_TEXT_LEN(&request_id));
grn_request_canceler_register(ctx,
GRN_TEXT_VALUE(&request_id),
GRN_TEXT_LEN(&request_id));
if (request_timeout > 0.0) {
ctx->impl->current_request_timer_id =
grn_request_timer_register(GRN_TEXT_VALUE(&request_id),
GRN_TEXT_LEN(&request_id),
request_timeout);
}
}
ctx->impl->curr_expr = expr;
grn_expr_exec(ctx, expr, 0);
} else {
ERR(GRN_INVALID_ARGUMENT, "invalid command name: %.*s",
command_name_size, command_name);
}
} else if ((expr = grn_ctx_get(ctx, GRN_EXPR_MISSING_NAME,
strlen(GRN_EXPR_MISSING_NAME)))) {
if ((val = grn_expr_get_var_by_offset(ctx, expr, 0))) {
grn_obj_reinit(ctx, val, GRN_DB_TEXT, 0);
GRN_TEXT_SET(ctx, val, v, filename_end - v);
}
ctx->impl->curr_expr = expr;
grn_expr_exec(ctx, expr, 0);
}
exit :
GRN_OBJ_FIN(ctx, &request_id);
GRN_OBJ_FIN(ctx, &buf);
return expr;
}
grn_obj *
grn_ctx_qe_exec(grn_ctx *ctx, const char *str, uint32_t str_len)
{
char tok_type;
int offset = 0;
grn_obj buf, *expr = NULL, *val = NULL;
grn_obj request_id;
double request_timeout;
const char *p = str, *e = str + str_len, *v;
request_timeout = grn_get_default_request_timeout();
GRN_TEXT_INIT(&buf, 0);
GRN_TEXT_INIT(&request_id, 0);
p = grn_text_unesc_tok(ctx, &buf, p, e, &tok_type);
expr = grn_ctx_get(ctx, GRN_TEXT_VALUE(&buf), GRN_TEXT_LEN(&buf));
while (p < e) {
GRN_BULK_REWIND(&buf);
p = grn_text_unesc_tok(ctx, &buf, p, e, &tok_type);
v = GRN_TEXT_VALUE(&buf);
switch (tok_type) {
case GRN_TOK_VOID :
p = e;
break;
case GRN_TOK_SYMBOL :
if (GRN_TEXT_LEN(&buf) > 2 && v[0] == '-' && v[1] == '-') {
int l = GRN_TEXT_LEN(&buf) - 2;
v += 2;
if (l == OUTPUT_TYPE_LEN && !memcmp(v, OUTPUT_TYPE, OUTPUT_TYPE_LEN)) {
GRN_BULK_REWIND(&buf);
p = grn_text_unesc_tok(ctx, &buf, p, e, &tok_type);
v = GRN_TEXT_VALUE(&buf);
get_content_mime_type(ctx, v, GRN_BULK_CURR(&buf));
} else if (l == COMMAND_VERSION_LEN &&
!memcmp(v, COMMAND_VERSION, COMMAND_VERSION_LEN)) {
GRN_BULK_REWIND(&buf);
p = grn_text_unesc_tok(ctx, &buf, p, e, &tok_type);
get_command_version(ctx, GRN_TEXT_VALUE(&buf), GRN_BULK_CURR(&buf));
if (ctx->rc) { goto exit; }
} else if (l == REQUEST_ID_LEN &&
!memcmp(v, REQUEST_ID, REQUEST_ID_LEN)) {
GRN_BULK_REWIND(&request_id);
p = grn_text_unesc_tok(ctx, &request_id, p, e, &tok_type);
} else if (l == REQUEST_TIMEOUT_LEN &&
!memcmp(v, REQUEST_TIMEOUT, REQUEST_TIMEOUT_LEN)) {
GRN_BULK_REWIND(&buf);
p = grn_text_unesc_tok(ctx, &buf, p, e, &tok_type);
GRN_TEXT_PUTC(ctx, &buf, '\0');
request_timeout = strtod(GRN_TEXT_VALUE(&buf), NULL);
} else if (l == OUTPUT_PRETTY_LEN &&
!memcmp(v, OUTPUT_PRETTY, OUTPUT_PRETTY_LEN)) {
GRN_BULK_REWIND(&buf);
p = grn_text_unesc_tok(ctx, &buf, p, e, &tok_type);
if (GRN_TEXT_LEN(&buf) == strlen("yes") &&
!memcmp(GRN_TEXT_VALUE(&buf), "yes", GRN_TEXT_LEN(&buf))) {
ctx->impl->output.is_pretty = GRN_TRUE;
} else {
ctx->impl->output.is_pretty = GRN_FALSE;
}
} else if (expr && (val = grn_expr_get_or_add_var(ctx, expr, v, l))) {
grn_obj_reinit(ctx, val, GRN_DB_TEXT, 0);
p = grn_text_unesc_tok(ctx, val, p, e, &tok_type);
} else {
p = e;
}
break;
}
// fallthru
case GRN_TOK_STRING :
case GRN_TOK_QUOTE :
if (expr && (val = grn_expr_get_var_by_offset(ctx, expr, offset++))) {
grn_obj_reinit(ctx, val, GRN_DB_TEXT, 0);
GRN_TEXT_PUT(ctx, val, GRN_TEXT_VALUE(&buf), GRN_TEXT_LEN(&buf));
} else {
p = e;
}
break;
}
}
if (request_timeout > 0 && GRN_TEXT_LEN(&request_id) == 0) {
grn_text_printf(ctx, &request_id, "%p", ctx);
}
if (GRN_TEXT_LEN(&request_id) > 0) {
GRN_TEXT_SET(ctx, &ctx->impl->current_request_id,
GRN_TEXT_VALUE(&request_id),
GRN_TEXT_LEN(&request_id));
grn_request_canceler_register(ctx,
GRN_TEXT_VALUE(&request_id),
GRN_TEXT_LEN(&request_id));
if (request_timeout > 0.0) {
ctx->impl->current_request_timer_id =
grn_request_timer_register(GRN_TEXT_VALUE(&request_id),
GRN_TEXT_LEN(&request_id),
request_timeout);
}
}
ctx->impl->curr_expr = expr;
if (expr && command_proc_p(expr)) {
grn_expr_exec(ctx, expr, 0);
} else {
GRN_BULK_REWIND(&buf);
grn_text_unesc_tok(ctx, &buf, str, str + str_len, &tok_type);
if (GRN_TEXT_LEN(&buf)) {
ERR(GRN_INVALID_ARGUMENT, "invalid command name: %.*s",
(int)GRN_TEXT_LEN(&buf), GRN_TEXT_VALUE(&buf));
}
}
exit :
GRN_OBJ_FIN(ctx, &request_id);
GRN_OBJ_FIN(ctx, &buf);
return expr;
}
grn_rc
grn_ctx_sendv(grn_ctx *ctx, int argc, char **argv, int flags)
{
grn_obj buf;
GRN_API_ENTER;
GRN_TEXT_INIT(&buf, 0);
while (argc--) {
// todo : encode into json like syntax
GRN_TEXT_PUTS(ctx, &buf, *argv);
argv++;
if (argc) { GRN_TEXT_PUTC(ctx, &buf, ' '); }
}
grn_ctx_send(ctx, GRN_TEXT_VALUE(&buf), GRN_TEXT_LEN(&buf), flags);
GRN_OBJ_FIN(ctx, &buf);
GRN_API_RETURN(ctx->rc);
}
static int
comment_command_p(const char *command, unsigned int length)
{
const char *p, *e;
e = command + length;
for (p = command; p < e; p++) {
switch (*p) {
case '#' :
return GRN_TRUE;
case ' ' :
case '\t' :
break;
default :
return GRN_FALSE;
}
}
return GRN_FALSE;
}
unsigned int
grn_ctx_send(grn_ctx *ctx, const char *str, unsigned int str_len, int flags)
{
if (!ctx) { return 0; }
GRN_API_ENTER;
if (ctx->impl) {
if ((flags & GRN_CTX_MORE)) { flags |= GRN_CTX_QUIET; }
if (ctx->stat == GRN_CTX_QUIT) { flags |= GRN_CTX_QUIT; }
ctx->impl->command.flags = flags;
if (ctx->impl->com) {
grn_rc rc;
grn_com_header sheader;
grn_timeval_now(ctx, &ctx->impl->tv);
sheader.proto = GRN_COM_PROTO_GQTP;
sheader.qtype = 0;
sheader.keylen = 0;
sheader.level = 0;
sheader.flags = flags;
sheader.status = 0;
sheader.opaque = 0;
sheader.cas = 0;
if ((rc = grn_com_send(ctx, ctx->impl->com, &sheader, (char *)str, str_len, 0))) {
ERR(rc, "grn_com_send failed");
}
goto exit;
} else {
grn_command_version command_version;
grn_obj *expr = NULL;
command_version = grn_ctx_get_command_version(ctx);
if (ctx->impl->command.keep.command) {
grn_obj *val;
expr = ctx->impl->command.keep.command;
ctx->impl->command.keep.command = NULL;
grn_ctx_set_command_version(ctx, ctx->impl->command.keep.version);
if ((val = grn_expr_get_var_by_offset(ctx, expr, 0))) {
grn_obj_reinit(ctx, val, GRN_DB_TEXT, 0);
GRN_TEXT_PUT(ctx, val, str, str_len);
}
grn_expr_exec(ctx, expr, 0);
} else {
if (comment_command_p(str, str_len)) { goto output; };
GRN_BULK_REWIND(ctx->impl->output.buf);
ctx->impl->output.type = GRN_CONTENT_JSON;
ctx->impl->output.mime_type = "application/json";
ctx->impl->output.is_pretty = GRN_FALSE;
grn_timeval_now(ctx, &ctx->impl->tv);
GRN_QUERY_LOG(ctx, GRN_QUERY_LOG_COMMAND,
">", "%.*s", str_len, str);
if (str_len && *str == '/') {
expr = grn_ctx_qe_exec_uri(ctx, str + 1, str_len - 1);
} else {
expr = grn_ctx_qe_exec(ctx, str, str_len);
}
}
if (ctx->stat == GRN_CTX_QUITTING) { ctx->stat = GRN_CTX_QUIT; }
if (ctx->impl->command.keep.command) {
ERRCLR(ctx);
} else {
if (ctx->impl->current_request_timer_id) {
void *timer_id = ctx->impl->current_request_timer_id;
ctx->impl->current_request_timer_id = NULL;
grn_request_timer_unregister(timer_id);
}
if (GRN_TEXT_LEN(&ctx->impl->current_request_id) > 0) {
grn_obj *request_id = &ctx->impl->current_request_id;
grn_request_canceler_unregister(ctx,
GRN_TEXT_VALUE(request_id),
GRN_TEXT_LEN(request_id));
GRN_BULK_REWIND(&ctx->impl->current_request_id);
}
GRN_QUERY_LOG(ctx, GRN_QUERY_LOG_RESULT_CODE,
"<", "rc=%d", ctx->rc);
}
output :
if (!(ctx->impl->command.flags & GRN_CTX_QUIET) &&
ctx->impl->output.func) {
ctx->impl->output.func(ctx, GRN_CTX_TAIL, ctx->impl->output.data.ptr);
}
if (expr) { grn_expr_clear_vars(ctx, expr); }
grn_ctx_set_command_version(ctx, command_version);
goto exit;
}
}
ERR(GRN_INVALID_ARGUMENT, "invalid ctx assigned");
exit :
GRN_API_RETURN(0);
}
unsigned int
grn_ctx_recv(grn_ctx *ctx, char **str, unsigned int *str_len, int *flags)
{
if (!ctx) { return GRN_INVALID_ARGUMENT; }
*flags = 0;
if (ctx->stat == GRN_CTX_QUIT) {
grn_bool have_buffer = GRN_FALSE;
if (ctx->impl &&
!ctx->impl->com &&
GRN_TEXT_LEN(ctx->impl->output.buf) > 0) {
have_buffer = GRN_TRUE;
}
*flags |= GRN_CTX_QUIT;
if (!have_buffer) {
*str = NULL;
*str_len = 0;
return 0;
}
}
GRN_API_ENTER;
if (ctx->impl) {
if (ctx->impl->com) {
grn_com_header header;
if (grn_com_recv(ctx, ctx->impl->com, &header, ctx->impl->output.buf)) {
*str = NULL;
*str_len = 0;
*flags = 0;
} else {
*str = GRN_BULK_HEAD(ctx->impl->output.buf);
*str_len = GRN_BULK_VSIZE(ctx->impl->output.buf);
if (header.flags & GRN_CTX_QUIT) {
ctx->stat = GRN_CTX_QUIT;
*flags |= GRN_CTX_QUIT;
} else {
if (!(header.flags & GRN_CTX_TAIL)) {
*flags |= GRN_CTX_MORE;
}
}
ctx->impl->output.type = header.qtype;
ctx->rc = (int16_t)ntohs(header.status);
ctx->errbuf[0] = '\0';
ctx->errline = 0;
ctx->errfile = NULL;
ctx->errfunc = NULL;
}
goto exit;
} else {
grn_obj *buf = ctx->impl->output.buf;
unsigned int head = 0, tail = GRN_BULK_VSIZE(buf);
*str = GRN_BULK_HEAD(buf) + head;
*str_len = tail - head;
GRN_BULK_REWIND(ctx->impl->output.buf);
goto exit;
}
}
ERR(GRN_INVALID_ARGUMENT, "invalid ctx assigned");
exit :
GRN_API_RETURN(0);
}
void
grn_ctx_stream_out_func(grn_ctx *ctx, int flags, void *stream)
{
if (ctx && ctx->impl) {
grn_obj *buf = ctx->impl->output.buf;
uint32_t size = GRN_BULK_VSIZE(buf);
if (size) {
if (fwrite(GRN_BULK_HEAD(buf), 1, size, (FILE *)stream)) {
fputc('\n', (FILE *)stream);
fflush((FILE *)stream);
}
GRN_BULK_REWIND(buf);
}
}
}
void
grn_ctx_recv_handler_set(grn_ctx *ctx, void (*func)(grn_ctx *, int, void *), void *func_arg)
{
if (ctx && ctx->impl) {
ctx->impl->output.func = func;
ctx->impl->output.data.ptr = func_arg;
}
}
grn_rc
grn_ctx_info_get(grn_ctx *ctx, grn_ctx_info *info)
{
if (!ctx || !ctx->impl) { return GRN_INVALID_ARGUMENT; }
if (ctx->impl->com) {
info->fd = ctx->impl->com->fd;
info->com_status = ctx->impl->com_status;
info->outbuf = ctx->impl->output.buf;
info->stat = ctx->stat;
} else {
info->fd = -1;
info->com_status = 0;
info->outbuf = ctx->impl->output.buf;
info->stat = ctx->stat;
}
return GRN_SUCCESS;
}
#define DB_P(s) ((s) && (s)->header.type == GRN_DB)
grn_rc
grn_ctx_use(grn_ctx *ctx, grn_obj *db)
{
GRN_API_ENTER;
if (db && !DB_P(db)) {
ctx->rc = GRN_INVALID_ARGUMENT;
} else {
if (!ctx->rc) {
ctx->impl->db = db;
if (db) {
grn_obj buf;
GRN_TEXT_INIT(&buf, 0);
grn_obj_get_info(ctx, db, GRN_INFO_ENCODING, &buf);
ctx->encoding = *(grn_encoding *)GRN_BULK_HEAD(&buf);
grn_obj_close(ctx, &buf);
}
}
}
GRN_API_RETURN(ctx->rc);
}
/* don't handle error inside logger functions */
void
grn_ctx_log(grn_ctx *ctx, const char *fmt, ...)
{
va_list ap;
va_start(ap, fmt);
grn_ctx_logv(ctx, fmt, ap);
va_end(ap);
}
void
grn_ctx_logv(grn_ctx *ctx, const char *fmt, va_list ap)
{
char buffer[GRN_CTX_MSGSIZE];
grn_vsnprintf(buffer, GRN_CTX_MSGSIZE, fmt, ap);
grn_strcpy(ctx->errbuf, GRN_CTX_MSGSIZE, buffer);
}
void
grn_assert(grn_ctx *ctx, int cond, const char* file, int line, const char* func)
{
if (!cond) {
GRN_LOG(ctx, GRN_LOG_WARNING, "ASSERT fail on %s %s:%d", func, file, line);
}
}
const char *
grn_get_version(void)
{
return GRN_VERSION;
}
const char *
grn_get_package(void)
{
return PACKAGE;
}
const char *
grn_get_package_label(void)
{
return PACKAGE_LABEL;
}
#if defined(HAVE_SIGNAL_H) && !defined(WIN32)
static int segv_received = 0;
static void
segv_handler(int signal_number, siginfo_t *info, void *context)
{
grn_ctx *ctx = &grn_gctx;
if (segv_received) {
GRN_LOG(ctx, GRN_LOG_CRIT, "SEGV received in SEGV handler.");
exit(EXIT_FAILURE);
}
segv_received = 1;
GRN_LOG(ctx, GRN_LOG_CRIT, "-- CRASHED!!! --");
#ifdef HAVE_BACKTRACE
# define N_TRACE_LEVEL 1024
{
static void *trace[N_TRACE_LEVEL];
int n = backtrace(trace, N_TRACE_LEVEL);
char **symbols = backtrace_symbols(trace, n);
int i;
if (symbols) {
for (i = 0; i < n; i++) {
GRN_LOG(ctx, GRN_LOG_CRIT, "%s", symbols[i]);
}
free(symbols);
}
}
#else /* HAVE_BACKTRACE */
GRN_LOG(ctx, GRN_LOG_CRIT, "backtrace() isn't available.");
#endif /* HAVE_BACKTRACE */
GRN_LOG(ctx, GRN_LOG_CRIT, "----------------");
abort();
}
#endif /* defined(HAVE_SIGNAL_H) && !defined(WIN32) */
grn_rc
grn_set_segv_handler(void)
{
grn_rc rc = GRN_SUCCESS;
#if defined(HAVE_SIGNAL_H) && !defined(WIN32)
grn_ctx *ctx = &grn_gctx;
struct sigaction action;
sigemptyset(&action.sa_mask);
action.sa_sigaction = segv_handler;
action.sa_flags = SA_SIGINFO | SA_ONSTACK;
if (sigaction(SIGSEGV, &action, NULL)) {
SERR("failed to set SIGSEGV action");
rc = ctx->rc;
};
#endif
return rc;
}
#if defined(HAVE_SIGNAL_H) && !defined(WIN32)
static struct sigaction old_int_handler;
static void
int_handler(int signal_number, siginfo_t *info, void *context)
{
grn_gctx.stat = GRN_CTX_QUIT;
sigaction(signal_number, &old_int_handler, NULL);
}
static struct sigaction old_term_handler;
static void
term_handler(int signal_number, siginfo_t *info, void *context)
{
grn_gctx.stat = GRN_CTX_QUIT;
sigaction(signal_number, &old_term_handler, NULL);
}
#endif /* defined(HAVE_SIGNAL_H) && !defined(WIN32) */
grn_rc
grn_set_int_handler(void)
{
grn_rc rc = GRN_SUCCESS;
#if defined(HAVE_SIGNAL_H) && !defined(WIN32)
grn_ctx *ctx = &grn_gctx;
struct sigaction action;
sigemptyset(&action.sa_mask);
action.sa_sigaction = int_handler;
action.sa_flags = SA_SIGINFO;
if (sigaction(SIGINT, &action, &old_int_handler)) {
SERR("failed to set SIGINT action");
rc = ctx->rc;
}
#endif
return rc;
}
grn_rc
grn_set_term_handler(void)
{
grn_rc rc = GRN_SUCCESS;
#if defined(HAVE_SIGNAL_H) && !defined(WIN32)
grn_ctx *ctx = &grn_gctx;
struct sigaction action;
sigemptyset(&action.sa_mask);
action.sa_sigaction = term_handler;
action.sa_flags = SA_SIGINFO;
if (sigaction(SIGTERM, &action, &old_term_handler)) {
SERR("failed to set SIGTERM action");
rc = ctx->rc;
}
#endif
return rc;
}
void
grn_ctx_output_flush(grn_ctx *ctx, int flags)
{
if (flags & GRN_CTX_QUIET) {
return;
}
if (!ctx->impl->output.func) {
return;
}
ctx->impl->output.func(ctx, 0, ctx->impl->output.data.ptr);
}
void
grn_ctx_output_array_open(grn_ctx *ctx, const char *name, int nelements)
{
grn_output_array_open(ctx,
ctx->impl->output.buf,
ctx->impl->output.type,
name, nelements);
}
void
grn_ctx_output_array_close(grn_ctx *ctx)
{
grn_output_array_close(ctx,
ctx->impl->output.buf,
ctx->impl->output.type);
}
void
grn_ctx_output_map_open(grn_ctx *ctx, const char *name, int nelements)
{
grn_output_map_open(ctx,
ctx->impl->output.buf,
ctx->impl->output.type,
name, nelements);
}
void
grn_ctx_output_map_close(grn_ctx *ctx)
{
grn_output_map_close(ctx,
ctx->impl->output.buf,
ctx->impl->output.type);
}
void
grn_ctx_output_null(grn_ctx *ctx)
{
grn_output_null(ctx,
ctx->impl->output.buf,
ctx->impl->output.type);
}
void
grn_ctx_output_int32(grn_ctx *ctx, int value)
{
grn_output_int32(ctx,
ctx->impl->output.buf,
ctx->impl->output.type,
value);
}
void
grn_ctx_output_int64(grn_ctx *ctx, int64_t value)
{
grn_output_int64(ctx,
ctx->impl->output.buf,
ctx->impl->output.type,
value);
}
void
grn_ctx_output_uint64(grn_ctx *ctx, uint64_t value)
{
grn_output_uint64(ctx,
ctx->impl->output.buf,
ctx->impl->output.type,
value);
}
void
grn_ctx_output_float(grn_ctx *ctx, double value)
{
grn_output_float(ctx,
ctx->impl->output.buf,
ctx->impl->output.type,
value);
}
void
grn_ctx_output_cstr(grn_ctx *ctx, const char *value)
{
grn_output_cstr(ctx,
ctx->impl->output.buf,
ctx->impl->output.type,
value);
}
void
grn_ctx_output_str(grn_ctx *ctx, const char *value, unsigned int value_len)
{
grn_output_str(ctx,
ctx->impl->output.buf,
ctx->impl->output.type,
value, value_len);
}
void
grn_ctx_output_bool(grn_ctx *ctx, grn_bool value)
{
grn_output_bool(ctx,
ctx->impl->output.buf,
ctx->impl->output.type,
value);
}
void
grn_ctx_output_obj(grn_ctx *ctx, grn_obj *value, grn_obj_format *format)
{
grn_output_obj(ctx,
ctx->impl->output.buf,
ctx->impl->output.type,
value, format);
}
void
grn_ctx_output_result_set_open(grn_ctx *ctx,
grn_obj *result_set,
grn_obj_format *format,
uint32_t n_additional_elements)
{
grn_output_result_set_open(ctx,
ctx->impl->output.buf,
ctx->impl->output.type,
result_set,
format,
n_additional_elements);
}
void
grn_ctx_output_result_set_close(grn_ctx *ctx,
grn_obj *result_set,
grn_obj_format *format)
{
grn_output_result_set_close(ctx,
ctx->impl->output.buf,
ctx->impl->output.type,
result_set,
format);
}
void
grn_ctx_output_result_set(grn_ctx *ctx,
grn_obj *result_set,
grn_obj_format *format)
{
grn_output_result_set(ctx,
ctx->impl->output.buf,
ctx->impl->output.type,
result_set,
format);
}
void
grn_ctx_output_table_columns(grn_ctx *ctx, grn_obj *table,
grn_obj_format *format)
{
grn_output_table_columns(ctx,
ctx->impl->output.buf,
ctx->impl->output.type,
table,
format);
}
void
grn_ctx_output_table_records(grn_ctx *ctx, grn_obj *table,
grn_obj_format *format)
{
grn_output_table_records(ctx,
ctx->impl->output.buf,
ctx->impl->output.type,
table,
format);
}