mirror of
https://github.com/nginx/nginx.git
synced 2024-12-16 14:19:00 +08:00
822fe46934
New directives: proxy_cache_lock on/off, proxy_cache_lock_timeout. With proxy_cache_lock set to on, only one request will be allowed to go to upstream for a particular cache item. Others will wait for a response to appear in cache (or cache lock released) up to proxy_cache_lock_timeout. Waiting requests will recheck if they have cached response ready (or are allowed to run) every 500ms. Note: we intentionally don't intercept NGX_DECLINED possibly returned by ngx_http_file_cache_read(). This needs more work (possibly safe, but needs further investigation). Anyway, it's exceptional situation. Note: probably there should be a way to disable caching of responses if there is already one request fetching resource to cache (without waiting at all). Two possible ways include another cache lock option ("no_cache") or using proxy_no_cache with some supplied variable. Note: probably there should be a way to lock updating requests as well. For now "proxy_cache_use_stale updating" is available.
159 lines
4.9 KiB
C
159 lines
4.9 KiB
C
|
|
/*
|
|
* Copyright (C) Igor Sysoev
|
|
*/
|
|
|
|
|
|
#ifndef _NGX_HTTP_CACHE_H_INCLUDED_
|
|
#define _NGX_HTTP_CACHE_H_INCLUDED_
|
|
|
|
|
|
#include <ngx_config.h>
|
|
#include <ngx_core.h>
|
|
#include <ngx_http.h>
|
|
|
|
|
|
#define NGX_HTTP_CACHE_MISS 1
|
|
#define NGX_HTTP_CACHE_BYPASS 2
|
|
#define NGX_HTTP_CACHE_EXPIRED 3
|
|
#define NGX_HTTP_CACHE_STALE 4
|
|
#define NGX_HTTP_CACHE_UPDATING 5
|
|
#define NGX_HTTP_CACHE_HIT 6
|
|
#define NGX_HTTP_CACHE_SCARCE 7
|
|
|
|
#define NGX_HTTP_CACHE_KEY_LEN 16
|
|
|
|
|
|
typedef struct {
|
|
ngx_uint_t status;
|
|
time_t valid;
|
|
} ngx_http_cache_valid_t;
|
|
|
|
|
|
typedef struct {
|
|
ngx_rbtree_node_t node;
|
|
ngx_queue_t queue;
|
|
|
|
u_char key[NGX_HTTP_CACHE_KEY_LEN
|
|
- sizeof(ngx_rbtree_key_t)];
|
|
|
|
unsigned count:20;
|
|
unsigned uses:10;
|
|
unsigned valid_msec:10;
|
|
unsigned error:10;
|
|
unsigned exists:1;
|
|
unsigned updating:1;
|
|
unsigned deleting:1;
|
|
/* 11 unused bits */
|
|
|
|
ngx_file_uniq_t uniq;
|
|
time_t expire;
|
|
time_t valid_sec;
|
|
size_t body_start;
|
|
off_t fs_size;
|
|
} ngx_http_file_cache_node_t;
|
|
|
|
|
|
struct ngx_http_cache_s {
|
|
ngx_file_t file;
|
|
ngx_array_t keys;
|
|
uint32_t crc32;
|
|
u_char key[NGX_HTTP_CACHE_KEY_LEN];
|
|
|
|
ngx_file_uniq_t uniq;
|
|
time_t valid_sec;
|
|
time_t last_modified;
|
|
time_t date;
|
|
|
|
size_t header_start;
|
|
size_t body_start;
|
|
off_t length;
|
|
off_t fs_size;
|
|
|
|
ngx_uint_t min_uses;
|
|
ngx_uint_t error;
|
|
ngx_uint_t valid_msec;
|
|
|
|
ngx_buf_t *buf;
|
|
|
|
ngx_http_file_cache_t *file_cache;
|
|
ngx_http_file_cache_node_t *node;
|
|
|
|
ngx_msec_t lock_timeout;
|
|
ngx_msec_t wait_time;
|
|
|
|
ngx_event_t wait_event;
|
|
|
|
unsigned lock:1;
|
|
unsigned waiting:1;
|
|
|
|
unsigned updated:1;
|
|
unsigned updating:1;
|
|
unsigned exists:1;
|
|
unsigned temp_file:1;
|
|
};
|
|
|
|
|
|
typedef struct {
|
|
time_t valid_sec;
|
|
time_t last_modified;
|
|
time_t date;
|
|
uint32_t crc32;
|
|
u_short valid_msec;
|
|
u_short header_start;
|
|
u_short body_start;
|
|
} ngx_http_file_cache_header_t;
|
|
|
|
|
|
typedef struct {
|
|
ngx_rbtree_t rbtree;
|
|
ngx_rbtree_node_t sentinel;
|
|
ngx_queue_t queue;
|
|
ngx_atomic_t cold;
|
|
ngx_atomic_t loading;
|
|
off_t size;
|
|
} ngx_http_file_cache_sh_t;
|
|
|
|
|
|
struct ngx_http_file_cache_s {
|
|
ngx_http_file_cache_sh_t *sh;
|
|
ngx_slab_pool_t *shpool;
|
|
|
|
ngx_path_t *path;
|
|
|
|
off_t max_size;
|
|
size_t bsize;
|
|
|
|
time_t inactive;
|
|
|
|
ngx_uint_t files;
|
|
ngx_uint_t loader_files;
|
|
ngx_msec_t last;
|
|
ngx_msec_t loader_sleep;
|
|
ngx_msec_t loader_threshold;
|
|
|
|
ngx_shm_zone_t *shm_zone;
|
|
};
|
|
|
|
|
|
ngx_int_t ngx_http_file_cache_new(ngx_http_request_t *r);
|
|
ngx_int_t ngx_http_file_cache_create(ngx_http_request_t *r);
|
|
void ngx_http_file_cache_create_key(ngx_http_request_t *r);
|
|
ngx_int_t ngx_http_file_cache_open(ngx_http_request_t *r);
|
|
void ngx_http_file_cache_set_header(ngx_http_request_t *r, u_char *buf);
|
|
void ngx_http_file_cache_update(ngx_http_request_t *r, ngx_temp_file_t *tf);
|
|
ngx_int_t ngx_http_cache_send(ngx_http_request_t *);
|
|
void ngx_http_file_cache_free(ngx_http_cache_t *c, ngx_temp_file_t *tf);
|
|
time_t ngx_http_file_cache_valid(ngx_array_t *cache_valid, ngx_uint_t status);
|
|
|
|
char *ngx_http_file_cache_set_slot(ngx_conf_t *cf, ngx_command_t *cmd,
|
|
void *conf);
|
|
char *ngx_http_file_cache_valid_set_slot(ngx_conf_t *cf, ngx_command_t *cmd,
|
|
void *conf);
|
|
|
|
|
|
extern ngx_str_t ngx_http_cache_status[];
|
|
|
|
|
|
#endif /* _NGX_HTTP_CACHE_H_INCLUDED_ */
|