加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
ngx_http_access_token_module.c 11.45 KB
一键复制 编辑 原始数据 按行查看 历史
刘俊勇 提交于 2017-04-14 00:49 . 秒数放在配置
/*
* Copyright (C) 2013 Tatsuhiko Kubo <cubicdaiya@gmail.com>
*/
#include <ngx_config.h>
#include <ngx_core.h>
#include <ngx_http.h>
#include <ngx_md5.h>
#define NGX_HTTP_ACCESS_TOKEN_SET_ARG(r, arg, var) \
if (var->not_found) { \
ngx_str_null(&arg); \
} else { \
arg.data = ngx_pnalloc(r->pool, var->len + 1); \
if (arg.data == NULL) { \
return; \
} \
ngx_cpystrn(arg.data, var->data, var->len + 1); \
arg.len = var->len; \
}
typedef struct ngx_http_access_token_conf_t {
ngx_flag_t enable;
ngx_int_t allow_second;
ngx_str_t access_key;
ngx_str_t secret;
} ngx_http_access_token_conf_t;
typedef struct ngx_http_access_token_ctx_t {
ngx_str_t access_key;
ngx_str_t expires;
ngx_str_t sig;
} ngx_http_access_token_ctx_t;
static void *ngx_http_access_token_create_loc_conf(ngx_conf_t *cf);
static ngx_int_t ngx_http_access_token_init(ngx_conf_t *cf);
static char *ngx_http_access_token_merge_loc_conf(ngx_conf_t *cf, void *parent, void *child);
static ngx_int_t ngx_http_access_token_handler(ngx_http_request_t *r);
static u_char *ngx_http_access_token_hmac(ngx_pool_t *pool, ngx_str_t *key, ngx_str_t *text);
static void ngx_http_access_token_set_args(ngx_http_request_t *r, ngx_http_access_token_ctx_t *ctx);
static ngx_int_t ngx_http_access_token_is_invalid_conf(ngx_http_access_token_conf_t *conf);
static ngx_int_t ngx_http_access_token_build_plain_text(ngx_http_request_t *r, ngx_http_access_token_ctx_t *ctx, ngx_str_t *text,ngx_str_t *sec);
static ngx_command_t ngx_http_access_token_commands[] = {
{
ngx_string("access_token_access_key"),
NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_TAKE1,
ngx_conf_set_str_slot,
NGX_HTTP_LOC_CONF_OFFSET,
offsetof(ngx_http_access_token_conf_t, access_key),
NULL
},
{
ngx_string("access_token_secret"),
NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_TAKE1,
ngx_conf_set_str_slot,
NGX_HTTP_LOC_CONF_OFFSET,
offsetof(ngx_http_access_token_conf_t, secret),
NULL
},
{
ngx_string("allow_second"),
NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_TAKE1,
ngx_conf_set_num_slot,
NGX_HTTP_LOC_CONF_OFFSET,
offsetof(ngx_http_access_token_conf_t, allow_second),
NULL
},
{
ngx_string("access_token_check"),
NGX_HTTP_MAIN_CONF|NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_FLAG,
ngx_conf_set_flag_slot,
NGX_HTTP_LOC_CONF_OFFSET,
offsetof(ngx_http_access_token_conf_t, enable),
NULL
},
ngx_null_command
};
static ngx_http_module_t ngx_http_access_token_module_ctx = {
NULL, /* preconfiguration */
ngx_http_access_token_init, /* postconfiguration */
NULL, /* create main configuration */
NULL, /* init main configuration */
NULL, /* create server configuration */
NULL, /* merge server configuration */
ngx_http_access_token_create_loc_conf, /* create location configuration */
ngx_http_access_token_merge_loc_conf /* merge location configuration */
};
ngx_module_t ngx_http_access_token_module = {
NGX_MODULE_V1,
&ngx_http_access_token_module_ctx, /* module context */
ngx_http_access_token_commands, /* module directives */
NGX_HTTP_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_http_access_token_create_loc_conf(ngx_conf_t *cf)
{
ngx_http_access_token_conf_t *loc_conf;
loc_conf = ngx_pcalloc(cf->pool, sizeof(ngx_http_access_token_conf_t));
if (loc_conf == NULL) {
return NGX_CONF_ERROR;
}
loc_conf->allow_second = NGX_CONF_UNSET;
loc_conf->enable = NGX_CONF_UNSET;
return loc_conf;
}
static char *ngx_http_access_token_merge_loc_conf(ngx_conf_t *cf, void *parent, void *child)
{
ngx_http_access_token_conf_t *prev = parent;
ngx_http_access_token_conf_t *conf = child;
ngx_conf_merge_value(conf->enable, prev->enable, 0);
ngx_conf_merge_value(conf->allow_second,prev->allow_second,300);
return NGX_CONF_OK;
}
static ngx_int_t ngx_http_access_token_init(ngx_conf_t *cf)
{
ngx_http_handler_pt *h;
ngx_http_core_main_conf_t *cmcf;
cmcf = ngx_http_conf_get_module_main_conf(cf, ngx_http_core_module);
h = ngx_array_push(&cmcf->phases[NGX_HTTP_ACCESS_PHASE].handlers);
if (h == NULL) {
return NGX_ERROR;
}
*h = ngx_http_access_token_handler;
return NGX_OK;
}
static ngx_int_t ngx_http_access_token_is_invalid_conf(ngx_http_access_token_conf_t *conf)
{
if (conf->access_key.len > 0 &&
conf->secret.len > 0 &&
conf->allow_second > 0
)
{
return NGX_OK;
}
return NGX_ERROR;
}
static ngx_int_t ngx_http_access_token_is_invalid_args(ngx_http_access_token_ctx_t *ctx)
{
if (ctx->access_key.len > 0 &&
ctx->expires.len > 0 &&
ctx->sig.len > 0)
{
return NGX_OK;
}
return NGX_ERROR;
}
static ngx_int_t ngx_http_access_token_handler(ngx_http_request_t *r)
{
ngx_http_access_token_conf_t *conf = ngx_http_get_module_loc_conf(r, ngx_http_access_token_module);
ngx_http_access_token_ctx_t ctx;
ngx_str_t plain;
u_char *check_sig;
ngx_int_t rc;
time_t t1,t2,base;
double offset = 0;
if (!conf->enable) {
return NGX_DECLINED;
}
if (ngx_http_access_token_is_invalid_conf(conf) != NGX_OK) {
ngx_log_error(NGX_LOG_ERR, r->connection->log, 0,
"Configuration error. You MUST specify access_token_access_key and access_token_secret");
return NGX_HTTP_INTERNAL_SERVER_ERROR;
}
ngx_memzero(&ctx, sizeof(ngx_http_access_token_ctx_t));
ngx_http_access_token_set_args(r, &ctx);
if (ngx_http_access_token_is_invalid_args(&ctx) != NGX_OK) {
ngx_log_error(NGX_LOG_ERR, r->connection->log, 0, "Invalid request arguments:%d %d %d",ctx.access_key.len,ctx.expires.len,ctx.sig.len);
return NGX_HTTP_FORBIDDEN;
}
if (conf->access_key.len != ctx.access_key.len ||
ngx_strncmp(ctx.access_key.data, conf->access_key.data, conf->access_key.len) != 0) {
ngx_log_error(NGX_LOG_ERR, r->connection->log, 0,
"AccessKey does not match. conf:%s, request:%s", conf->access_key.data, ctx.access_key.data);
return NGX_HTTP_FORBIDDEN;
}
base = time(NULL);
t1 = mktime(gmtime(&base));
t2 = mktime(localtime(&base));
offset = difftime(t2,t1);
if(ngx_atoi(ctx.expires.data, ctx.expires.len) < ngx_time()-offset-conf->allow_second) {
ngx_log_error(NGX_LOG_ERR, r->connection->log, 0, "Request has expired:%d",conf->allow_second);
return NGX_HTTP_FORBIDDEN;
}
if(ngx_atoi(ctx.expires.data, ctx.expires.len) > ngx_time()-offset+conf->allow_second) {
ngx_log_error(NGX_LOG_ERR, r->connection->log, 0, "Request in the future:%d",conf->allow_second);
return NGX_HTTP_FORBIDDEN;
}
rc = ngx_http_access_token_build_plain_text(r, &ctx, &plain,&conf->secret);
if (rc != NGX_OK) {
ngx_log_error(NGX_LOG_ERR, r->connection->log, 0, "no memory");
return NGX_HTTP_FORBIDDEN;
}
check_sig = ngx_http_access_token_hmac(r->pool, &conf->secret, &plain);
if(ngx_strcmp(ctx.sig.data, check_sig) == 0) {
return NGX_OK;
} else {
if (ctx.sig.len > 0 && check_sig != NULL) {
ngx_log_error(NGX_LOG_ERR, r->connection->log, 0, "Invalid signature: %s => %s:%s params:%s", plain.data, check_sig, ctx.sig.data ,r->args.data);
}
return NGX_HTTP_FORBIDDEN;
}
return NGX_OK;
}
static void ngx_http_access_token_set_args(ngx_http_request_t *r, ngx_http_access_token_ctx_t *ctx)
{
ngx_int_t i;
ngx_http_variable_value_t *var;
ngx_uint_t key;
ngx_str_t args[3] = {
ngx_string("arg_accesskey"),
ngx_string("arg_expires"),
ngx_string("arg_signature")
};
for (i=0;i<3;i++) {
key = ngx_hash_key_lc(args[i].data, args[i].len);
var = ngx_http_get_variable(r, &args[i], key);
switch (i) {
case 0:
NGX_HTTP_ACCESS_TOKEN_SET_ARG(r, ctx->access_key, var);
break;
case 1:
NGX_HTTP_ACCESS_TOKEN_SET_ARG(r, ctx->expires, var);
break;
case 2:
NGX_HTTP_ACCESS_TOKEN_SET_ARG(r, ctx->sig, var);
break;
}
}
}
static u_char *ngx_http_access_token_hmac(ngx_pool_t *pool, ngx_str_t *key, ngx_str_t *text)
{
u_char *result = NULL;
if ((result = ngx_palloc(pool, 32 + 1)) == NULL) {
return NULL;
}
u_char hash[32];
ngx_md5_t md5;
ngx_md5_init(&md5);
ngx_md5_update(&md5, text->data, text->len);
ngx_md5_final(hash, &md5);
ngx_hex_dump(result, hash, 16);
result[32] =0;
// u_char md[EVP_MAX_MD_SIZE];
// unsigned int md_len;
// const EVP_MD *evp_md;
// u_char *result;
// ngx_str_t src, dst;
//
// evp_md = EVP_sha1();
// HMAC(evp_md, key->data, key->len, text->data, text->len, md, &md_len);
// if ((result = ngx_pcalloc(pool, ngx_base64_encoded_length(md_len) + 1)) == NULL) {
// return NULL;
// }
//
// dst.data = result;
// dst.len = ngx_base64_encoded_length(md_len);
// src.data = md;
// src.len = md_len;
// ngx_encode_base64(&dst, &src);
return result;
}
static ngx_int_t ngx_http_access_token_build_plain_text(ngx_http_request_t *r, ngx_http_access_token_ctx_t *ctx, ngx_str_t *plain,ngx_str_t* sec)
{
u_char *buf, *p;
size_t len;
len = 6 + r->uri.len + ctx->expires.len + sec->len + 1;
if(r->args.len > 50){
len += (r->args.len - 42); //不要末尾的43个字节,加上?符号,一共42个
}
if ((buf = ngx_palloc(r->pool, len + 1)) == NULL) {
return NGX_ERROR;
}
// md5(feiyu/timestamp/appstr/uri?params)
//其中:参数部分,不要: &Signature=848163fe4f560f0c3b91fd5cdc1e20b2 ,共43个字节
// /aaa/version-2.0.0-a.manifest?AccessKey=22&Expires=1492065783&Signature=848163fe4f560f0c3b91fd5cdc1e20b2
p = buf;
p = ngx_cpystrn(p, (u_char*)"feiyu/", 7);
p = ngx_cpystrn(p, ctx->expires.data, ctx->expires.len + 1);
p = ngx_cpystrn(p, (u_char*)"/", 2);
p = ngx_cpystrn(p, sec->data, sec->len + 1);
p = ngx_cpystrn(p, r->uri.data, r->uri.len + 1);
if(r->args.len > 50){
p = ngx_cpystrn(p, (u_char*)"?", 2);
p = ngx_cpystrn(p, r->args.data,r->args.len - 42);
p = ngx_cpystrn(p, (u_char*)"", 1);
}
plain->data = buf;
plain->len = len;
return NGX_OK;
}
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化