| |
| /* |
| * Copyright (C) Igor Sysoev |
| */ |
| |
| |
| #include <ngx_config.h> |
| #include <ngx_core.h> |
| |
| |
| static ngx_uint_t ngx_temp_number; |
| static ngx_uint_t ngx_random; |
| |
| |
| int ngx_write_chain_to_temp_file(ngx_temp_file_t *tf, ngx_chain_t *chain) |
| { |
| int rc; |
| |
| if (tf->file.fd == NGX_INVALID_FILE) { |
| rc = ngx_create_temp_file(&tf->file, tf->path, tf->pool, |
| tf->persistent); |
| |
| if (rc == NGX_ERROR || rc == NGX_AGAIN) { |
| return rc; |
| } |
| |
| if (!tf->persistent && tf->warn) { |
| ngx_log_error(NGX_LOG_WARN, tf->file.log, 0, tf->warn); |
| } |
| } |
| |
| return ngx_write_chain_to_file(&tf->file, chain, tf->offset, tf->pool); |
| } |
| |
| |
| int ngx_create_temp_file(ngx_file_t *file, ngx_path_t *path, |
| ngx_pool_t *pool, int persistent) |
| { |
| int num; |
| ngx_err_t err; |
| |
| file->name.len = path->name.len + 1 + path->len + 10; |
| |
| ngx_test_null(file->name.data, ngx_palloc(pool, file->name.len + 1), |
| NGX_ERROR); |
| |
| #if 0 |
| for (i = 0; i < file->name.len; i++) { |
| file->name.data[i] = 'X'; |
| } |
| #endif |
| |
| ngx_memcpy(file->name.data, path->name.data, path->name.len); |
| |
| num = ngx_next_temp_number(0); |
| |
| for ( ;; ) { |
| ngx_sprintf(file->name.data + path->name.len + 1 + path->len, |
| "%010ud%Z", num); |
| |
| ngx_create_hashed_filename(file, path); |
| |
| #if 1 |
| file->fd = ngx_open_tempfile(file->name.data, persistent); |
| #else |
| file->fd = ngx_open_tempfile(file->name.data, 1); |
| #endif |
| |
| ngx_log_debug1(NGX_LOG_DEBUG_CORE, file->log, 0, |
| "temp fd:%d", file->fd); |
| |
| if (file->fd != NGX_INVALID_FILE) { |
| return NGX_OK; |
| } |
| |
| err = ngx_errno; |
| |
| if (err == NGX_EEXIST) { |
| num = ngx_next_temp_number(1); |
| continue; |
| } |
| |
| if ((path->level[0] == 0) |
| || (err != NGX_ENOENT |
| #if (NGX_WIN32) |
| && err != NGX_ENOTDIR |
| #endif |
| )) { |
| ngx_log_error(NGX_LOG_CRIT, file->log, err, |
| ngx_open_tempfile_n " \"%s\" failed", |
| file->name.data); |
| return NGX_ERROR; |
| } |
| |
| if (ngx_create_path(file, path) == NGX_ERROR) { |
| return NGX_ERROR; |
| } |
| } |
| } |
| |
| |
| void ngx_create_hashed_filename(ngx_file_t *file, ngx_path_t *path) |
| { |
| int i, name, pos; |
| size_t level; |
| |
| name = file->name.len; |
| pos = path->name.len + 1; |
| |
| file->name.data[path->name.len + path->len] = '/'; |
| |
| for (i = 0; i < 3; i++) { |
| level = path->level[i]; |
| |
| if (level == 0) { |
| break; |
| } |
| |
| name -= level; |
| file->name.data[pos - 1] = '/'; |
| ngx_memcpy(&file->name.data[pos], &file->name.data[name], level); |
| pos += level + 1; |
| } |
| |
| ngx_log_debug1(NGX_LOG_DEBUG_CORE, file->log, 0, |
| "hashed path: %s", file->name.data); |
| } |
| |
| |
| int ngx_create_path(ngx_file_t *file, ngx_path_t *path) |
| { |
| int i, pos; |
| ngx_err_t err; |
| |
| pos = path->name.len; |
| |
| for (i = 0; i < 3; i++) { |
| if (path->level[i] == 0) { |
| break; |
| } |
| |
| pos += path->level[i] + 1; |
| |
| file->name.data[pos] = '\0'; |
| |
| ngx_log_debug1(NGX_LOG_DEBUG_CORE, file->log, 0, |
| "temp file: \"%s\"", file->name.data); |
| |
| if (ngx_create_dir(file->name.data) == NGX_FILE_ERROR) { |
| err = ngx_errno; |
| if (err != NGX_EEXIST) { |
| ngx_log_error(NGX_LOG_CRIT, file->log, err, |
| ngx_create_dir_n " \"%s\" failed", |
| file->name.data); |
| return NGX_ERROR; |
| } |
| } |
| |
| file->name.data[pos] = '/'; |
| } |
| |
| return NGX_OK; |
| } |
| |
| |
| void ngx_init_temp_number() |
| { |
| ngx_random = 0; |
| |
| ngx_temp_number = ngx_random; |
| |
| while (ngx_random < 10000) { |
| ngx_random = 123456; |
| } |
| } |
| |
| |
| ngx_uint_t ngx_next_temp_number(ngx_uint_t collision) |
| { |
| if (collision) { |
| ngx_temp_number += ngx_random; |
| } |
| |
| return ngx_temp_number++; |
| } |
| |
| |
| char *ngx_conf_set_path_slot(ngx_conf_t *cf, ngx_command_t *cmd, void *conf) |
| { |
| char *p = conf; |
| |
| ngx_int_t level; |
| ngx_uint_t i, n; |
| ngx_str_t *value; |
| ngx_path_t *path, **pp; |
| |
| pp = (ngx_path_t **) (p + cmd->offset); |
| |
| if (*pp) { |
| return "is duplicate"; |
| } |
| |
| /* TODO: check duplicate in cf->cycle->pathes */ |
| |
| ngx_test_null(path, ngx_pcalloc(cf->pool, sizeof(ngx_path_t)), |
| NGX_CONF_ERROR); |
| |
| *pp = path; |
| |
| ngx_test_null(pp, ngx_push_array(&cf->cycle->pathes), NGX_CONF_ERROR); |
| *pp = path; |
| |
| value = (ngx_str_t *) cf->args->elts; |
| |
| path->name = value[1]; |
| |
| path->len = 0; |
| |
| for (i = 0, n = 2; n < cf->args->nelts; i++, n++) { |
| level = ngx_atoi(value[n].data, value[n].len); |
| if (level == NGX_ERROR || level == 0) { |
| return "invalid value"; |
| } |
| |
| path->level[i] = level; |
| path->len += level + 1; |
| } |
| |
| while (i < 3) { |
| path->level[i++] = 0; |
| } |
| |
| path->gc_handler = (ngx_gc_handler_pt) cmd->post; |
| |
| return NGX_CONF_OK; |
| } |