[PATCH] Add $upstream_cache_key

Maxim Dounin mdounin at mdounin.ru
Thu May 23 16:59:17 UTC 2024


Hello!

On Thu, May 23, 2024 at 10:52:13AM +0100, Kirill A. Korinsky wrote:

> Greetings,
> 
> Here is a patch that exposes the constructed cache key as
> $upstream_cache_key variable.
> 
> Sometimes it's quite useful when debugging complicated setups and fighting
> some typos.

Thanks for the patch.

I have no objections in general, see below for some minor 
comments.

> 
> index 2ce9f2114..561108681 100644
> --- src/http/ngx_http_upstream.c
> +++ src/http/ngx_http_upstream.c
> @@ -23,6 +23,8 @@ static ngx_int_t ngx_http_upstream_cache_check_range(ngx_http_request_t *r,
>      ngx_http_upstream_t *u);
>  static ngx_int_t ngx_http_upstream_cache_status(ngx_http_request_t *r,
>      ngx_http_variable_value_t *v, uintptr_t data);
> +static ngx_int_t ngx_http_upstream_cache_key(ngx_http_request_t *r,
> +    ngx_http_variable_value_t *v, uintptr_t data);
>  static ngx_int_t ngx_http_upstream_cache_last_modified(ngx_http_request_t *r,
>      ngx_http_variable_value_t *v, uintptr_t data);
>  static ngx_int_t ngx_http_upstream_cache_etag(ngx_http_request_t *r,
> @@ -414,6 +416,10 @@ static ngx_http_variable_t  ngx_http_upstream_vars[] = {
>        ngx_http_upstream_cache_status, 0,
>        NGX_HTTP_VAR_NOCACHEABLE, 0 },
>  
> +    { ngx_string("upstream_cache_key"), NULL,
> +      ngx_http_upstream_cache_key, 0,
> +      NGX_HTTP_VAR_NOCACHEABLE, 0 },
> +
>      { ngx_string("upstream_cache_last_modified"), NULL,
>        ngx_http_upstream_cache_last_modified, 0,
>        NGX_HTTP_VAR_NOCACHEABLE|NGX_HTTP_VAR_NOHASH, 0 },
> @@ -5990,6 +5996,47 @@ ngx_http_upstream_cache_status(ngx_http_request_t *r,
>  }
>  
>  
> +static ngx_int_t
> +ngx_http_upstream_cache_key(ngx_http_request_t *r,
> +    ngx_http_variable_value_t *v, uintptr_t data)
> +{
> +    size_t             len;
> +    ngx_uint_t         i;
> +    ngx_str_t         *key;

Nitpicking: wrong variables order, should be sorted by type length.

> +    ngx_http_cache_t  *c;
> +
> +    if (r->cache == NULL || r->cache->keys.nelts == 0) {
> +        v->not_found = 1;
> +        return NGX_OK;
> +    }
> +
> +    c = r->cache;
> +
> +    key = c->keys.elts;
> +    len = 0;
> +    for (i = 0; i < c->keys.nelts; i++) {
> +        len += key[i].len;
> +    }
> +
> +    v->data = ngx_pcalloc(r->pool, len);

There is no need to use ngx_pcalloc() here: the memory is anyway 
to be written to, so just normal ngx_palloc(), without "c", would 
be enough.  Further, given the result is a string, no alignment is 
needed, so ngx_pnalloc() would be appropriate.

> +    if (v->data == 0) {

In [free]nginx code, NULL is used in pointer contexts.

> +        return NGX_ERROR;
> +    }
> +
> +    v->len = 0;
> +    for (i = 0; i < c->keys.nelts; i++) {
> +        memcpy(v->data + v->len, key[i].data, key[i].len);
> +        v->len += key[i].len;

Usual approach for such loops is doing something like

    p = ngx_cpymem(p, key[i].data, key[i].len);

This avoids an extra addition per iteration, and generally easier 
to read.

(Also, using memcpy() directly is discouraged, it should be either 
ngx_memcpy() or ngx_cpymem().)

> +    }
> +
> +    v->valid = 1;
> +    v->no_cacheable = 0;
> +    v->not_found = 0;
> +
> +    return NGX_OK;
> +}
> +
> +
>  static ngx_int_t
>  ngx_http_upstream_cache_last_modified(ngx_http_request_t *r,
>      ngx_http_variable_value_t *v, uintptr_t data)
> 

Patch with the above comments incorporated, please take a look:

# HG changeset patch
# User Kirill A. Korinsky <kirill at korins.ky>
# Date 1716479312 -10800
#      Thu May 23 18:48:32 2024 +0300
# Node ID bd920ccd6f1aa43fcb40507ba463e44548ca4676
# Parent  46ecad404a296042c0088e699f275a92758e5ab9
Upstream: $upstream_cache_key variable.

diff --git a/src/http/ngx_http_upstream.c b/src/http/ngx_http_upstream.c
--- a/src/http/ngx_http_upstream.c
+++ b/src/http/ngx_http_upstream.c
@@ -23,6 +23,8 @@ static ngx_int_t ngx_http_upstream_cache
     ngx_http_upstream_t *u);
 static ngx_int_t ngx_http_upstream_cache_status(ngx_http_request_t *r,
     ngx_http_variable_value_t *v, uintptr_t data);
