src/http/modules/ngx_http_secure_link_module.c - nginx source code

Global variables defined

Data types defined

Functions defined

Source code


  1. /*
  2. * Copyright (C) Igor Sysoev
  3. * Copyright (C) Nginx, Inc.
  4. */


  5. #include <ngx_config.h>
  6. #include <ngx_core.h>
  7. #include <ngx_http.h>
  8. #include <ngx_md5.h>


  9. typedef struct {
  10.     ngx_http_complex_value_t  *variable;
  11.     ngx_http_complex_value_t  *md5;
  12.     ngx_str_t                  secret;
  13. } ngx_http_secure_link_conf_t;


  14. typedef struct {
  15.     ngx_str_t                  expires;
  16. } ngx_http_secure_link_ctx_t;


  17. static ngx_int_t ngx_http_secure_link_old_variable(ngx_http_request_t *r,
  18.     ngx_http_secure_link_conf_t *conf, ngx_http_variable_value_t *v,
  19.     uintptr_t data);
  20. static ngx_int_t ngx_http_secure_link_expires_variable(ngx_http_request_t *r,
  21.     ngx_http_variable_value_t *v, uintptr_t data);
  22. static void *ngx_http_secure_link_create_conf(ngx_conf_t *cf);
  23. static char *ngx_http_secure_link_merge_conf(ngx_conf_t *cf, void *parent,
  24.     void *child);
  25. static ngx_int_t ngx_http_secure_link_add_variables(ngx_conf_t *cf);


  26. static ngx_command_t  ngx_http_secure_link_commands[] = {

  27.     { ngx_string("secure_link"),
  28.       NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_TAKE1,
  29.       ngx_http_set_complex_value_slot,
  30.       NGX_HTTP_LOC_CONF_OFFSET,
  31.       offsetof(ngx_http_secure_link_conf_t, variable),
  32.       NULL },

  33.     { ngx_string("secure_link_md5"),
  34.       NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_TAKE1,
  35.       ngx_http_set_complex_value_slot,
  36.       NGX_HTTP_LOC_CONF_OFFSET,
  37.       offsetof(ngx_http_secure_link_conf_t, md5),
  38.       NULL },

  39.     { ngx_string("secure_link_secret"),
  40.       NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_TAKE1,
  41.       ngx_conf_set_str_slot,
  42.       NGX_HTTP_LOC_CONF_OFFSET,
  43.       offsetof(ngx_http_secure_link_conf_t, secret),
  44.       NULL },

  45.       ngx_null_command
  46. };


  47. static ngx_http_module_t  ngx_http_secure_link_module_ctx = {
  48.     ngx_http_secure_link_add_variables,    /* preconfiguration */
  49.     NULL,                                  /* postconfiguration */

  50.     NULL,                                  /* create main configuration */
  51.     NULL,                                  /* init main configuration */

  52.     NULL,                                  /* create server configuration */
  53.     NULL,                                  /* merge server configuration */

  54.     ngx_http_secure_link_create_conf,      /* create location configuration */
  55.     ngx_http_secure_link_merge_conf        /* merge location configuration */
  56. };


  57. ngx_module_t  ngx_http_secure_link_module = {
  58.     NGX_MODULE_V1,
  59.     &ngx_http_secure_link_module_ctx,      /* module context */
  60.     ngx_http_secure_link_commands,         /* module directives */
  61.     NGX_HTTP_MODULE,                       /* module type */
  62.     NULL,                                  /* init master */
  63.     NULL,                                  /* init module */
  64.     NULL,                                  /* init process */
  65.     NULL,                                  /* init thread */
  66.     NULL,                                  /* exit thread */
  67.     NULL,                                  /* exit process */
  68.     NULL,                                  /* exit master */
  69.     NGX_MODULE_V1_PADDING
  70. };


  71. static ngx_str_t  ngx_http_secure_link_name = ngx_string("secure_link");
  72. static ngx_str_t  ngx_http_secure_link_expires_name =
  73.     ngx_string("secure_link_expires");


  74. static ngx_int_t
  75. ngx_http_secure_link_variable(ngx_http_request_t *r,
  76.     ngx_http_variable_value_t *v, uintptr_t data)
  77. {
  78.     u_char                       *p, *last;
  79.     ngx_str_t                     val, hash;
  80.     time_t                        expires;
  81.     ngx_md5_t                     md5;
  82.     ngx_http_secure_link_ctx_t   *ctx;
  83.     ngx_http_secure_link_conf_t  *conf;
  84.     u_char                        hash_buf[18], md5_buf[16];

  85.     conf = ngx_http_get_module_loc_conf(r, ngx_http_secure_link_module);

  86.     if (conf->secret.data) {
  87.         return ngx_http_secure_link_old_variable(r, conf, v, data);
  88.     }

  89.     if (conf->variable == NULL || conf->md5 == NULL) {
  90.         goto not_found;
  91.     }

  92.     if (ngx_http_complex_value(r, conf->variable, &val) != NGX_OK) {
  93.         return NGX_ERROR;
  94.     }

  95.     ngx_log_debug1(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
  96.                    "secure link: \"%V\"", &val);

  97.     last = val.data + val.len;

  98.     p = ngx_strlchr(val.data, last, ',');
  99.     expires = 0;

  100.     if (p) {
  101.         val.len = p++ - val.data;

  102.         expires = ngx_atotm(p, last - p);
  103.         if (expires <= 0) {
  104.             goto not_found;
  105.         }

  106.         ctx = ngx_pcalloc(r->pool, sizeof(ngx_http_secure_link_ctx_t));
  107.         if (ctx == NULL) {
  108.             return NGX_ERROR;
  109.         }

  110.         ngx_http_set_ctx(r, ctx, ngx_http_secure_link_module);

  111.         ctx->expires.len = last - p;
  112.         ctx->expires.data = p;
  113.     }

  114.     if (val.len > 24) {
  115.         goto not_found;
  116.     }

  117.     hash.data = hash_buf;

  118.     if (ngx_decode_base64url(&hash, &val) != NGX_OK) {
  119.         goto not_found;
  120.     }

  121.     if (hash.len != 16) {
  122.         goto not_found;
  123.     }

  124.     if (ngx_http_complex_value(r, conf->md5, &val) != NGX_OK) {
  125.         return NGX_ERROR;
  126.     }

  127.     ngx_log_debug1(NGX_LOG_DEBUG_HTTP, r->connection->log, 0,
  128.                    "secure link md5: \"%V\"", &val);

  129.     ngx_md5_init(&md5);
  130.     ngx_md5_update(&md5, val.data, val.len);
  131.     ngx_md5_final(md5_buf, &md5);

  132.     if (ngx_memcmp(hash_buf, md5_buf, 16) != 0) {
  133.         goto not_found;
  134.     }

  135.     v->data = (u_char *) ((expires && expires < ngx_time()) ? "0" : "1");
  136.     v->len = 1;
  137.     v->valid = 1;
  138.     v->no_cacheable = 0;
  139.     v->not_found = 0;

  140.     return NGX_OK;

  141. not_found:

  142.     v->not_found = 1;

  143.     return NGX_OK;
  144. }


  145. static ngx_int_t
  146. ngx_http_secure_link_old_variable(ngx_http_request_t *r,
  147.     ngx_http_secure_link_conf_t *conf, ngx_http_variable_value_t *v,
  148.     uintptr_t data)
  149. {
  150.     u_char      *p, *start, *end, *last;
  151.     size_t       len;
  152.     ngx_int_t    n;
  153.     ngx_uint_t   i;
  154.     ngx_md5_t    md5;
  155.     u_char       hash[16];

  156.     p = &r->unparsed_uri.data[1];
  157.     last = r->unparsed_uri.data + r->unparsed_uri.len;

  158.     while (p < last) {
  159.         if (*p++ == '/') {
  160.             start = p;
  161.             goto md5_start;
  162.         }
  163.     }

  164.     goto not_found;

  165. md5_start:

  166.     while (p < last) {
  167.         if (*p++ == '/') {
  168.             end = p - 1;
  169.             goto url_start;
  170.         }
  171.     }

  172.     goto not_found;

  173. url_start:

  174.     len = last - p;

  175.     if (end - start != 32 || len == 0) {
  176.         goto not_found;
  177.     }

  178.     ngx_md5_init(&md5);
  179.     ngx_md5_update(&md5, p, len);
  180.     ngx_md5_update(&md5, conf->secret.data, conf->secret.len);
  181.     ngx_md5_final(hash, &md5);

  182.     for (i = 0; i < 16; i++) {
  183.         n = ngx_hextoi(&start[2 * i], 2);
  184.         if (n == NGX_ERROR || n != hash[i]) {
  185.             goto not_found;
  186.         }
  187.     }

  188.     v->len = len;
  189.     v->valid = 1;
  190.     v->no_cacheable = 0;
  191.     v->not_found = 0;
  192.     v->data = p;

  193.     return NGX_OK;

  194. not_found:

  195.     v->not_found = 1;

  196.     return NGX_OK;
  197. }


  198. static ngx_int_t
  199. ngx_http_secure_link_expires_variable(ngx_http_request_t *r,
  200.     ngx_http_variable_value_t *v, uintptr_t data)
  201. {
  202.     ngx_http_secure_link_ctx_t  *ctx;

  203.     ctx = ngx_http_get_module_ctx(r, ngx_http_secure_link_module);

  204.     if (ctx) {
  205.         v->len = ctx->expires.len;
  206.         v->valid = 1;
  207.         v->no_cacheable = 0;
  208.         v->not_found = 0;
  209.         v->data = ctx->expires.data;

  210.     } else {
  211.         v->not_found = 1;
  212.     }

  213.     return NGX_OK;
  214. }


  215. static void *
  216. ngx_http_secure_link_create_conf(ngx_conf_t *cf)
  217. {
  218.     ngx_http_secure_link_conf_t  *conf;

  219.     conf = ngx_pcalloc(cf->pool, sizeof(ngx_http_secure_link_conf_t));
  220.     if (conf == NULL) {
  221.         return NULL;
  222.     }

  223.     /*
  224.      * set by ngx_pcalloc():
  225.      *
  226.      *     conf->secret = { 0, NULL };
  227.      */

  228.     conf->variable = NGX_CONF_UNSET_PTR;
  229.     conf->md5 = NGX_CONF_UNSET_PTR;

  230.     return conf;
  231. }


  232. static char *
  233. ngx_http_secure_link_merge_conf(ngx_conf_t *cf, void *parent, void *child)
  234. {
  235.     ngx_http_secure_link_conf_t *prev = parent;
  236.     ngx_http_secure_link_conf_t *conf = child;

  237.     if (conf->secret.data) {
  238.         ngx_conf_init_ptr_value(conf->variable, NULL);
  239.         ngx_conf_init_ptr_value(conf->md5, NULL);

  240.         if (conf->variable || conf->md5) {
  241.             ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
  242.                                "\"secure_link_secret\" cannot be mixed with "
  243.                                "\"secure_link\" and \"secure_link_md5\"");
  244.             return NGX_CONF_ERROR;
  245.         }

  246.         return NGX_CONF_OK;
  247.     }

  248.     ngx_conf_merge_ptr_value(conf->variable, prev->variable, NULL);
  249.     ngx_conf_merge_ptr_value(conf->md5, prev->md5, NULL);

  250.     if (conf->variable == NULL && conf->md5 == NULL) {
  251.         conf->secret = prev->secret;
  252.     }

  253.     return NGX_CONF_OK;
  254. }


  255. static ngx_int_t
  256. ngx_http_secure_link_add_variables(ngx_conf_t *cf)
  257. {
  258.     ngx_http_variable_t  *var;

  259.     var = ngx_http_add_variable(cf, &ngx_http_secure_link_name, 0);
  260.     if (var == NULL) {
  261.         return NGX_ERROR;
  262.     }

  263.     var->get_handler = ngx_http_secure_link_variable;

  264.     var = ngx_http_add_variable(cf, &ngx_http_secure_link_expires_name, 0);
  265.     if (var == NULL) {
  266.         return NGX_ERROR;
  267.     }

  268.     var->get_handler = ngx_http_secure_link_expires_variable;

  269.     return NGX_OK;
  270. }