2020-11-16 14:56:09 +00:00
|
|
|
varnishtest "Vary support"
|
|
|
|
|
2020-12-15 16:13:39 +00:00
|
|
|
#REQUIRE_VERSION=2.4
|
2020-11-16 14:56:09 +00:00
|
|
|
|
|
|
|
feature ignore_unknown_macro
|
|
|
|
|
|
|
|
server s1 {
|
2020-12-29 11:43:53 +00:00
|
|
|
# Response varying on "accept-encoding" with
|
|
|
|
# an unacceptable content-encoding
|
2020-11-16 14:56:09 +00:00
|
|
|
rxreq
|
|
|
|
expect req.url == "/accept-encoding"
|
2020-12-23 17:13:50 +00:00
|
|
|
txresp -hdr "Content-Encoding: gzip" \
|
2020-11-16 14:56:09 +00:00
|
|
|
-hdr "Vary: accept-encoding" \
|
2020-12-23 17:13:50 +00:00
|
|
|
-hdr "Cache-Control: max-age=5" \
|
|
|
|
-bodylen 45
|
2020-11-16 14:56:09 +00:00
|
|
|
|
|
|
|
# Response varying on "accept-encoding"
|
|
|
|
rxreq
|
|
|
|
expect req.url == "/accept-encoding"
|
2020-12-29 11:43:53 +00:00
|
|
|
txresp -hdr "Content-Encoding: gzip" \
|
|
|
|
-hdr "Vary: accept-encoding" \
|
|
|
|
-hdr "Cache-Control: max-age=5" \
|
|
|
|
-bodylen 45
|
|
|
|
|
|
|
|
# Response varying on "accept-encoding" with
|
|
|
|
# no content-encoding
|
|
|
|
rxreq
|
|
|
|
expect req.url == "/accept-encoding"
|
2020-12-23 17:13:50 +00:00
|
|
|
txresp -hdr "Content-Type: text/plain" \
|
2020-11-16 14:56:09 +00:00
|
|
|
-hdr "Vary: accept-encoding" \
|
2020-12-23 17:13:50 +00:00
|
|
|
-hdr "Cache-Control: max-age=5" \
|
|
|
|
-bodylen 48
|
2020-11-16 14:56:09 +00:00
|
|
|
|
|
|
|
# Response varying on "accept-encoding" but having two different encodings
|
|
|
|
rxreq
|
|
|
|
expect req.url == "/accept-encoding-multiple"
|
2020-12-23 17:13:50 +00:00
|
|
|
txresp -hdr "Vary: accept-encoding" \
|
|
|
|
-hdr "Cache-Control: max-age=5" \
|
|
|
|
-bodylen 51
|
|
|
|
|
2020-11-16 14:56:09 +00:00
|
|
|
|
|
|
|
# Unmanaged vary
|
|
|
|
rxreq
|
|
|
|
expect req.url == "/unmanaged"
|
2020-12-23 17:13:50 +00:00
|
|
|
txresp -hdr "Vary: accept-encoding,unmanaged" \
|
|
|
|
-hdr "Cache-Control: max-age=5" \
|
|
|
|
-bodylen 51
|
|
|
|
|
2020-11-16 14:56:09 +00:00
|
|
|
|
|
|
|
rxreq
|
|
|
|
expect req.url == "/unmanaged"
|
2020-12-23 17:13:50 +00:00
|
|
|
txresp -hdr "Vary: accept-encoding,unmanaged" \
|
|
|
|
-hdr "Cache-Control: max-age=5" \
|
|
|
|
-bodylen 51
|
|
|
|
|
2020-11-16 14:56:09 +00:00
|
|
|
|
|
|
|
|
|
|
|
# Mixed Vary (Accept-Encoding + Referer)
|
|
|
|
rxreq
|
|
|
|
expect req.url == "/referer-accept-encoding"
|
2020-12-23 17:13:50 +00:00
|
|
|
txresp -hdr "Vary: accept-encoding,referer" \
|
|
|
|
-hdr "Cache-Control: max-age=5" \
|
|
|
|
-hdr "Content-Encoding: gzip" \
|
|
|
|
-bodylen 51
|
2020-11-16 14:56:09 +00:00
|
|
|
|
|
|
|
rxreq
|
|
|
|
expect req.url == "/referer-accept-encoding"
|
2020-12-23 17:13:50 +00:00
|
|
|
txresp -hdr "Vary: referer,accept-encoding" \
|
|
|
|
-hdr "Cache-Control: max-age=5" \
|
2020-12-29 11:43:53 +00:00
|
|
|
-hdr "Content-Encoding: br" \
|
2020-12-23 17:13:50 +00:00
|
|
|
-bodylen 54
|
2020-11-16 14:56:09 +00:00
|
|
|
|
|
|
|
rxreq
|
|
|
|
expect req.url == "/referer-accept-encoding"
|
2020-12-23 17:13:50 +00:00
|
|
|
txresp -hdr "Vary: referer,accept-encoding" \
|
|
|
|
-hdr "Cache-Control: max-age=5" \
|
|
|
|
-hdr "Content-Encoding: gzip" \
|
|
|
|
-bodylen 57
|
2020-11-16 14:56:09 +00:00
|
|
|
|
2020-12-23 17:13:46 +00:00
|
|
|
# Multiple Accept-Encoding headers
|
|
|
|
rxreq
|
|
|
|
expect req.url == "/multiple_headers"
|
|
|
|
txresp -hdr "Vary: accept-encoding" \
|
2020-12-23 17:13:50 +00:00
|
|
|
-hdr "Cache-Control: max-age=5" \
|
2020-12-29 11:43:53 +00:00
|
|
|
-hdr "Content-Encoding: br" \
|
2020-12-23 17:13:50 +00:00
|
|
|
-bodylen 155
|
2020-12-23 17:13:46 +00:00
|
|
|
|
|
|
|
rxreq
|
|
|
|
expect req.url == "/multiple_headers"
|
|
|
|
txresp -hdr "Vary: accept-encoding" \
|
2020-12-23 17:13:50 +00:00
|
|
|
-hdr "Cache-Control: max-age=5" \
|
2020-12-29 11:43:53 +00:00
|
|
|
-hdr "Content-Encoding: br" \
|
2020-12-23 17:13:50 +00:00
|
|
|
-bodylen 166
|
2020-12-23 17:13:46 +00:00
|
|
|
|
|
|
|
|
|
|
|
# Too many Accept-Encoding values (we will not cache responses with more than 16 encodings)
|
|
|
|
rxreq
|
|
|
|
expect req.url == "/too_many_encodings"
|
|
|
|
txresp -hdr "Vary: accept-encoding" \
|
2020-12-23 17:13:50 +00:00
|
|
|
-hdr "Cache-Control: max-age=5" \
|
|
|
|
-hdr "Content-Encoding: gzip" \
|
|
|
|
-bodylen 177
|
2020-12-23 17:13:46 +00:00
|
|
|
|
|
|
|
rxreq
|
|
|
|
expect req.url == "/too_many_encodings"
|
|
|
|
txresp -hdr "Vary: accept-encoding" \
|
2020-12-23 17:13:50 +00:00
|
|
|
-hdr "Cache-Control: max-age=5" \
|
|
|
|
-hdr "Content-Encoding: gzip" \
|
|
|
|
-bodylen 188
|
2020-12-23 17:13:46 +00:00
|
|
|
|
2021-06-18 13:09:28 +00:00
|
|
|
rxreq
|
|
|
|
expect req.url == "/empty-vs-missing"
|
|
|
|
txresp -hdr "Content-Encoding: gzip" \
|
|
|
|
-hdr "Vary: accept-encoding" \
|
|
|
|
-hdr "Cache-Control: max-age=5" \
|
|
|
|
-bodylen 234
|
2020-11-16 14:56:10 +00:00
|
|
|
|
2021-06-18 13:09:28 +00:00
|
|
|
rxreq
|
|
|
|
expect req.url == "/empty-vs-missing"
|
|
|
|
txresp -hdr "Vary: accept-encoding" \
|
|
|
|
-hdr "Cache-Control: max-age=5" \
|
|
|
|
-bodylen 256
|
2020-11-16 14:56:10 +00:00
|
|
|
} -start
|
|
|
|
|
|
|
|
server s2 {
|
|
|
|
# Responses that should not be cached
|
|
|
|
rxreq
|
|
|
|
expect req.url == "/no_vary_support"
|
2020-12-23 17:13:50 +00:00
|
|
|
txresp -hdr "Vary: accept-encoding" \
|
|
|
|
-hdr "Cache-Control: max-age=5" \
|
|
|
|
-bodylen 57
|
2020-11-16 14:56:10 +00:00
|
|
|
|
|
|
|
rxreq
|
|
|
|
expect req.url == "/no_vary_support"
|
2020-12-23 17:13:50 +00:00
|
|
|
txresp -hdr "Vary: accept-encoding" \
|
|
|
|
-hdr "Cache-Control: max-age=5" \
|
|
|
|
-bodylen 57
|
2020-11-16 14:56:09 +00:00
|
|
|
} -start
|
|
|
|
|
|
|
|
haproxy h1 -conf {
|
2021-05-09 12:41:41 +00:00
|
|
|
global
|
|
|
|
# WT: limit false-positives causing "HTTP header incomplete" due to
|
|
|
|
# idle server connections being randomly used and randomly expiring
|
|
|
|
# under us.
|
|
|
|
tune.idle-pool.shared off
|
|
|
|
|
2020-11-16 14:56:09 +00:00
|
|
|
defaults
|
|
|
|
mode http
|
2021-11-18 16:46:22 +00:00
|
|
|
timeout connect "${HAPROXY_TEST_TIMEOUT-5s}"
|
|
|
|
timeout client "${HAPROXY_TEST_TIMEOUT-5s}"
|
|
|
|
timeout server "${HAPROXY_TEST_TIMEOUT-5s}"
|
2020-11-16 14:56:09 +00:00
|
|
|
|
|
|
|
frontend fe
|
|
|
|
bind "fd@${fe}"
|
2020-11-16 14:56:10 +00:00
|
|
|
use_backend no_vary_be if { path_beg /no_vary_support }
|
2020-11-16 14:56:09 +00:00
|
|
|
default_backend test
|
|
|
|
|
|
|
|
backend test
|
|
|
|
http-request cache-use my_cache
|
|
|
|
server www ${s1_addr}:${s1_port}
|
|
|
|
http-response cache-store my_cache
|
|
|
|
http-response set-header X-Cache-Hit %[res.cache_hit]
|
|
|
|
|
2020-11-16 14:56:10 +00:00
|
|
|
backend no_vary_be
|
|
|
|
http-request cache-use no_vary_cache
|
|
|
|
server www ${s2_addr}:${s2_port}
|
|
|
|
http-response cache-store no_vary_cache
|
|
|
|
http-response set-header X-Cache-Hit %[res.cache_hit]
|
|
|
|
|
2020-11-16 14:56:09 +00:00
|
|
|
cache my_cache
|
|
|
|
total-max-size 3
|
|
|
|
max-age 20
|
|
|
|
max-object-size 3072
|
2020-12-23 17:13:53 +00:00
|
|
|
process-vary on
|
2020-11-16 14:56:10 +00:00
|
|
|
|
|
|
|
cache no_vary_cache
|
|
|
|
total-max-size 3
|
|
|
|
max-age 20
|
|
|
|
max-object-size 3072
|
2020-12-23 17:13:53 +00:00
|
|
|
process-vary off
|
2020-11-16 14:56:09 +00:00
|
|
|
} -start
|
|
|
|
|
|
|
|
|
|
|
|
client c1 -connect ${h1_fe_sock} {
|
|
|
|
# Accept-Encoding Vary
|
|
|
|
txreq -url "/accept-encoding" -hdr "Accept-Encoding: first_value"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
2020-12-23 17:13:50 +00:00
|
|
|
expect resp.http.content-encoding == "gzip"
|
2020-11-16 14:56:09 +00:00
|
|
|
expect resp.bodylen == 45
|
|
|
|
|
2020-12-29 11:43:53 +00:00
|
|
|
# The response for the first request had an unacceptable `content-encoding`
|
|
|
|
# which might happen if that's the only thing the server supports, but
|
|
|
|
# we must not cache that and instead defer to the server.
|
|
|
|
txreq -url "/accept-encoding" -hdr "Accept-Encoding: first_value"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.http.content-encoding == "gzip"
|
|
|
|
expect resp.bodylen == 45
|
|
|
|
expect resp.http.X-Cache-Hit == 0
|
|
|
|
|
2020-11-16 14:56:09 +00:00
|
|
|
txreq -url "/accept-encoding" -hdr "Accept-Encoding: second_value"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 48
|
|
|
|
expect resp.http.content-type == "text/plain"
|
|
|
|
expect resp.http.X-Cache-Hit == 0
|
|
|
|
|
2020-12-29 11:43:53 +00:00
|
|
|
# This request matches the cache entry for the request above, despite
|
|
|
|
# matching the `accept-encoding` of the first request because the
|
|
|
|
# request above only has the `identity` encoding which is implicitly
|
2021-01-05 17:10:46 +00:00
|
|
|
# added, unless explicitly forbidden.
|
2020-11-16 14:56:09 +00:00
|
|
|
txreq -url "/accept-encoding" -hdr "Accept-Encoding: first_value"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
2020-12-29 11:43:53 +00:00
|
|
|
expect resp.bodylen == 48
|
|
|
|
expect resp.http.content-type == "text/plain"
|
2020-11-16 14:56:09 +00:00
|
|
|
expect resp.http.X-Cache-Hit == 1
|
|
|
|
|
|
|
|
txreq -url "/accept-encoding" -hdr "Accept-Encoding: second_value"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 48
|
|
|
|
expect resp.http.content-type == "text/plain"
|
|
|
|
expect resp.http.X-Cache-Hit == 1
|
|
|
|
|
2020-11-30 16:06:03 +00:00
|
|
|
# The accept-encoding normalizer function converts the header values
|
|
|
|
# to lower case then calculates the hash of every sub part before
|
|
|
|
# sorting the hashes and xor'ing them (while removing duplicates).
|
2020-11-16 14:56:09 +00:00
|
|
|
txreq -url "/accept-encoding-multiple" -hdr "Accept-Encoding: first,second"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 51
|
|
|
|
expect resp.http.X-Cache-Hit == 0
|
|
|
|
|
|
|
|
txreq -url "/accept-encoding-multiple" -hdr "Accept-Encoding: first,second"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 51
|
|
|
|
expect resp.http.X-Cache-Hit == 1
|
|
|
|
|
|
|
|
txreq -url "/accept-encoding-multiple" -hdr "Accept-Encoding: second,first"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 51
|
|
|
|
expect resp.http.X-Cache-Hit == 1
|
|
|
|
|
2020-11-30 16:06:03 +00:00
|
|
|
txreq -url "/accept-encoding-multiple" -hdr "Accept-Encoding: FirsT,SECOND,first"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 51
|
|
|
|
expect resp.http.X-Cache-Hit == 1
|
|
|
|
|
2020-11-16 14:56:09 +00:00
|
|
|
# Unmanaged vary
|
|
|
|
txreq -url "/unmanaged" -hdr "Accept-Encoding: first_value"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 51
|
|
|
|
expect resp.http.X-Cache-Hit == 0
|
|
|
|
|
|
|
|
txreq -url "/unmanaged" -hdr "Accept-Encoding: first_value"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 51
|
|
|
|
expect resp.http.X-Cache-Hit == 0
|
|
|
|
|
|
|
|
|
|
|
|
# Mixed Vary (Accept-Encoding + Referer)
|
|
|
|
txreq -url "/referer-accept-encoding" \
|
2020-12-29 11:43:53 +00:00
|
|
|
-hdr "Accept-Encoding: br, gzip" \
|
2020-11-16 14:56:09 +00:00
|
|
|
-hdr "Referer: referer"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 51
|
|
|
|
expect resp.http.X-Cache-Hit == 0
|
|
|
|
|
|
|
|
txreq -url "/referer-accept-encoding" \
|
2020-12-29 11:43:53 +00:00
|
|
|
-hdr "Accept-Encoding: br" \
|
2020-11-16 14:56:09 +00:00
|
|
|
-hdr "Referer: other-referer"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 54
|
|
|
|
expect resp.http.X-Cache-Hit == 0
|
|
|
|
|
|
|
|
txreq -url "/referer-accept-encoding" \
|
2020-12-29 11:43:53 +00:00
|
|
|
-hdr "Accept-Encoding: gzip" \
|
2020-11-16 14:56:09 +00:00
|
|
|
-hdr "Referer: other-referer"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 57
|
|
|
|
expect resp.http.X-Cache-Hit == 0
|
|
|
|
|
|
|
|
txreq -url "/referer-accept-encoding" \
|
|
|
|
-hdr "Referer: referer" \
|
2020-12-29 11:43:53 +00:00
|
|
|
-hdr "Accept-Encoding: gzip, br"
|
2020-11-16 14:56:09 +00:00
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 51
|
|
|
|
expect resp.http.X-Cache-Hit == 1
|
|
|
|
|
|
|
|
txreq -url "/referer-accept-encoding" \
|
2020-12-29 11:43:53 +00:00
|
|
|
-hdr "Accept-Encoding: br" \
|
2020-11-16 14:56:09 +00:00
|
|
|
-hdr "Referer: other-referer"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 54
|
|
|
|
expect resp.http.X-Cache-Hit == 1
|
|
|
|
|
|
|
|
txreq -url "/referer-accept-encoding" \
|
2020-12-29 11:43:53 +00:00
|
|
|
-hdr "Accept-Encoding: gzip" \
|
2020-11-16 14:56:09 +00:00
|
|
|
-hdr "Referer: other-referer"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 57
|
|
|
|
expect resp.http.X-Cache-Hit == 1
|
2020-11-16 14:56:10 +00:00
|
|
|
|
2020-12-23 17:13:46 +00:00
|
|
|
|
|
|
|
# Multiple Accept-encoding headers
|
|
|
|
txreq -url "/multiple_headers" \
|
2020-12-29 11:43:53 +00:00
|
|
|
-hdr "Accept-Encoding: gzip" \
|
|
|
|
-hdr "Accept-Encoding: br, deflate"
|
2020-12-23 17:13:46 +00:00
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 155
|
|
|
|
expect resp.http.X-Cache-Hit == 0
|
|
|
|
|
|
|
|
txreq -url "/multiple_headers" \
|
2020-12-29 11:43:53 +00:00
|
|
|
-hdr "Accept-Encoding: deflate" \
|
|
|
|
-hdr "Accept-Encoding: br,gzip"
|
2020-12-23 17:13:46 +00:00
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 155
|
|
|
|
expect resp.http.X-Cache-Hit == 1
|
|
|
|
|
|
|
|
# Should not match a cache entry
|
|
|
|
txreq -url "/multiple_headers" \
|
|
|
|
-hdr "Accept-Encoding: first_encoding"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 166
|
|
|
|
expect resp.http.X-Cache-Hit == 0
|
|
|
|
|
|
|
|
# Too many accept encodings
|
|
|
|
txreq -url "/too_many_encodings" \
|
|
|
|
-hdr "Accept-Encoding: a1,a2,a3,a4,a5,a6,a7,a8,a9,a10,a11,a12,a13,a14,a15,a16,a17"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 177
|
|
|
|
expect resp.http.X-Cache-Hit == 0
|
|
|
|
|
|
|
|
txreq -url "/too_many_encodings" \
|
|
|
|
-hdr "Accept-Encoding: a1,a2,a3,a4,a5,a6,a7,a8,a9,a10,a11,a12,a13,a14,a15,a16,a17"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 188
|
|
|
|
expect resp.http.X-Cache-Hit == 0
|
|
|
|
|
2021-06-18 13:09:28 +00:00
|
|
|
# A missing 'Accept-Encoding' implies that anything is acceptable,
|
|
|
|
# while an empty 'Accept-Encoding' implies nothing is acceptable.
|
|
|
|
|
|
|
|
# Start by caching a gzip response.
|
|
|
|
txreq -url "/empty-vs-missing" -hdr "Accept-Encoding: gzip"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 234
|
|
|
|
expect resp.http.content-encoding == "gzip"
|
|
|
|
expect resp.http.X-Cache-Hit == 0
|
|
|
|
|
|
|
|
# Check that it is cached.
|
|
|
|
txreq -url "/empty-vs-missing" -hdr "Accept-Encoding: gzip"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 234
|
|
|
|
expect resp.http.content-encoding == "gzip"
|
|
|
|
expect resp.http.X-Cache-Hit == 1
|
|
|
|
|
|
|
|
# Check that the cached response is returned when no accept-encoding is
|
|
|
|
# specified.
|
|
|
|
txreq -url "/empty-vs-missing"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 234
|
|
|
|
expect resp.http.content-encoding == "gzip"
|
|
|
|
expect resp.http.X-Cache-Hit == 1
|
|
|
|
|
|
|
|
# Check that the cached response is not returned when an empty
|
|
|
|
# accept-encoding is specified.
|
|
|
|
txreq -url "/empty-vs-missing" -hdr "Accept-Encoding:"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 256
|
|
|
|
expect resp.http.content-encoding == "<undef>"
|
|
|
|
expect resp.http.X-Cache-Hit == 0
|
2020-12-23 17:13:46 +00:00
|
|
|
|
2020-11-30 16:06:03 +00:00
|
|
|
# The following requests are treated by a backend that does not cache
|
2020-11-16 14:56:10 +00:00
|
|
|
# responses containing a Vary header
|
|
|
|
txreq -url "/no_vary_support"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 57
|
|
|
|
expect resp.http.X-Cache-Hit == 0
|
|
|
|
|
|
|
|
txreq -url "/no_vary_support"
|
|
|
|
rxresp
|
|
|
|
expect resp.status == 200
|
|
|
|
expect resp.bodylen == 57
|
|
|
|
expect resp.http.X-Cache-Hit == 0
|
|
|
|
|
|
|
|
|
2020-11-16 14:56:09 +00:00
|
|
|
} -run
|