+static ngx_int_t ngx_http_upstream_cache_key(ngx_http_request_t *r,
+    ngx_http_variable_value_t *v, uintptr_t data);
 static ngx_int_t ngx_http_upstream_cache_last_modified(ngx_http_request_t *r,
     ngx_http_variable_value_t *v, uintptr_t data);
 static ngx_int_t ngx_http_upstream_cache_etag(ngx_http_request_t *r,
@@ -414,6 +416,10 @@ static ngx_http_variable_t  ngx_http_ups
       ngx_http_upstream_cache_status, 0,
       NGX_HTTP_VAR_NOCACHEABLE, 0 },
 
+    { ngx_string("upstream_cache_key"), NULL,
+      ngx_http_upstream_cache_key, 0,
+      NGX_HTTP_VAR_NOCACHEABLE, 0 },
+
     { ngx_string("upstream_cache_last_modified"), NULL,
       ngx_http_upstream_cache_last_modified, 0,
       NGX_HTTP_VAR_NOCACHEABLE|NGX_HTTP_VAR_NOHASH, 0 },
@@ -6004,6 +6010,49 @@ ngx_http_upstream_cache_status(ngx_http_
 
 
 static ngx_int_t
+ngx_http_upstream_cache_key(ngx_http_request_t *r,
+    ngx_http_variable_value_t *v, uintptr_t data)
+{
+    u_char            *p;
+    size_t             len;
+    ngx_str_t         *key;
+    ngx_uint_t         i;
+    ngx_http_cache_t  *c;
+
+    if (r->cache == NULL || r->cache->keys.nelts == 0) {
+        v->not_found = 1;
+        return NGX_OK;
+    }
+
+    c = r->cache;
+
+    len = 0;
+    key = c->keys.elts;
+
+    for (i = 0; i < c->keys.nelts; i++) {
+        len += key[i].len;
+    }
+
+    p = ngx_pnalloc(r->pool, len);
+    if (p == NULL) {
+        return NGX_ERROR;
+    }
+
+    v->len = len;
+    v->valid = 1;
+    v->no_cacheable = 0;
+    v->not_found = 0;
+    v->data = p;
+
+    for (i = 0; i < c->keys.nelts; i++) {
+        p = ngx_cpymem(p, key[i].data, key[i].len);
+    }
+
+    return NGX_OK;
+}
+
+
+static ngx_int_t
 ngx_http_upstream_cache_last_modified(ngx_http_request_t *r,
     ngx_http_variable_value_t *v, uintptr_t data)
 {


-- 
Maxim Dounin
http://mdounin.ru/



More information about the nginx mailing list