/* * Copyright (C) Roman Arutyunyan */ #include #include #include "ngx_rtmp.h" #include "ngx_rtmp_eval.h" #include "ngx_rtmp_cmd_module.h" #include "ngx_rtmp_record_module.h" #include #ifdef NGX_LINUX #include #endif static ngx_rtmp_publish_pt next_publish; static ngx_rtmp_play_pt next_play; static ngx_rtmp_close_stream_pt next_close_stream; static ngx_rtmp_record_done_pt next_record_done; static char *ngx_rtmp_enotify_on_event(ngx_conf_t *cf, ngx_command_t *cmd, void *conf); static ngx_int_t ngx_rtmp_enotify_postconfiguration(ngx_conf_t *cf); static void * ngx_rtmp_enotify_create_app_conf(ngx_conf_t *cf); static char * ngx_rtmp_enotify_merge_app_conf(ngx_conf_t *cf, void *parent, void *child); #define NGX_RTMP_ENOTIFY_PUBLISHING 0x01 #define NGX_RTMP_ENOTIFY_PLAYING 0x02 enum { NGX_RTMP_ENOTIFY_PUBLISH, NGX_RTMP_ENOTIFY_PLAY, NGX_RTMP_ENOTIFY_PUBLISH_DONE, NGX_RTMP_ENOTIFY_PLAY_DONE, NGX_RTMP_ENOTIFY_RECORD_DONE, NGX_RTMP_ENOTIFY_MAX }; typedef struct { ngx_str_t cmd; ngx_array_t args; /* ngx_str_t */ } ngx_rtmp_enotify_conf_t; typedef struct { ngx_rtmp_enotify_conf_t *event[NGX_RTMP_ENOTIFY_MAX]; ngx_flag_t active; } ngx_rtmp_enotify_app_conf_t; typedef struct { ngx_uint_t flags; u_char name[NGX_RTMP_MAX_NAME]; u_char args[NGX_RTMP_MAX_ARGS]; ngx_str_t path; ngx_str_t recorder; } ngx_rtmp_enotify_ctx_t; static ngx_command_t ngx_rtmp_enotify_commands[] = { { ngx_string("exec_publish"), NGX_RTMP_MAIN_CONF|NGX_RTMP_SRV_CONF|NGX_RTMP_APP_CONF|NGX_CONF_1MORE, ngx_rtmp_enotify_on_event, NGX_RTMP_APP_CONF_OFFSET, 0, NULL }, { ngx_string("exec_play"), NGX_RTMP_MAIN_CONF|NGX_RTMP_SRV_CONF|NGX_RTMP_APP_CONF|NGX_CONF_1MORE, ngx_rtmp_enotify_on_event, NGX_RTMP_APP_CONF_OFFSET, 0, NULL }, { ngx_string("exec_publish_done"), NGX_RTMP_MAIN_CONF|NGX_RTMP_SRV_CONF|NGX_RTMP_APP_CONF|NGX_CONF_1MORE, ngx_rtmp_enotify_on_event, NGX_RTMP_APP_CONF_OFFSET, 0, NULL }, { ngx_string("exec_play_done"), NGX_RTMP_MAIN_CONF|NGX_RTMP_SRV_CONF|NGX_RTMP_APP_CONF|NGX_CONF_1MORE, ngx_rtmp_enotify_on_event, NGX_RTMP_APP_CONF_OFFSET, 0, NULL }, { ngx_string("exec_record_done"), NGX_RTMP_MAIN_CONF|NGX_RTMP_SRV_CONF|NGX_RTMP_APP_CONF|NGX_RTMP_REC_CONF| NGX_CONF_1MORE, ngx_rtmp_enotify_on_event, NGX_RTMP_APP_CONF_OFFSET, 0, NULL }, ngx_null_command }; static ngx_rtmp_module_t ngx_rtmp_enotify_module_ctx = { NULL, /* preconfiguration */ ngx_rtmp_enotify_postconfiguration, /* postconfiguration */ NULL, /* create main configuration */ NULL, /* init main configuration */ NULL, /* create server configuration */ NULL, /* merge server configuration */ ngx_rtmp_enotify_create_app_conf, /* create app configuration */ ngx_rtmp_enotify_merge_app_conf /* merge app configuration */ }; ngx_module_t ngx_rtmp_enotify_module = { NGX_MODULE_V1, &ngx_rtmp_enotify_module_ctx, /* module context */ ngx_rtmp_enotify_commands, /* module directives */ NGX_RTMP_MODULE, /* module type */ NULL, /* init master */ NULL, /* init module */ NULL, /* init process */ NULL, /* init thread */ NULL, /* exit thread */ NULL, /* exit process */ NULL, /* exit master */ NGX_MODULE_V1_PADDING }; static void ngx_rtmp_enotify_eval_astr(ngx_rtmp_session_t *s, ngx_rtmp_eval_t *e, ngx_str_t *ret) { ngx_rtmp_enotify_ctx_t *ctx; ctx = ngx_rtmp_get_module_ctx(s, ngx_rtmp_enotify_module); if (ctx == NULL) { ret->len = 0; return; } ret->data = (u_char *) ctx + e->offset; ret->len = ngx_strlen(ret->data); } static void ngx_rtmp_enotify_eval_str(ngx_rtmp_session_t *s, ngx_rtmp_eval_t *e, ngx_str_t *ret) { ngx_rtmp_enotify_ctx_t *ctx; ctx = ngx_rtmp_get_module_ctx(s, ngx_rtmp_enotify_module); if (ctx == NULL) { ret->len = 0; return; } *ret = *(ngx_str_t *) ((u_char *) ctx + e->offset); } static ngx_rtmp_eval_t ngx_rtmp_enotify_eval[] = { { ngx_string("name"), ngx_rtmp_enotify_eval_astr, offsetof(ngx_rtmp_enotify_ctx_t, name) }, { ngx_string("args"), ngx_rtmp_enotify_eval_astr, offsetof(ngx_rtmp_enotify_ctx_t, args) }, { ngx_string("path"), ngx_rtmp_enotify_eval_str, offsetof(ngx_rtmp_enotify_ctx_t, path) }, { ngx_string("recorder"), ngx_rtmp_enotify_eval_str, offsetof(ngx_rtmp_enotify_ctx_t, recorder) }, ngx_rtmp_null_eval }; static ngx_rtmp_eval_t * ngx_rtmp_enotify_eval_p[] = { ngx_rtmp_eval_session, ngx_rtmp_enotify_eval, NULL }; static void * ngx_rtmp_enotify_create_app_conf(ngx_conf_t *cf) { ngx_rtmp_enotify_app_conf_t *enacf; ngx_uint_t n; enacf = ngx_pcalloc(cf->pool, sizeof(ngx_rtmp_enotify_app_conf_t)); if (enacf == NULL) { return NULL; } for (n = 0; n < NGX_RTMP_ENOTIFY_MAX; ++n) { enacf->event[n] = NGX_CONF_UNSET_PTR; } return enacf; } static char * ngx_rtmp_enotify_merge_app_conf(ngx_conf_t *cf, void *parent, void *child) { ngx_rtmp_enotify_app_conf_t *prev = parent; ngx_rtmp_enotify_app_conf_t *conf = child; ngx_uint_t n; for (n = 0; n < NGX_RTMP_ENOTIFY_MAX; ++n) { ngx_conf_merge_ptr_value(conf->event[n], prev->event[n], NULL); if (conf->event[n]) { conf->active = 1; } } if (conf->active) { prev->active = 1; } return NGX_CONF_OK; } static ngx_int_t ngx_rtmp_enotify_exec(ngx_rtmp_session_t *s, ngx_rtmp_enotify_conf_t *ec) { #if !(NGX_WIN32) int pid, fd, maxfd; ngx_str_t a, *arg_in; char **args, **arg_out; ngx_uint_t n; pid = fork(); switch (pid) { case -1: ngx_log_error(NGX_LOG_INFO, s->connection->log, ngx_errno, "enotify: fork failed"); return NGX_ERROR; case 0: /* child */ /* close all descriptors */ maxfd = sysconf(_SC_OPEN_MAX); for (fd = 0; fd < maxfd; ++fd) { close(fd); } fd = open("/dev/null", O_RDWR); dup2(fd, STDIN_FILENO); dup2(fd, STDOUT_FILENO); dup2(fd, STDERR_FILENO); args = ngx_palloc(s->connection->pool, (ec->args.nelts + 2) * sizeof(char *)); if (args == NULL) { exit(1); } arg_in = ec->args.elts; arg_out = args; *arg_out++ = (char *) ec->cmd.data; for (n = 0; n < ec->args.nelts; ++n, ++arg_in) { ngx_rtmp_eval(s, arg_in, ngx_rtmp_enotify_eval_p, &a); if (ngx_rtmp_eval_streams(&a) != NGX_DONE) { continue; } *arg_out++ = (char *) a.data; } *arg_out = NULL; if (execvp((char *) ec->cmd.data, args) == -1) { exit(1); } break; default: /* parent */ ngx_log_debug2(NGX_LOG_DEBUG_RTMP, s->connection->log, 0, "enotify: child '%V' started pid=%ui", &ec->cmd, (ngx_uint_t)pid); break; } #endif /* NGX_WIN32 */ return NGX_OK; } static void ngx_rtmp_enotify_init(ngx_rtmp_session_t *s, u_char name[NGX_RTMP_MAX_NAME], u_char args[NGX_RTMP_MAX_ARGS], ngx_uint_t flags) { ngx_rtmp_enotify_ctx_t *ctx; ngx_rtmp_enotify_app_conf_t *enacf; enacf = ngx_rtmp_get_module_app_conf(s, ngx_rtmp_enotify_module); if (!enacf->active) { return; } ctx = ngx_rtmp_get_module_ctx(s, ngx_rtmp_enotify_module); if (ctx == NULL) { ctx = ngx_pcalloc(s->connection->pool, sizeof(ngx_rtmp_enotify_ctx_t)); if (ctx == NULL) { return; } ngx_rtmp_set_ctx(s, ctx, ngx_rtmp_enotify_module); } ngx_memcpy(ctx->name, name, NGX_RTMP_MAX_NAME); ngx_memcpy(ctx->args, args, NGX_RTMP_MAX_ARGS); ctx->flags |= flags; } static ngx_int_t ngx_rtmp_enotify_publish(ngx_rtmp_session_t *s, ngx_rtmp_publish_t *v) { ngx_rtmp_enotify_app_conf_t *enacf; ngx_rtmp_enotify_conf_t *ec; if (s->auto_pushed) { goto next; } enacf = ngx_rtmp_get_module_app_conf(s, ngx_rtmp_enotify_module); if (enacf == NULL) { goto next; } ngx_rtmp_enotify_init(s, v->name, v->args, NGX_RTMP_ENOTIFY_PUBLISHING); ec = enacf->event[NGX_RTMP_ENOTIFY_PUBLISH]; if (ec == NULL) { goto next; } ngx_log_error(NGX_LOG_INFO, s->connection->log, 0, "enotify: publish '%V'", &ec->cmd); ngx_rtmp_enotify_exec(s, ec); next: return next_publish(s, v); } static ngx_int_t ngx_rtmp_enotify_play(ngx_rtmp_session_t *s, ngx_rtmp_play_t *v) { ngx_rtmp_enotify_app_conf_t *enacf; ngx_rtmp_enotify_conf_t *ec; if (s->auto_pushed) { goto next; } enacf = ngx_rtmp_get_module_app_conf(s, ngx_rtmp_enotify_module); if (enacf == NULL) { goto next; } ngx_rtmp_enotify_init(s, v->name, v->args, NGX_RTMP_ENOTIFY_PLAYING); ec = enacf->event[NGX_RTMP_ENOTIFY_PLAY]; if (ec == NULL) { goto next; } ngx_log_error(NGX_LOG_INFO, s->connection->log, 0, "enotify: play '%V'", &ec->cmd); ngx_rtmp_enotify_exec(s, ec); next: return next_play(s, v); } static ngx_int_t ngx_rtmp_enotify_close_stream(ngx_rtmp_session_t *s, ngx_rtmp_close_stream_t *v) { ngx_rtmp_enotify_ctx_t *ctx; ngx_rtmp_enotify_app_conf_t *enacf; ngx_rtmp_enotify_conf_t *ec; if (s->auto_pushed) { goto next; } ctx = ngx_rtmp_get_module_ctx(s, ngx_rtmp_enotify_module); if (ctx == NULL) { goto next; } enacf = ngx_rtmp_get_module_app_conf(s, ngx_rtmp_enotify_module); if (enacf == NULL) { goto next; } if (enacf->event[NGX_RTMP_ENOTIFY_PUBLISH_DONE] && (ctx->flags & NGX_RTMP_ENOTIFY_PUBLISHING)) { ec = enacf->event[NGX_RTMP_ENOTIFY_PUBLISH_DONE]; ngx_log_error(NGX_LOG_INFO, s->connection->log, 0, "enotify: publish_done '%V'", &ec->cmd); ngx_rtmp_enotify_exec(s, ec); } if (enacf->event[NGX_RTMP_ENOTIFY_PLAY_DONE] && (ctx->flags & NGX_RTMP_ENOTIFY_PLAYING)) { ec = enacf->event[NGX_RTMP_ENOTIFY_PLAY_DONE]; ngx_log_error(NGX_LOG_INFO, s->connection->log, 0, "enotify: play_done '%V'", &ec->cmd); ngx_rtmp_enotify_exec(s, ec); } ctx->flags = 0; next: return next_close_stream(s, v); } static ngx_int_t ngx_rtmp_enotify_record_done(ngx_rtmp_session_t *s, ngx_rtmp_record_done_t *v) { ngx_rtmp_enotify_app_conf_t *enacf; ngx_rtmp_enotify_conf_t *ec; ngx_rtmp_enotify_ctx_t *ctx; if (s->auto_pushed) { goto next; } enacf = ngx_rtmp_get_module_app_conf(s, ngx_rtmp_enotify_module); if (enacf == NULL || enacf->event[NGX_RTMP_ENOTIFY_RECORD_DONE] == NULL) { goto next; } ec = enacf->event[NGX_RTMP_ENOTIFY_RECORD_DONE]; ngx_log_error(NGX_LOG_INFO, s->connection->log, 0, "enotify: record_done %V recorder=%V path='%V'", &ec->cmd, &v->recorder, &v->path); ctx = ngx_rtmp_get_module_ctx(s, ngx_rtmp_enotify_module); if (ctx == NULL) { goto next; } ctx->recorder = v->recorder; ctx->path = v->path; ngx_rtmp_enotify_exec(s, ec); next: return next_record_done(s, v); } static char * ngx_rtmp_enotify_on_event(ngx_conf_t *cf, ngx_command_t *cmd, void *conf) { ngx_rtmp_enotify_app_conf_t *enacf; ngx_rtmp_enotify_conf_t *ec; ngx_str_t *name, *value, *s; size_t nargs; ngx_uint_t n; value = cf->args->elts; name = &value[0]; ec = ngx_pcalloc(cf->pool, sizeof(ngx_rtmp_enotify_conf_t)); if (ec == NULL) { return NGX_CONF_ERROR; } ec->cmd = value[1]; nargs = cf->args->nelts - 2; if (nargs) { if (ngx_array_init(&ec->args, cf->pool, nargs, sizeof(ngx_str_t)) != NGX_OK) { return NGX_CONF_ERROR; } s = ngx_array_push_n(&ec->args, nargs); for (n = 2; n < cf->args->nelts; ++n, ++s) { *s = value[n]; } } enacf = ngx_rtmp_conf_get_module_app_conf(cf, ngx_rtmp_enotify_module); n = 0; switch (name->len) { case sizeof("exec_play") - 1: n = NGX_RTMP_ENOTIFY_PLAY; break; case sizeof("exec_publish") - 1: n = NGX_RTMP_ENOTIFY_PUBLISH; break; case sizeof("exec_play_done") - 1: n = NGX_RTMP_ENOTIFY_PLAY_DONE; break; case sizeof("exec_record_done") - 1: n = NGX_RTMP_ENOTIFY_RECORD_DONE; break; case sizeof("exec_publish_done") - 1: n = NGX_RTMP_ENOTIFY_PUBLISH_DONE; break; } enacf->event[n] = ec; return NGX_CONF_OK; } static ngx_int_t ngx_rtmp_enotify_postconfiguration(ngx_conf_t *cf) { next_publish = ngx_rtmp_publish; ngx_rtmp_publish = ngx_rtmp_enotify_publish; next_play = ngx_rtmp_play; ngx_rtmp_play = ngx_rtmp_enotify_play; next_close_stream = ngx_rtmp_close_stream; ngx_rtmp_close_stream = ngx_rtmp_enotify_close_stream; next_record_done = ngx_rtmp_record_done; ngx_rtmp_record_done = ngx_rtmp_enotify_record_done; return NGX_OK; }