mirror of
https://github.com/h2o/h2o.git
synced 2025-05-17 04:02:11 +08:00
"Note that the pthreads functions do not set errno" from PTHREADS(7) so we should use return value instead of 'errno' when pthread_create error
177 lines
6.1 KiB
C
177 lines
6.1 KiB
C
/*
|
|
* Copyright (c) 2015 DeNA Co., Ltd., Kazuho Oku, Ryosuke Matsumoto
|
|
*
|
|
* 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.
|
|
*/
|
|
#include <errno.h>
|
|
#include <inttypes.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include "h2o.h"
|
|
#include "h2o/configurator.h"
|
|
#include "h2o/mruby_.h"
|
|
|
|
struct mruby_configurator_t {
|
|
h2o_configurator_t super;
|
|
h2o_mruby_config_vars_t *vars;
|
|
h2o_mruby_config_vars_t _vars_stack[H2O_CONFIGURATOR_NUM_LEVELS + 1];
|
|
mrb_state *mrb; /* will be lazily initialized */
|
|
};
|
|
|
|
static int compile_test(mrb_state *mrb, h2o_mruby_config_vars_t *config, char *errbuf)
|
|
{
|
|
struct RProc *result = h2o_mruby_compile_code(mrb, config, errbuf);
|
|
return result != NULL;
|
|
}
|
|
|
|
static mrb_state *get_mrb(struct mruby_configurator_t *self)
|
|
{
|
|
if (self->mrb == NULL) {
|
|
self->mrb = mrb_open();
|
|
if (self->mrb == NULL) {
|
|
h2o_fatal("%s: no memory\n", H2O_MRUBY_MODULE_NAME);
|
|
}
|
|
h2o_mruby_setup_globals(self->mrb);
|
|
}
|
|
return self->mrb;
|
|
}
|
|
|
|
static int on_config_mruby_handler(h2o_configurator_command_t *cmd, h2o_configurator_context_t *ctx, yoml_t *node)
|
|
{
|
|
struct mruby_configurator_t *self = (void *)cmd->configurator;
|
|
|
|
/* set source */
|
|
self->vars->source = h2o_strdup(NULL, node->data.scalar, SIZE_MAX);
|
|
self->vars->path = node->filename;
|
|
self->vars->lineno = (int)node->line + 1;
|
|
|
|
/* check if there is any error in source */
|
|
char errbuf[1024];
|
|
if (!compile_test(get_mrb(self), self->vars, errbuf)) {
|
|
h2o_configurator_errprintf(cmd, node, "ruby compile error:%s", errbuf);
|
|
return -1;
|
|
}
|
|
|
|
/* register */
|
|
h2o_mruby_register(ctx->pathconf, self->vars);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int on_config_mruby_handler_file(h2o_configurator_command_t *cmd, h2o_configurator_context_t *ctx, yoml_t *node)
|
|
{
|
|
struct mruby_configurator_t *self = (void *)cmd->configurator;
|
|
FILE *fp = NULL;
|
|
h2o_iovec_t buf = {NULL};
|
|
int ret = -1;
|
|
|
|
/* open and read file */
|
|
if ((fp = fopen(node->data.scalar, "rt")) == NULL) {
|
|
h2o_configurator_errprintf(cmd, node, "failed to open file: %s:%s", node->data.scalar, strerror(errno));
|
|
goto Exit;
|
|
}
|
|
while (!feof(fp)) {
|
|
buf.base = h2o_mem_realloc(buf.base, buf.len + 65536);
|
|
buf.len += fread(buf.base + buf.len, 1, 65536, fp);
|
|
if (ferror(fp)) {
|
|
h2o_configurator_errprintf(cmd, node, "I/O error occurred while reading file:%s:%s", node->data.scalar,
|
|
strerror(errno));
|
|
goto Exit;
|
|
}
|
|
}
|
|
|
|
/* set source */
|
|
self->vars->source = buf;
|
|
buf.base = NULL;
|
|
self->vars->path = node->data.scalar; /* the value is retained until the end of the configuration phase */
|
|
self->vars->lineno = 0;
|
|
|
|
/* check if there is any error in source */
|
|
char errbuf[1024];
|
|
if (!compile_test(get_mrb(self), self->vars, errbuf)) {
|
|
h2o_configurator_errprintf(cmd, node, "failed to compile file:%s:%s", node->data.scalar, errbuf);
|
|
goto Exit;
|
|
}
|
|
|
|
/* register */
|
|
h2o_mruby_register(ctx->pathconf, self->vars);
|
|
|
|
ret = 0;
|
|
|
|
Exit:
|
|
if (fp != NULL)
|
|
fclose(fp);
|
|
if (buf.base != NULL)
|
|
free(buf.base);
|
|
return ret;
|
|
}
|
|
|
|
static int on_config_mruby_handler_path(h2o_configurator_command_t *cmd, h2o_configurator_context_t *ctx, yoml_t *node)
|
|
{
|
|
h2o_configurator_errprintf(cmd, node, "the command has been removed; see https://github.com/h2o/h2o/pull/467");
|
|
return -1;
|
|
}
|
|
|
|
static int on_config_enter(h2o_configurator_t *_self, h2o_configurator_context_t *ctx, yoml_t *node)
|
|
{
|
|
struct mruby_configurator_t *self = (void *)_self;
|
|
|
|
memcpy(self->vars + 1, self->vars, sizeof(*self->vars));
|
|
++self->vars;
|
|
return 0;
|
|
}
|
|
|
|
static int on_config_exit(h2o_configurator_t *_self, h2o_configurator_context_t *ctx, yoml_t *node)
|
|
{
|
|
struct mruby_configurator_t *self = (void *)_self;
|
|
|
|
/* free if the to-be-exitted frame level contains a different source */
|
|
if (self->vars[-1].source.base != self->vars[0].source.base)
|
|
free(self->vars->source.base);
|
|
|
|
--self->vars;
|
|
|
|
/* release mrb only when global configuration exited */
|
|
if (self->mrb != NULL && ctx->parent == NULL) {
|
|
mrb_close(self->mrb);
|
|
self->mrb = NULL;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void h2o_mruby_register_configurator(h2o_globalconf_t *conf)
|
|
{
|
|
struct mruby_configurator_t *c = (void *)h2o_configurator_create(conf, sizeof(*c));
|
|
|
|
c->vars = c->_vars_stack;
|
|
c->super.enter = on_config_enter;
|
|
c->super.exit = on_config_exit;
|
|
|
|
h2o_configurator_define_command(
|
|
&c->super, "mruby.handler",
|
|
H2O_CONFIGURATOR_FLAG_PATH | H2O_CONFIGURATOR_FLAG_DEFERRED | H2O_CONFIGURATOR_FLAG_EXPECT_SCALAR, on_config_mruby_handler);
|
|
h2o_configurator_define_command(&c->super, "mruby.handler-file",
|
|
H2O_CONFIGURATOR_FLAG_PATH | H2O_CONFIGURATOR_FLAG_DEFERRED |
|
|
H2O_CONFIGURATOR_FLAG_EXPECT_SCALAR,
|
|
on_config_mruby_handler_file);
|
|
h2o_configurator_define_command(&c->super, "mruby.handler_path", H2O_CONFIGURATOR_FLAG_PATH | H2O_CONFIGURATOR_FLAG_DEFERRED,
|
|
on_config_mruby_handler_path);
|
|
}
|