blob: f3a9439f519a2ebcf286e366bab37b675f7a7d48 [file] [log] [blame] [edit]
#!/bin/sh
test_description='test HTTP 429 Too Many Requests retry logic'
. ./test-lib.sh
. "$TEST_DIRECTORY"/lib-httpd.sh
start_httpd
test_expect_success 'setup test repository' '
test_commit initial &&
git clone --bare . "$HTTPD_DOCUMENT_ROOT_PATH/repo.git" &&
git --git-dir="$HTTPD_DOCUMENT_ROOT_PATH/repo.git" config http.receivepack true
'
# This test suite uses a special HTTP 429 endpoint at /http_429/ that simulates
# rate limiting. The endpoint format is:
# /http_429/<test-context>/<retry-after-value>/<repo-path>
# The http-429.sh script (in t/lib-httpd) returns a 429 response with the
# specified Retry-After header on the first request for each test context,
# then forwards subsequent requests to git-http-backend. Each test context
# is isolated, allowing multiple tests to run independently.
test_expect_success 'HTTP 429 with retries disabled (maxRetries=0) fails immediately' '
# Set maxRetries to 0 (disabled)
test_config http.maxRetries 0 &&
test_config http.retryAfter 1 &&
# Should fail immediately without any retry attempt
test_must_fail git ls-remote "$HTTPD_URL/http_429/retries-disabled/1/repo.git" 2>err &&
# Verify no retry happened (no "waiting" message in stderr)
test_grep ! -i "waiting.*retry" err
'
test_expect_success 'HTTP 429 permanent should fail after max retries' '
# Enable retries with a limit
test_config http.maxRetries 2 &&
# Git should retry but eventually fail when 429 persists
test_must_fail git ls-remote "$HTTPD_URL/http_429/permanent-fail/permanent/repo.git" 2>err
'
test_expect_success 'HTTP 429 with Retry-After is retried and succeeds' '
# Enable retries
test_config http.maxRetries 3 &&
# Git should retry after receiving 429 and eventually succeed
git ls-remote "$HTTPD_URL/http_429/retry-succeeds/1/repo.git" >output 2>err &&
test_grep "refs/heads/" output
'
test_expect_success 'HTTP 429 without Retry-After uses configured default' '
# Enable retries and configure default delay
test_config http.maxRetries 3 &&
test_config http.retryAfter 1 &&
# Git should retry using configured default and succeed
git ls-remote "$HTTPD_URL/http_429/no-retry-after-header/none/repo.git" >output 2>err &&
test_grep "refs/heads/" output
'
test_expect_success 'HTTP 429 retry delays are respected' '
# Enable retries
test_config http.maxRetries 3 &&
# Time the operation - it should take at least 2 seconds due to retry delay
start=$(test-tool date getnanos) &&
git ls-remote "$HTTPD_URL/http_429/retry-delays-respected/2/repo.git" >output 2>err &&
duration=$(test-tool date getnanos $start) &&
# Verify it took at least 2 seconds (allowing some tolerance)
duration_int=${duration%.*} &&
test "$duration_int" -ge 1 &&
test_grep "refs/heads/" output
'
test_expect_success 'HTTP 429 fails immediately if Retry-After exceeds http.maxRetryTime' '
# Configure max retry time to 3 seconds (much less than requested 100)
test_config http.maxRetries 3 &&
test_config http.maxRetryTime 3 &&
# Should fail immediately without waiting
start=$(test-tool date getnanos) &&
test_must_fail git ls-remote "$HTTPD_URL/http_429/retry-after-exceeds-max-time/100/repo.git" 2>err &&
duration=$(test-tool date getnanos $start) &&
# Should fail quickly (less than 2 seconds, no 100 second wait)
duration_int=${duration%.*} &&
test "$duration_int" -lt 2 &&
test_grep "greater than http.maxRetryTime" err
'
test_expect_success 'HTTP 429 fails if configured http.retryAfter exceeds http.maxRetryTime' '
# Test misconfiguration: retryAfter > maxRetryTime
# Configure retryAfter larger than maxRetryTime
test_config http.maxRetries 3 &&
test_config http.retryAfter 100 &&
test_config http.maxRetryTime 5 &&
# Should fail immediately with configuration error
start=$(test-tool date getnanos) &&
test_must_fail git ls-remote "$HTTPD_URL/http_429/config-retry-after-exceeds-max-time/none/repo.git" 2>err &&
duration=$(test-tool date getnanos $start) &&
# Should fail quickly
duration_int=${duration%.*} &&
test "$duration_int" -lt 2 &&
test_grep "configured http.retryAfter.*exceeds.*http.maxRetryTime" err
'
test_expect_success 'HTTP 429 with Retry-After HTTP-date format' '
# Test HTTP-date format (RFC 2822) in Retry-After header
raw=$(test-tool date timestamp now) &&
now="${raw#* -> }" &&
future_time=$((now + 2)) &&
raw=$(test-tool date show:rfc2822 $future_time) &&
future_date="${raw#* -> }" &&
future_date_encoded=$(echo "$future_date" | sed "s/ /%20/g") &&
# Enable retries
test_config http.maxRetries 3 &&
# Git should parse the HTTP-date and retry after the delay
start=$(test-tool date getnanos) &&
git ls-remote "$HTTPD_URL/http_429/http-date-format/$future_date_encoded/repo.git" >output 2>err &&
duration=$(test-tool date getnanos $start) &&
# Should take at least 1 second (allowing tolerance for processing time)
duration_int=${duration%.*} &&
test "$duration_int" -ge 1 &&
test_grep "refs/heads/" output
'
test_expect_success 'HTTP 429 with HTTP-date exceeding maxRetryTime fails immediately' '
raw=$(test-tool date timestamp now) &&
now="${raw#* -> }" &&
future_time=$((now + 200)) &&
raw=$(test-tool date show:rfc2822 $future_time) &&
future_date="${raw#* -> }" &&
future_date_encoded=$(echo "$future_date" | sed "s/ /%20/g") &&
# Configure max retry time much less than the 200 second delay
test_config http.maxRetries 3 &&
test_config http.maxRetryTime 10 &&
# Should fail immediately without waiting 200 seconds
start=$(test-tool date getnanos) &&
test_must_fail git ls-remote "$HTTPD_URL/http_429/http-date-exceeds-max-time/$future_date_encoded/repo.git" 2>err &&
duration=$(test-tool date getnanos $start) &&
# Should fail quickly (not wait 200 seconds)
duration_int=${duration%.*} &&
test "$duration_int" -lt 2 &&
test_grep "http.maxRetryTime" err
'
test_expect_success 'HTTP 429 with past HTTP-date should not wait' '
raw=$(test-tool date timestamp now) &&
now="${raw#* -> }" &&
past_time=$((now - 10)) &&
raw=$(test-tool date show:rfc2822 $past_time) &&
past_date="${raw#* -> }" &&
past_date_encoded=$(echo "$past_date" | sed "s/ /%20/g") &&
# Enable retries
test_config http.maxRetries 3 &&
# Git should retry immediately without waiting
start=$(test-tool date getnanos) &&
git ls-remote "$HTTPD_URL/http_429/past-http-date/$past_date_encoded/repo.git" >output 2>err &&
duration=$(test-tool date getnanos $start) &&
# Should complete quickly (less than 2 seconds)
duration_int=${duration%.*} &&
test "$duration_int" -lt 2 &&
test_grep "refs/heads/" output
'
test_expect_success 'HTTP 429 with invalid Retry-After format uses configured default' '
# Configure default retry-after
test_config http.maxRetries 3 &&
test_config http.retryAfter 1 &&
# Should use configured default (1 second) since header is invalid
start=$(test-tool date getnanos) &&
git ls-remote "$HTTPD_URL/http_429/invalid-retry-after-format/invalid/repo.git" >output 2>err &&
duration=$(test-tool date getnanos $start) &&
# Should take at least 1 second (the configured default)
duration_int=${duration%.*} &&
test "$duration_int" -ge 1 &&
test_grep "refs/heads/" output &&
test_grep "waiting.*retry" err
'
test_expect_success 'HTTP 429 will not be retried without config' '
# Default config means http.maxRetries=0 (retries disabled)
# When 429 is received, it should fail immediately without retry
# Do NOT configure anything - use defaults (http.maxRetries defaults to 0)
# Should fail immediately without retry
test_must_fail git ls-remote "$HTTPD_URL/http_429/no-retry-without-config/1/repo.git" 2>err &&
# Verify no retry happened (no "waiting" message)
test_grep ! -i "waiting.*retry" err &&
# Should get 429 error
test_grep "429" err
'
test_expect_success 'GIT_HTTP_RETRY_AFTER overrides http.retryAfter config' '
# Configure retryAfter to 10 seconds
test_config http.maxRetries 3 &&
test_config http.retryAfter 10 &&
# Override with environment variable to 1 second
start=$(test-tool date getnanos) &&
GIT_HTTP_RETRY_AFTER=1 git ls-remote "$HTTPD_URL/http_429/env-retry-after-override/none/repo.git" >output 2>err &&
duration=$(test-tool date getnanos $start) &&
# Should use env var (1 second), not config (10 seconds)
duration_int=${duration%.*} &&
test "$duration_int" -ge 1 &&
test "$duration_int" -lt 5 &&
test_grep "refs/heads/" output &&
test_grep "waiting.*retry" err
'
test_expect_success 'GIT_HTTP_MAX_RETRIES overrides http.maxRetries config' '
# Configure maxRetries to 0 (disabled)
test_config http.maxRetries 0 &&
test_config http.retryAfter 1 &&
# Override with environment variable to enable retries
GIT_HTTP_MAX_RETRIES=3 git ls-remote "$HTTPD_URL/http_429/env-max-retries-override/1/repo.git" >output 2>err &&
# Should retry (env var enables it despite config saying disabled)
test_grep "refs/heads/" output &&
test_grep "waiting.*retry" err
'
test_expect_success 'GIT_HTTP_MAX_RETRY_TIME overrides http.maxRetryTime config' '
# Configure maxRetryTime to 100 seconds (would accept 50 second delay)
test_config http.maxRetries 3 &&
test_config http.maxRetryTime 100 &&
# Override with environment variable to 10 seconds (should reject 50 second delay)
start=$(test-tool date getnanos) &&
test_must_fail env GIT_HTTP_MAX_RETRY_TIME=10 \
git ls-remote "$HTTPD_URL/http_429/env-max-retry-time-override/50/repo.git" 2>err &&
duration=$(test-tool date getnanos $start) &&
# Should fail quickly (not wait 50 seconds) because env var limits to 10
duration_int=${duration%.*} &&
test "$duration_int" -lt 5 &&
test_grep "greater than http.maxRetryTime" err
'
test_expect_success 'verify normal repository access still works' '
git ls-remote "$HTTPD_URL/smart/repo.git" >output &&
test_grep "refs/heads/" output
'
test_done