diff --git a/CMakeLists.txt b/CMakeLists.txt index ff82bc71644988..50902df2255002 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -30608,7 +30608,6 @@ add_executable(test_core_security_credentials_test ${_gRPC_PROTO_GENS_DIR}/test/core/event_engine/fuzzing_event_engine/fuzzing_event_engine.grpc.pb.cc ${_gRPC_PROTO_GENS_DIR}/test/core/event_engine/fuzzing_event_engine/fuzzing_event_engine.pb.h ${_gRPC_PROTO_GENS_DIR}/test/core/event_engine/fuzzing_event_engine/fuzzing_event_engine.grpc.pb.h - src/core/lib/security/credentials/gcp_service_account_identity/gcp_service_account_identity_credentials.cc test/core/event_engine/event_engine_test_utils.cc test/core/event_engine/fuzzing_event_engine/fuzzing_event_engine.cc test/core/security/credentials_test.cc diff --git a/build_autogenerated.yaml b/build_autogenerated.yaml index 74a7ec764a52a1..b2189eecfa3714 100644 --- a/build_autogenerated.yaml +++ b/build_autogenerated.yaml @@ -19655,7 +19655,6 @@ targets: build: test language: c++ headers: - - src/core/lib/security/credentials/gcp_service_account_identity/gcp_service_account_identity_credentials.h - test/core/event_engine/event_engine_test_utils.h - test/core/event_engine/fuzzing_event_engine/fuzzing_event_engine.h - test/core/test_util/cmdline.h @@ -19670,7 +19669,6 @@ targets: - test/core/test_util/tracer_util.h src: - test/core/event_engine/fuzzing_event_engine/fuzzing_event_engine.proto - - src/core/lib/security/credentials/gcp_service_account_identity/gcp_service_account_identity_credentials.cc - test/core/event_engine/event_engine_test_utils.cc - test/core/event_engine/fuzzing_event_engine/fuzzing_event_engine.cc - test/core/security/credentials_test.cc diff --git a/doc/trace_flags.md b/doc/trace_flags.md index 4ea7129a28f48d..56ad7191781d38 100644 --- a/doc/trace_flags.md +++ b/doc/trace_flags.md @@ -73,7 +73,6 @@ processing requests via debug logs. Available tracers include: - tcp - Bytes in and out of a channel. - timer - Timers (alarms) in the grpc internals. - timer_check - more detailed trace of timer logic in grpc internals. - - token_fetcher_credentials - Token fetcher call credentials framework, used for (e.g.) oauth2 token fetcher credentials. - tsi - TSI transport security. - weighted_round_robin_lb - Weighted round robin load balancing policy. - weighted_target_lb - Weighted target LB policy. diff --git a/src/core/BUILD b/src/core/BUILD index 62dfec0eb41cc3..36497395ee2488 100644 --- a/src/core/BUILD +++ b/src/core/BUILD @@ -4335,17 +4335,14 @@ grpc_cc_library( deps = [ "arena_promise", "context", - "default_event_engine", "metadata", "poll", "pollset_set", "ref_counted", "time", "useful", - "//:backoff", "//:gpr", "//:grpc_security_base", - "//:grpc_trace", "//:httpcli", "//:iomgr", "//:orphanable", @@ -4354,52 +4351,6 @@ grpc_cc_library( ], ) -grpc_cc_library( - name = "gcp_service_account_identity_credentials", - srcs = [ - "lib/security/credentials/gcp_service_account_identity/gcp_service_account_identity_credentials.cc", - ], - hdrs = [ - "lib/security/credentials/gcp_service_account_identity/gcp_service_account_identity_credentials.h", - ], - external_deps = [ - "absl/functional:any_invocable", - "absl/status", - "absl/status:statusor", - "absl/strings", - ], - language = "c++", - deps = [ - "activity", - "arena_promise", - "closure", - "error", - "json", - "json_args", - "json_object_loader", - "json_reader", - "metadata", - "pollset_set", - "ref_counted", - "slice", - "status_conversion", - "status_helper", - "time", - "token_fetcher_credentials", - "unique_type_name", - "//:gpr", - "//:grpc_base", - "//:grpc_core_credentials_header", - "//:grpc_security_base", - "//:httpcli", - "//:iomgr", - "//:orphanable", - "//:promise", - "//:ref_counted_ptr", - "//:uri_parser", - ], -) - grpc_cc_library( name = "grpc_oauth2_credentials", srcs = [ @@ -4485,6 +4436,7 @@ grpc_cc_library( language = "c++", deps = [ "closure", + "default_event_engine", "env", "error", "error_utils", diff --git a/src/core/lib/debug/trace_flags.cc b/src/core/lib/debug/trace_flags.cc index 8f7183782c8195..f3fa6c63314ab8 100644 --- a/src/core/lib/debug/trace_flags.cc +++ b/src/core/lib/debug/trace_flags.cc @@ -114,7 +114,6 @@ TraceFlag subchannel_pool_trace(false, "subchannel_pool"); TraceFlag tcp_trace(false, "tcp"); TraceFlag timer_trace(false, "timer"); TraceFlag timer_check_trace(false, "timer_check"); -TraceFlag token_fetcher_credentials_trace(false, "token_fetcher_credentials"); TraceFlag tsi_trace(false, "tsi"); TraceFlag weighted_round_robin_lb_trace(false, "weighted_round_robin_lb"); TraceFlag weighted_target_lb_trace(false, "weighted_target_lb"); @@ -207,7 +206,6 @@ const absl::flat_hash_map& GetAllTraceFlags() { {"tcp", &tcp_trace}, {"timer", &timer_trace}, {"timer_check", &timer_check_trace}, - {"token_fetcher_credentials", &token_fetcher_credentials_trace}, {"tsi", &tsi_trace}, {"weighted_round_robin_lb", &weighted_round_robin_lb_trace}, {"weighted_target_lb", &weighted_target_lb_trace}, diff --git a/src/core/lib/debug/trace_flags.h b/src/core/lib/debug/trace_flags.h index 9aa4df691f0ec1..4aaf5e0111169d 100644 --- a/src/core/lib/debug/trace_flags.h +++ b/src/core/lib/debug/trace_flags.h @@ -112,7 +112,6 @@ extern TraceFlag subchannel_pool_trace; extern TraceFlag tcp_trace; extern TraceFlag timer_trace; extern TraceFlag timer_check_trace; -extern TraceFlag token_fetcher_credentials_trace; extern TraceFlag tsi_trace; extern TraceFlag weighted_round_robin_lb_trace; extern TraceFlag weighted_target_lb_trace; diff --git a/src/core/lib/debug/trace_flags.yaml b/src/core/lib/debug/trace_flags.yaml index 64c665af79958b..247d830e81f432 100644 --- a/src/core/lib/debug/trace_flags.yaml +++ b/src/core/lib/debug/trace_flags.yaml @@ -308,9 +308,6 @@ timer: timer_check: default: false description: more detailed trace of timer logic in grpc internals. -token_fetcher_credentials: - default: false - description: Token fetcher call credentials framework, used for (e.g.) oauth2 token fetcher credentials. tsi: default: false description: TSI transport security. diff --git a/src/core/lib/promise/map.h b/src/core/lib/promise/map.h index 3ba8c19c2f6ac9..a2a2a773eea2bf 100644 --- a/src/core/lib/promise/map.h +++ b/src/core/lib/promise/map.h @@ -86,7 +86,7 @@ GPR_ATTRIBUTE_ALWAYS_INLINE_FUNCTION auto CheckDelayed(Promise promise) { delayed = true; return Pending{}; } - return std::make_tuple(std::move(r.value()), delayed); + return std::make_tuple(r.value(), delayed); }; } diff --git a/src/core/lib/security/credentials/external/external_account_credentials.cc b/src/core/lib/security/credentials/external/external_account_credentials.cc index c83351e67e2966..100b4ddd8e4d65 100644 --- a/src/core/lib/security/credentials/external/external_account_credentials.cc +++ b/src/core/lib/security/credentials/external/external_account_credentials.cc @@ -45,6 +45,7 @@ #include #include +#include "src/core/lib/event_engine/default_event_engine.h" #include "src/core/lib/gprpp/status_helper.h" #include "src/core/lib/security/credentials/credentials.h" #include "src/core/lib/security/credentials/external/aws_external_account_credentials.h" @@ -590,7 +591,10 @@ ExternalAccountCredentials::Create( ExternalAccountCredentials::ExternalAccountCredentials( Options options, std::vector scopes, std::shared_ptr event_engine) - : TokenFetcherCredentials(std::move(event_engine)), + : event_engine_( + event_engine == nullptr + ? grpc_event_engine::experimental::GetDefaultEventEngine() + : std::move(event_engine)), options_(std::move(options)) { if (scopes.empty()) { scopes.push_back(GOOGLE_CLOUD_PLATFORM_DEFAULT_SCOPE); diff --git a/src/core/lib/security/credentials/external/external_account_credentials.h b/src/core/lib/security/credentials/external/external_account_credentials.h index 5dfbb24a36ede4..0617c3e6f07173 100644 --- a/src/core/lib/security/credentials/external/external_account_credentials.h +++ b/src/core/lib/security/credentials/external/external_account_credentials.h @@ -185,6 +185,10 @@ class ExternalAccountCredentials : public TokenFetcherCredentials { absl::string_view audience() const { return options_.audience; } + grpc_event_engine::experimental::EventEngine& event_engine() const { + return *event_engine_; + } + private: OrphanablePtr FetchToken( Timestamp deadline, @@ -200,6 +204,7 @@ class ExternalAccountCredentials : public TokenFetcherCredentials { Timestamp deadline, absl::AnyInvocable)> on_done) = 0; + std::shared_ptr event_engine_; Options options_; std::vector scopes_; }; diff --git a/src/core/lib/security/credentials/external/file_external_account_credentials.cc b/src/core/lib/security/credentials/external/file_external_account_credentials.cc index 086af8c79c7be6..cad9b7f7ee43a3 100644 --- a/src/core/lib/security/credentials/external/file_external_account_credentials.cc +++ b/src/core/lib/security/credentials/external/file_external_account_credentials.cc @@ -26,6 +26,7 @@ #include #include +#include "src/core/lib/event_engine/default_event_engine.h" #include "src/core/lib/gprpp/load_file.h" #include "src/core/lib/slice/slice.h" #include "src/core/lib/slice/slice_internal.h" diff --git a/src/core/lib/security/credentials/gcp_service_account_identity/gcp_service_account_identity_credentials.cc b/src/core/lib/security/credentials/gcp_service_account_identity/gcp_service_account_identity_credentials.cc deleted file mode 100644 index dba35759431d88..00000000000000 --- a/src/core/lib/security/credentials/gcp_service_account_identity/gcp_service_account_identity_credentials.cc +++ /dev/null @@ -1,196 +0,0 @@ -// -// Copyright 2024 gRPC authors. -// -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. -// - -#include "src/core/lib/security/credentials/gcp_service_account_identity/gcp_service_account_identity_credentials.h" - -#include "absl/functional/any_invocable.h" -#include "absl/status/status.h" -#include "absl/status/statusor.h" -#include "absl/strings/escaping.h" -#include "absl/strings/str_cat.h" -#include "absl/strings/str_split.h" -#include "absl/strings/string_view.h" - -#include - -#include "src/core/lib/gprpp/ref_counted_ptr.h" -#include "src/core/lib/gprpp/status_helper.h" -#include "src/core/lib/iomgr/error.h" -#include "src/core/lib/transport/metadata.h" -#include "src/core/lib/transport/status_conversion.h" -#include "src/core/lib/uri/uri_parser.h" -#include "src/core/util/json/json.h" -#include "src/core/util/json/json_args.h" -#include "src/core/util/json/json_object_loader.h" -#include "src/core/util/json/json_reader.h" - -namespace grpc_core { - -// -// JwtTokenFetcherCallCredentials -// - -// State held for a pending HTTP request. -class JwtTokenFetcherCallCredentials::HttpFetchRequest final - : public TokenFetcherCredentials::FetchRequest { - public: - HttpFetchRequest( - JwtTokenFetcherCallCredentials* creds, Timestamp deadline, - absl::AnyInvocable< - void(absl::StatusOr>)> - on_done) - : on_done_(std::move(on_done)) { - GRPC_CLOSURE_INIT(&on_http_response_, OnHttpResponse, this, nullptr); - Ref().release(); // Ref held by HTTP request callback. - http_request_ = creds->StartHttpRequest(creds->pollent(), deadline, - &response_, &on_http_response_); - } - - ~HttpFetchRequest() override { grpc_http_response_destroy(&response_); } - - void Orphan() override { - http_request_.reset(); - Unref(); - } - - private: - static void OnHttpResponse(void* arg, grpc_error_handle error) { - RefCountedPtr self(static_cast(arg)); - if (!error.ok()) { - // TODO(roth): It shouldn't be necessary to explicitly set the - // status to UNAVAILABLE here. Once the HTTP client code is - // migrated to stop using legacy grpc_error APIs to create - // statuses, we should be able to just propagate the status as-is. - self->on_done_(absl::UnavailableError(StatusToString(error))); - return; - } - if (self->response_.status != 200) { - grpc_status_code status_code = - grpc_http2_status_to_grpc_status(self->response_.status); - if (status_code != GRPC_STATUS_UNAVAILABLE) { - status_code = GRPC_STATUS_UNAUTHENTICATED; - } - self->on_done_(absl::Status(static_cast(status_code), - absl::StrCat("JWT fetch failed with status ", - self->response_.status))); - return; - } - absl::string_view body(self->response_.body, self->response_.body_length); - // Parse JWT token based on https://datatracker.ietf.org/doc/html/rfc7519. - // We don't do full verification here, just enough to extract the - // expiration time. - // First, split the 3 '.'-delimited parts. - std::vector parts = absl::StrSplit(body, '.'); - if (parts.size() != 3) { - self->on_done_(absl::UnauthenticatedError("error parsing JWT token")); - return; - } - // Base64-decode the payload. - std::string payload; - if (!absl::WebSafeBase64Unescape(parts[1], &payload)) { - self->on_done_(absl::UnauthenticatedError("error parsing JWT token")); - return; - } - // Parse as JSON. - auto json = JsonParse(payload); - if (!json.ok()) { - self->on_done_(absl::UnauthenticatedError("error parsing JWT token")); - return; - } - // Extract "exp" field. - struct ParsedPayload { - uint64_t exp = 0; - - static const JsonLoaderInterface* JsonLoader(const JsonArgs&) { - static const auto kJsonLoader = JsonObjectLoader() - .Field("exp", &ParsedPayload::exp) - .Finish(); - return kJsonLoader; - } - }; - auto parsed_payload = LoadFromJson(*json, JsonArgs(), ""); - if (!parsed_payload.ok()) { - self->on_done_(absl::UnauthenticatedError("error parsing JWT token")); - return; - } - gpr_timespec ts = gpr_time_0(GPR_CLOCK_REALTIME); - ts.tv_sec = parsed_payload->exp; - Timestamp expiration_time = Timestamp::FromTimespecRoundDown(ts); - // Return token object. - self->on_done_(MakeRefCounted( - Slice::FromCopiedString(absl::StrCat("Bearer ", body)), - expiration_time)); - } - - OrphanablePtr http_request_; - grpc_closure on_http_response_; - grpc_http_response response_; - absl::AnyInvocable>)> - on_done_; -}; - -OrphanablePtr -JwtTokenFetcherCallCredentials::FetchToken( - Timestamp deadline, - absl::AnyInvocable< - void(absl::StatusOr>)> - on_done) { - return MakeOrphanable(this, deadline, std::move(on_done)); -} - -// -// GcpServiceAccountIdentityCallCredentials -// - -std::string GcpServiceAccountIdentityCallCredentials::debug_string() { - return absl::StrCat("GcpServiceAccountIdentityCallCredentials(", audience_, - ")"); -} - -UniqueTypeName GcpServiceAccountIdentityCallCredentials::type() const { - static UniqueTypeName::Factory kFactory("GcpServiceAccountIdentity"); - return kFactory.Create(); -} - -OrphanablePtr -GcpServiceAccountIdentityCallCredentials::StartHttpRequest( - grpc_polling_entity* pollent, Timestamp deadline, - grpc_http_response* response, grpc_closure* on_complete) { - grpc_http_header header = {const_cast("Metadata-Flavor"), - const_cast("Google")}; - grpc_http_request request; - memset(&request, 0, sizeof(grpc_http_request)); - request.hdr_count = 1; - request.hdrs = &header; - // TODO(ctiller): Carry the memory quota in ctx and share it with the host - // channel. This would allow us to cancel an authentication query when under - // extreme memory pressure. - auto uri = URI::Create( - "http", "metadata.google.internal.", - "/computeMetadata/v1/instance/service-accounts/default/identity", - {{"audience", audience_}}, /*fragment=*/""); - CHECK_OK(uri); // params are hardcoded - auto http_request = - HttpRequest::Get(std::move(*uri), /*args=*/nullptr, pollent, &request, - deadline, on_complete, response, - RefCountedPtr( - grpc_insecure_credentials_create())); - http_request->Start(); - return http_request; -} - -} // namespace grpc_core diff --git a/src/core/lib/security/credentials/gcp_service_account_identity/gcp_service_account_identity_credentials.h b/src/core/lib/security/credentials/gcp_service_account_identity/gcp_service_account_identity_credentials.h deleted file mode 100644 index 1581b5a499fd9a..00000000000000 --- a/src/core/lib/security/credentials/gcp_service_account_identity/gcp_service_account_identity_credentials.h +++ /dev/null @@ -1,87 +0,0 @@ -// -// Copyright 2024 gRPC authors. -// -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. -// - -#ifndef GRPC_SRC_CORE_LIB_SECURITY_CREDENTIALS_GCP_SERVICE_ACCOUNT_IDENTITY_GCP_SERVICE_ACCOUNT_IDENTITY_CREDENTIALS_H -#define GRPC_SRC_CORE_LIB_SECURITY_CREDENTIALS_GCP_SERVICE_ACCOUNT_IDENTITY_GCP_SERVICE_ACCOUNT_IDENTITY_CREDENTIALS_H - -#include -#include - -#include -#include - -#include "src/core/lib/gprpp/orphanable.h" -#include "src/core/lib/gprpp/ref_counted_ptr.h" -#include "src/core/lib/gprpp/time.h" -#include "src/core/lib/gprpp/unique_type_name.h" -#include "src/core/lib/iomgr/closure.h" -#include "src/core/lib/iomgr/polling_entity.h" -#include "src/core/lib/security/credentials/credentials.h" -#include "src/core/lib/security/credentials/token_fetcher/token_fetcher_credentials.h" -#include "src/core/lib/slice/slice.h" -#include "src/core/lib/transport/metadata.h" -#include "src/core/util/http_client/httpcli.h" -#include "src/core/util/http_client/parser.h" - -namespace grpc_core { - -// A base class for JWT token fetching credentials. -// Subclasses must implement StartHttpRequest(). -class JwtTokenFetcherCallCredentials : public TokenFetcherCredentials { - public: - OrphanablePtr FetchToken( - Timestamp deadline, - absl::AnyInvocable< - void(absl::StatusOr>)> - on_done) final; - - private: - class HttpFetchRequest; - - virtual OrphanablePtr StartHttpRequest( - grpc_polling_entity* pollent, Timestamp deadline, - grpc_http_response* response, grpc_closure* on_complete) = 0; -}; - -// GCP service account identity call credentials. -class GcpServiceAccountIdentityCallCredentials - : public JwtTokenFetcherCallCredentials { - public: - explicit GcpServiceAccountIdentityCallCredentials(absl::string_view audience) - : audience_(audience) {} - - std::string debug_string() override; - - UniqueTypeName type() const override; - - absl::string_view audience() const { return audience_; } - - private: - OrphanablePtr StartHttpRequest( - grpc_polling_entity* pollent, Timestamp deadline, - grpc_http_response* response, grpc_closure* on_complete) override; - - int cmp_impl(const grpc_call_credentials* other) const override { - // TODO(yashykt): Check if we can do something better here - return QsortCompare(static_cast(this), other); - } - - std::string audience_; -}; - -} // namespace grpc_core - -#endif // GRPC_SRC_CORE_LIB_SECURITY_CREDENTIALS_GCP_SERVICE_ACCOUNT_IDENTITY_GCP_SERVICE_ACCOUNT_IDENTITY_CREDENTIALS_H diff --git a/src/core/lib/security/credentials/token_fetcher/token_fetcher_credentials.cc b/src/core/lib/security/credentials/token_fetcher/token_fetcher_credentials.cc index 596641f0fffdcf..eb5bee6478a962 100644 --- a/src/core/lib/security/credentials/token_fetcher/token_fetcher_credentials.cc +++ b/src/core/lib/security/credentials/token_fetcher/token_fetcher_credentials.cc @@ -18,8 +18,6 @@ #include "src/core/lib/security/credentials/token_fetcher/token_fetcher_credentials.h" -#include "src/core/lib/debug/trace.h" -#include "src/core/lib/event_engine/default_event_engine.h" #include "src/core/lib/iomgr/pollset_set.h" #include "src/core/lib/promise/context.h" #include "src/core/lib/promise/poll.h" @@ -30,11 +28,8 @@ namespace grpc_core { namespace { // Amount of time before the token's expiration that we consider it -// invalid to account for server processing time and clock skew. -constexpr Duration kTokenExpirationAdjustmentDuration = Duration::Seconds(30); - -// Amount of time before the token's expiration that we pre-fetch a new -// token. Also determines the timeout for the fetch request. +// invalid and start a new fetch. Also determines the timeout for the +// fetch request. constexpr Duration kTokenRefreshDuration = Duration::Seconds(60); } // namespace @@ -43,193 +38,18 @@ constexpr Duration kTokenRefreshDuration = Duration::Seconds(60); // TokenFetcherCredentials::Token // -TokenFetcherCredentials::Token::Token(Slice token, Timestamp expiration) - : token_(std::move(token)), - expiration_(expiration - kTokenExpirationAdjustmentDuration) {} - void TokenFetcherCredentials::Token::AddTokenToClientInitialMetadata( ClientMetadata& metadata) const { metadata.Append(GRPC_AUTHORIZATION_METADATA_KEY, token_.Ref(), [](absl::string_view, const Slice&) { abort(); }); } -// -// TokenFetcherCredentials::FetchState::BackoffTimer -// - -TokenFetcherCredentials::FetchState::BackoffTimer::BackoffTimer( - RefCountedPtr fetch_state) - : fetch_state_(std::move(fetch_state)) { - const Timestamp next_attempt_time = fetch_state_->backoff_.NextAttemptTime(); - const Duration duration = next_attempt_time - Timestamp::Now(); - GRPC_TRACE_LOG(token_fetcher_credentials, INFO) - << "[TokenFetcherCredentials " << fetch_state_->creds_.get() - << "]: fetch_state=" << fetch_state_.get() << " backoff_timer=" << this - << ": starting backoff timer for " << next_attempt_time << " (" - << duration << " from now)"; - timer_handle_ = fetch_state_->creds_->event_engine().RunAfter( - duration, [self = Ref()]() mutable { - ApplicationCallbackExecCtx callback_exec_ctx; - ExecCtx exec_ctx; - self->OnTimer(); - self.reset(); - }); -} - -void TokenFetcherCredentials::FetchState::BackoffTimer::Orphan() { - GRPC_TRACE_LOG(token_fetcher_credentials, INFO) - << "[TokenFetcherCredentials " << fetch_state_->creds_.get() - << "]: fetch_state=" << fetch_state_.get() << " backoff_timer=" << this - << ": backoff timer shut down"; - if (timer_handle_.has_value()) { - GRPC_TRACE_LOG(token_fetcher_credentials, INFO) - << "[TokenFetcherCredentials " << fetch_state_->creds_.get() - << "]: fetch_state=" << fetch_state_.get() << " backoff_timer=" << this - << ": cancelling timer"; - fetch_state_->creds_->event_engine().Cancel(*timer_handle_); - timer_handle_.reset(); - fetch_state_->ResumeQueuedCalls( - absl::CancelledError("credentials shutdown")); - } - Unref(); -} - -void TokenFetcherCredentials::FetchState::BackoffTimer::OnTimer() { - MutexLock lock(&fetch_state_->creds_->mu_); - if (!timer_handle_.has_value()) return; - timer_handle_.reset(); - GRPC_TRACE_LOG(token_fetcher_credentials, INFO) - << "[TokenFetcherCredentials " << fetch_state_->creds_.get() - << "]: fetch_state=" << fetch_state_.get() << " backoff_timer=" << this - << ": backoff timer fired"; - if (fetch_state_->queued_calls_.empty()) { - // If there are no pending calls when the timer fires, then orphan - // the FetchState object. Note that this drops the backoff state, - // but that's probably okay, because if we didn't have any pending - // calls during the backoff period, we probably won't see any - // immediately now either. - GRPC_TRACE_LOG(token_fetcher_credentials, INFO) - << "[TokenFetcherCredentials " << fetch_state_->creds_.get() - << "]: fetch_state=" << fetch_state_.get() << " backoff_timer=" << this - << ": no pending calls, clearing state"; - fetch_state_->creds_->fetch_state_.reset(); - } else { - // If there are pending calls, then start a new fetch attempt. - GRPC_TRACE_LOG(token_fetcher_credentials, INFO) - << "[TokenFetcherCredentials " << fetch_state_->creds_.get() - << "]: fetch_state=" << fetch_state_.get() << " backoff_timer=" << this - << ": starting new fetch attempt"; - fetch_state_->StartFetchAttempt(); - } -} - -// -// TokenFetcherCredentials::FetchState -// - -TokenFetcherCredentials::FetchState::FetchState( - WeakRefCountedPtr creds) - : creds_(std::move(creds)), - backoff_(BackOff::Options() - .set_initial_backoff(Duration::Seconds(1)) - .set_multiplier(1.6) - .set_jitter(creds_->test_only_use_backoff_jitter_ ? 0.2 : 0) - .set_max_backoff(Duration::Seconds(120))) { - StartFetchAttempt(); -} - -void TokenFetcherCredentials::FetchState::Orphan() { - GRPC_TRACE_LOG(token_fetcher_credentials, INFO) - << "[TokenFetcherCredentials " << creds_.get() - << "]: fetch_state=" << this << ": shutting down"; - // Cancels fetch or backoff timer, if any. - state_ = Shutdown{}; - Unref(); -} - -void TokenFetcherCredentials::FetchState::StartFetchAttempt() { - GRPC_TRACE_LOG(token_fetcher_credentials, INFO) - << "[TokenFetcherCredentials " << creds_.get() - << "]: fetch_state=" << this << ": starting fetch"; - state_ = creds_->FetchToken( - /*deadline=*/Timestamp::Now() + kTokenRefreshDuration, - [self = Ref()](absl::StatusOr> token) mutable { - self->TokenFetchComplete(std::move(token)); - }); -} - -void TokenFetcherCredentials::FetchState::TokenFetchComplete( - absl::StatusOr> token) { - MutexLock lock(&creds_->mu_); - // If we were shut down, clean up. - if (absl::holds_alternative(state_)) { - if (token.ok()) token = absl::CancelledError("credentials shutdown"); - GRPC_TRACE_LOG(token_fetcher_credentials, INFO) - << "[TokenFetcherCredentials " << creds_.get() - << "]: fetch_state=" << this - << ": shut down before fetch completed: " << token.status(); - ResumeQueuedCalls(std::move(token)); - return; - } - // If succeeded, update cache in creds object. - if (token.ok()) { - GRPC_TRACE_LOG(token_fetcher_credentials, INFO) - << "[TokenFetcherCredentials " << creds_.get() - << "]: fetch_state=" << this << ": token fetch succeeded"; - creds_->token_ = *token; - creds_->fetch_state_.reset(); // Orphan ourselves. - } else { - GRPC_TRACE_LOG(token_fetcher_credentials, INFO) - << "[TokenFetcherCredentials " << creds_.get() - << "]: fetch_state=" << this - << ": token fetch failed: " << token.status(); - // If failed, start backoff timer. - state_ = OrphanablePtr(new BackoffTimer(Ref())); - } - ResumeQueuedCalls(std::move(token)); -} - -void TokenFetcherCredentials::FetchState::ResumeQueuedCalls( - absl::StatusOr> token) { - // Invoke callbacks for all pending requests. - for (auto& queued_call : queued_calls_) { - queued_call->result = token; - queued_call->done.store(true, std::memory_order_release); - queued_call->waker.Wakeup(); - grpc_polling_entity_del_from_pollset_set( - queued_call->pollent, - grpc_polling_entity_pollset_set(&creds_->pollent_)); - } - queued_calls_.clear(); -} - -RefCountedPtr -TokenFetcherCredentials::FetchState::QueueCall( - ClientMetadataHandle initial_metadata) { - // Add call to pending list. - auto queued_call = MakeRefCounted(); - queued_call->waker = GetContext()->MakeNonOwningWaker(); - queued_call->pollent = GetContext(); - grpc_polling_entity_add_to_pollset_set( - queued_call->pollent, grpc_polling_entity_pollset_set(&creds_->pollent_)); - queued_call->md = std::move(initial_metadata); - queued_calls_.insert(queued_call); - return queued_call; -} - // // TokenFetcherCredentials // -TokenFetcherCredentials::TokenFetcherCredentials( - std::shared_ptr event_engine, - bool test_only_use_backoff_jitter) - : event_engine_( - event_engine == nullptr - ? grpc_event_engine::experimental::GetDefaultEventEngine() - : std::move(event_engine)), - test_only_use_backoff_jitter_(test_only_use_backoff_jitter), - pollent_(grpc_polling_entity_create_from_pollset_set( +TokenFetcherCredentials::TokenFetcherCredentials() + : pollent_(grpc_polling_entity_create_from_pollset_set( grpc_pollset_set_create())) {} TokenFetcherCredentials::~TokenFetcherCredentials() { @@ -238,63 +58,73 @@ TokenFetcherCredentials::~TokenFetcherCredentials() { void TokenFetcherCredentials::Orphaned() { MutexLock lock(&mu_); - fetch_state_.reset(); + auto* fetch_request = absl::get_if>(&token_); + if (fetch_request != nullptr) fetch_request->reset(); } ArenaPromise> TokenFetcherCredentials::GetRequestMetadata( ClientMetadataHandle initial_metadata, const GetRequestMetadataArgs*) { - RefCountedPtr queued_call; + RefCountedPtr pending_call; { MutexLock lock(&mu_); - // If we don't have a cached token or the token is within the - // refresh duration, start a new fetch if there isn't a pending one. - if ((token_ == nullptr || (token_->ExpirationTime() - Timestamp::Now()) <= - kTokenRefreshDuration) && - fetch_state_ == nullptr) { - GRPC_TRACE_LOG(token_fetcher_credentials, INFO) - << "[TokenFetcherCredentials " << this - << "]: " << GetContext()->DebugTag() - << " triggering new token fetch"; - fetch_state_ = OrphanablePtr( - new FetchState(WeakRefAsSubclass())); - } - // If we have a cached non-expired token, use it. - if (token_ != nullptr && - (token_->ExpirationTime() - Timestamp::Now()) > Duration::Zero()) { - GRPC_TRACE_LOG(token_fetcher_credentials, INFO) - << "[TokenFetcherCredentials " << this - << "]: " << GetContext()->DebugTag() - << " using cached token"; - token_->AddTokenToClientInitialMetadata(*initial_metadata); + // Check if we can use the cached token. + auto* cached_token = absl::get_if>(&token_); + if (cached_token != nullptr && *cached_token != nullptr && + ((*cached_token)->ExpirationTime() - Timestamp::Now()) > + kTokenRefreshDuration) { + (*cached_token)->AddTokenToClientInitialMetadata(*initial_metadata); return Immediate(std::move(initial_metadata)); } - // If we don't have a cached token, this call will need to be queued. - GRPC_TRACE_LOG(token_fetcher_credentials, INFO) - << "[TokenFetcherCredentials " << this - << "]: " << GetContext()->DebugTag() - << " no cached token; queuing call"; - queued_call = fetch_state_->QueueCall(std::move(initial_metadata)); + // Couldn't get the token from the cache. + // Add this call to the pending list. + pending_call = MakeRefCounted(); + pending_call->waker = GetContext()->MakeNonOwningWaker(); + pending_call->pollent = GetContext(); + grpc_polling_entity_add_to_pollset_set( + pending_call->pollent, grpc_polling_entity_pollset_set(&pollent_)); + pending_call->md = std::move(initial_metadata); + pending_calls_.insert(pending_call); + // Start a new fetch if needed. + if (!absl::holds_alternative>(token_)) { + token_ = FetchToken( + /*deadline=*/Timestamp::Now() + kTokenRefreshDuration, + [self = WeakRefAsSubclass()]( + absl::StatusOr> token) mutable { + self->TokenFetchComplete(std::move(token)); + }); + } } - return [this, queued_call = std::move(queued_call)]() - -> Poll> { - if (!queued_call->done.load(std::memory_order_acquire)) { + return [pending_call = std::move( + pending_call)]() -> Poll> { + if (!pending_call->done.load(std::memory_order_acquire)) { return Pending{}; } - if (!queued_call->result.ok()) { - GRPC_TRACE_LOG(token_fetcher_credentials, INFO) - << "[TokenFetcherCredentials " << this - << "]: " << GetContext()->DebugTag() - << " token fetch failed; failing call"; - return queued_call->result.status(); + if (!pending_call->result.ok()) { + return pending_call->result.status(); } - GRPC_TRACE_LOG(token_fetcher_credentials, INFO) - << "[TokenFetcherCredentials " << this - << "]: " << GetContext()->DebugTag() - << " token fetch complete; resuming call"; - (*queued_call->result)->AddTokenToClientInitialMetadata(*queued_call->md); - return std::move(queued_call->md); + (*pending_call->result)->AddTokenToClientInitialMetadata(*pending_call->md); + return std::move(pending_call->md); }; } +void TokenFetcherCredentials::TokenFetchComplete( + absl::StatusOr> token) { + // Update cache and grab list of pending requests. + absl::flat_hash_set> pending_calls; + { + MutexLock lock(&mu_); + token_ = token.value_or(nullptr); + pending_calls_.swap(pending_calls); + } + // Invoke callbacks for all pending requests. + for (auto& pending_call : pending_calls) { + pending_call->result = token; + pending_call->done.store(true, std::memory_order_release); + pending_call->waker.Wakeup(); + grpc_polling_entity_del_from_pollset_set( + pending_call->pollent, grpc_polling_entity_pollset_set(&pollent_)); + } +} + } // namespace grpc_core diff --git a/src/core/lib/security/credentials/token_fetcher/token_fetcher_credentials.h b/src/core/lib/security/credentials/token_fetcher/token_fetcher_credentials.h index c73d469234b066..72793afcdee253 100644 --- a/src/core/lib/security/credentials/token_fetcher/token_fetcher_credentials.h +++ b/src/core/lib/security/credentials/token_fetcher/token_fetcher_credentials.h @@ -26,9 +26,6 @@ #include "absl/status/statusor.h" #include "absl/types/variant.h" -#include - -#include "src/core/lib/backoff/backoff.h" #include "src/core/lib/gprpp/orphanable.h" #include "src/core/lib/gprpp/ref_counted.h" #include "src/core/lib/gprpp/ref_counted_ptr.h" @@ -50,7 +47,8 @@ class TokenFetcherCredentials : public grpc_call_credentials { // Represents a token. class Token : public RefCounted { public: - Token(Slice token, Timestamp expiration); + Token(Slice token, Timestamp expiration) + : token_(std::move(token)), expiration_(expiration) {} // Returns the token's expiration time. Timestamp ExpirationTime() const { return expiration_; } @@ -75,10 +73,7 @@ class TokenFetcherCredentials : public grpc_call_credentials { // Base class for fetch requests. class FetchRequest : public InternallyRefCounted {}; - explicit TokenFetcherCredentials( - std::shared_ptr - event_engine = nullptr, - bool test_only_use_backoff_jitter = true); + TokenFetcherCredentials(); // Fetches a token. The on_done callback will be invoked when complete. virtual OrphanablePtr FetchToken( @@ -86,15 +81,11 @@ class TokenFetcherCredentials : public grpc_call_credentials { absl::AnyInvocable>)> on_done) = 0; - grpc_event_engine::experimental::EventEngine& event_engine() const { - return *event_engine_; - } - grpc_polling_entity* pollent() { return &pollent_; } private: // A call that is waiting for a token fetch request to complete. - struct QueuedCall : public RefCounted { + struct PendingCall : public RefCounted { std::atomic done{false}; Waker waker; grpc_polling_entity* pollent; @@ -102,72 +93,20 @@ class TokenFetcherCredentials : public grpc_call_credentials { absl::StatusOr> result; }; - class FetchState : public InternallyRefCounted { - public: - explicit FetchState(WeakRefCountedPtr creds) - ABSL_EXCLUSIVE_LOCKS_REQUIRED(&TokenFetcherCredentials::mu_); - - // Disabling thread safety annotations, since Orphan() is called - // by OrpahanablePtr<>, which does not have the right lock - // annotations. - void Orphan() override ABSL_NO_THREAD_SAFETY_ANALYSIS; - - RefCountedPtr QueueCall(ClientMetadataHandle initial_metadata) - ABSL_EXCLUSIVE_LOCKS_REQUIRED(&TokenFetcherCredentials::mu_); - - private: - class BackoffTimer : public InternallyRefCounted { - public: - explicit BackoffTimer(RefCountedPtr fetch_state) - ABSL_EXCLUSIVE_LOCKS_REQUIRED(&TokenFetcherCredentials::mu_); - - // Disabling thread safety annotations, since Orphan() is called - // by OrpahanablePtr<>, which does not have the right lock - // annotations. - void Orphan() override ABSL_NO_THREAD_SAFETY_ANALYSIS; - - private: - void OnTimer(); - - RefCountedPtr fetch_state_; - absl::optional - timer_handle_ ABSL_GUARDED_BY(&TokenFetcherCredentials::mu_); - }; - - struct Shutdown {}; - - void StartFetchAttempt() - ABSL_EXCLUSIVE_LOCKS_REQUIRED(&TokenFetcherCredentials::mu_); - void TokenFetchComplete(absl::StatusOr> token); - void ResumeQueuedCalls(absl::StatusOr> token) - ABSL_EXCLUSIVE_LOCKS_REQUIRED(&TokenFetcherCredentials::mu_); - - WeakRefCountedPtr creds_; - // Pending token-fetch request or backoff timer, if any. - absl::variant, OrphanablePtr, - Shutdown> - state_ ABSL_GUARDED_BY(&TokenFetcherCredentials::mu_); - // Calls that are queued up waiting for the token. - absl::flat_hash_set> queued_calls_ - ABSL_GUARDED_BY(&TokenFetcherCredentials::mu_); - // Backoff state. - BackOff backoff_ ABSL_GUARDED_BY(&TokenFetcherCredentials::mu_); - }; - int cmp_impl(const grpc_call_credentials* other) const override { // TODO(yashykt): Check if we can do something better here return QsortCompare(static_cast(this), other); } - std::shared_ptr event_engine_; - const bool test_only_use_backoff_jitter_; + void TokenFetchComplete(absl::StatusOr> token); Mutex mu_; - // Cached token, if any. - RefCountedPtr token_ ABSL_GUARDED_BY(&mu_); - // Fetch state, if any. - OrphanablePtr fetch_state_ ABSL_GUARDED_BY(&mu_); - + // Either the cached token or a pending request to fetch the token. + absl::variant, OrphanablePtr> token_ + ABSL_GUARDED_BY(&mu_); + // Calls that are queued up waiting for the token. + absl::flat_hash_set> pending_calls_ + ABSL_GUARDED_BY(&mu_); grpc_polling_entity pollent_ ABSL_GUARDED_BY(&mu_); }; diff --git a/test/core/security/BUILD b/test/core/security/BUILD index ed139dd4ffb3e8..e11332fb22e9f3 100644 --- a/test/core/security/BUILD +++ b/test/core/security/BUILD @@ -111,7 +111,6 @@ grpc_cc_test( "//:gpr", "//:grpc", "//src/core:channel_args", - "//src/core:gcp_service_account_identity_credentials", "//test/core/event_engine:event_engine_test_utils", "//test/core/event_engine/fuzzing_event_engine", "//test/core/test_util:grpc_test_util", diff --git a/test/core/security/credentials_test.cc b/test/core/security/credentials_test.cc index c122857fe079cf..fb93d522a7c9fd 100644 --- a/test/core/security/credentials_test.cc +++ b/test/core/security/credentials_test.cc @@ -50,7 +50,6 @@ #include "src/core/lib/iomgr/error.h" #include "src/core/lib/iomgr/timer_manager.h" #include "src/core/lib/promise/exec_ctx_wakeup_scheduler.h" -#include "src/core/lib/promise/map.h" #include "src/core/lib/promise/promise.h" #include "src/core/lib/promise/seq.h" #include "src/core/lib/security/context/security_context.h" @@ -60,7 +59,6 @@ #include "src/core/lib/security/credentials/external/file_external_account_credentials.h" #include "src/core/lib/security/credentials/external/url_external_account_credentials.h" #include "src/core/lib/security/credentials/fake/fake_credentials.h" -#include "src/core/lib/security/credentials/gcp_service_account_identity/gcp_service_account_identity_credentials.h" #include "src/core/lib/security/credentials/google_default/google_default_credentials.h" #include "src/core/lib/security/credentials/iam/iam_credentials.h" #include "src/core/lib/security/credentials/jwt/jwt_credentials.h" @@ -81,7 +79,6 @@ namespace grpc_core { -using grpc_event_engine::experimental::FuzzingEventEngine; using internal::grpc_flush_cached_google_default_credentials; using internal::set_gce_tenancy_checker_for_testing; @@ -429,19 +426,16 @@ TEST_F(CredentialsTest, class RequestMetadataState : public RefCounted { public: static RefCountedPtr NewInstance( - grpc_error_handle expected_error, std::string expected, - absl::optional expect_delay = absl::nullopt) { + grpc_error_handle expected_error, std::string expected) { return MakeRefCounted( - expected_error, std::move(expected), expect_delay, + expected_error, std::move(expected), grpc_polling_entity_create_from_pollset_set(grpc_pollset_set_create())); } RequestMetadataState(grpc_error_handle expected_error, std::string expected, - absl::optional expect_delay, grpc_polling_entity pollent) : expected_error_(expected_error), expected_(std::move(expected)), - expect_delay_(expect_delay), pollent_(pollent) {} ~RequestMetadataState() override { @@ -459,18 +453,12 @@ class RequestMetadataState : public RefCounted { activity_ = MakeActivity( [this, creds] { return Seq( - CheckDelayed(creds->GetRequestMetadata( + creds->GetRequestMetadata( ClientMetadataHandle(&md_, Arena::PooledDeleter(nullptr)), - &get_request_metadata_args_)), - [this](std::tuple, bool> - metadata_and_delayed) { - auto& metadata = std::get<0>(metadata_and_delayed); - const bool delayed = std::get<1>(metadata_and_delayed); - if (expect_delay_.has_value()) { - EXPECT_EQ(delayed, *expect_delay_); - } + &get_request_metadata_args_), + [this](absl::StatusOr metadata) { if (metadata.ok()) { - EXPECT_EQ(metadata->get(), &md_); + CHECK(metadata->get() == &md_); } return metadata.status(); }); @@ -520,13 +508,12 @@ class RequestMetadataState : public RefCounted { if (expected_error_.ok()) { ASSERT_TRUE(error.ok()) << error; } else { - grpc_status_code actual_code; - std::string actual_message; - grpc_error_get_status(error, Timestamp::InfFuture(), &actual_code, - &actual_message, nullptr, nullptr); - EXPECT_EQ(absl::Status(static_cast(actual_code), - actual_message), - expected_error_); + std::string expected_error; + grpc_error_get_str(expected_error_, StatusStrProperty::kDescription, + &expected_error); + std::string actual_error; + grpc_error_get_str(error, StatusStrProperty::kDescription, &actual_error); + EXPECT_EQ(expected_error, actual_error); } md_.Remove(HttpAuthorityMetadata()); md_.Remove(HttpPathMetadata()); @@ -536,7 +523,6 @@ class RequestMetadataState : public RefCounted { grpc_error_handle expected_error_; std::string expected_; - absl::optional expect_delay_; RefCountedPtr arena_ = SimpleArenaAllocator()->MakeArena(); grpc_metadata_batch md_; grpc_call_credentials::GetRequestMetadataArgs get_request_metadata_args_; @@ -812,8 +798,7 @@ TEST_F(CredentialsTest, TestComputeEngineCredsFailure) { "GoogleComputeEngineTokenFetcherCredentials{" "OAuth2TokenFetcherCredentials}"; auto state = RequestMetadataState::NewInstance( - // TODO(roth): This should return UNAUTHENTICATED. - absl::UnavailableError("error parsing oauth2 token"), {}); + GRPC_ERROR_CREATE("error parsing oauth2 token"), {}); grpc_call_credentials* creds = grpc_google_compute_engine_credentials_create(nullptr); HttpRequest::SetOverride(compute_engine_httpcli_get_failure_override, @@ -905,8 +890,7 @@ TEST_F(CredentialsTest, TestRefreshTokenCredsFailure) { "GoogleRefreshToken{ClientID:32555999999.apps.googleusercontent.com," "OAuth2TokenFetcherCredentials}"; auto state = RequestMetadataState::NewInstance( - // TODO(roth): This should return UNAUTHENTICATED. - absl::UnavailableError("error parsing oauth2 token"), {}); + GRPC_ERROR_CREATE("error parsing oauth2 token"), {}); grpc_call_credentials* creds = grpc_google_refresh_token_credentials_create( test_refresh_token_str, nullptr); HttpRequest::SetOverride(httpcli_get_should_not_be_called, @@ -1166,8 +1150,7 @@ TEST_F(CredentialsTest, TestStsCredsTokenFileNotFound) { CHECK_EQ(creds->min_security_level(), GRPC_PRIVACY_AND_INTEGRITY); auto state = RequestMetadataState::NewInstance( - // TODO(roth): This should return UNAVAILABLE. - absl::InternalError( + GRPC_ERROR_CREATE( "Failed to load file: /some/completely/random/path due to " "error(fdopen): No such file or directory"), {}); @@ -1238,9 +1221,8 @@ TEST_F(CredentialsTest, TestStsCredsLoadTokenFailure) { "token-exchange,Authority:foo.com:5555,OAuth2TokenFetcherCredentials}"; ExecCtx exec_ctx; auto state = RequestMetadataState::NewInstance( - // TODO(roth): This should return UNAVAILABLE. - absl::InternalError("Failed to load file: invalid_path due to " - "error(fdopen): No such file or directory"), + GRPC_ERROR_CREATE("Failed to load file: invalid_path due to " + "error(fdopen): No such file or directory"), {}); char* test_signed_jwt_path = write_tmp_jwt_file(test_signed_jwt); grpc_sts_credentials_options options = { @@ -1274,8 +1256,7 @@ TEST_F(CredentialsTest, TestStsCredsHttpFailure) { "token-exchange,Authority:foo.com:5555,OAuth2TokenFetcherCredentials}"; ExecCtx exec_ctx; auto state = RequestMetadataState::NewInstance( - // TODO(roth): This should return UNAUTHENTICATED. - absl::UnavailableError("error parsing oauth2 token"), {}); + GRPC_ERROR_CREATE("error parsing oauth2 token"), {}); char* test_signed_jwt_path = write_tmp_jwt_file(test_signed_jwt); grpc_sts_credentials_options valid_options = { test_sts_endpoint_url, // sts_endpoint_url @@ -1472,7 +1453,7 @@ TEST_F(CredentialsTest, TestJwtCredsSigningFailure) { char* json_key_string = test_json_key_str(); ExecCtx exec_ctx; auto state = RequestMetadataState::NewInstance( - absl::UnauthenticatedError("Could not generate JWT."), {}); + GRPC_ERROR_CREATE("Could not generate JWT."), {}); grpc_call_credentials* creds = grpc_service_account_jwt_access_credentials_create( json_key_string, grpc_max_auth_token_lifetime(), nullptr); @@ -1931,8 +1912,7 @@ TEST_F(CredentialsTest, TestMetadataPluginFailure) { grpc_metadata_credentials_plugin plugin; ExecCtx exec_ctx; auto md_state = RequestMetadataState::NewInstance( - // TODO(roth): Is this the right status to use here? - absl::UnavailableError( + GRPC_ERROR_CREATE( absl::StrCat("Getting metadata from plugin failed with error: ", plugin_error_details)), {}); @@ -2389,315 +2369,6 @@ int aws_external_account_creds_httpcli_post_success( return 1; } -class TokenFetcherCredentialsTest : public ::testing::Test { - protected: - class TestTokenFetcherCredentials final : public TokenFetcherCredentials { - public: - explicit TestTokenFetcherCredentials( - std::shared_ptr - event_engine = nullptr) - : TokenFetcherCredentials(std::move(event_engine), - /*test_only_use_backoff_jitter=*/false) {} - - ~TestTokenFetcherCredentials() override { CHECK_EQ(queue_.size(), 0); } - - void AddResult(absl::StatusOr> result) { - MutexLock lock(&mu_); - queue_.push_front(std::move(result)); - } - - size_t num_fetches() const { return num_fetches_; } - - private: - class TestFetchRequest final : public FetchRequest { - public: - TestFetchRequest( - grpc_event_engine::experimental::EventEngine& event_engine, - absl::AnyInvocable>)> - on_done, - absl::StatusOr> result) { - event_engine.Run([on_done = std::move(on_done), - result = std::move(result)]() mutable { - ApplicationCallbackExecCtx application_exec_ctx; - ExecCtx exec_ctx; - std::exchange(on_done, nullptr)(std::move(result)); - }); - } - - void Orphan() override { Unref(); } - }; - - OrphanablePtr FetchToken( - Timestamp deadline, - absl::AnyInvocable>)> on_done) - override { - absl::StatusOr> result; - { - MutexLock lock(&mu_); - CHECK(!queue_.empty()); - result = std::move(queue_.back()); - queue_.pop_back(); - } - num_fetches_.fetch_add(1); - return MakeOrphanable( - event_engine(), std::move(on_done), std::move(result)); - } - - std::string debug_string() override { - return "TestTokenFetcherCredentials"; - } - - UniqueTypeName type() const override { - static UniqueTypeName::Factory kFactory("TestTokenFetcherCredentials"); - return kFactory.Create(); - } - - Mutex mu_; - std::deque>> queue_ - ABSL_GUARDED_BY(&mu_); - - std::atomic num_fetches_{0}; - }; - - void SetUp() override { - grpc_timer_manager_set_start_threaded(false); - grpc_init(); - } - - void TearDown() override { - event_engine_->FuzzingDone(); - event_engine_->TickUntilIdle(); - event_engine_->UnsetGlobalHooks(); - creds_.reset(); - grpc_event_engine::experimental::WaitForSingleOwner( - std::move(event_engine_)); - grpc_shutdown_blocking(); - } - - static RefCountedPtr MakeToken( - absl::string_view token, Timestamp expiration = Timestamp::InfFuture()) { - return MakeRefCounted( - Slice::FromCopiedString(token), expiration); - } - - std::shared_ptr event_engine_ = - std::make_shared(FuzzingEventEngine::Options(), - fuzzing_event_engine::Actions()); - RefCountedPtr creds_ = - MakeRefCounted(event_engine_); -}; - -TEST_F(TokenFetcherCredentialsTest, Basic) { - const auto kExpirationTime = Timestamp::Now() + Duration::Hours(1); - ExecCtx exec_ctx; - creds_->AddResult(MakeToken("foo", kExpirationTime)); - // First request will trigger a fetch. - auto state = RequestMetadataState::NewInstance( - absl::OkStatus(), "authorization: foo", /*expect_delay=*/true); - state->RunRequestMetadataTest(creds_.get(), kTestUrlScheme, kTestAuthority, - kTestPath); - EXPECT_EQ(creds_->num_fetches(), 1); - // Second request while fetch is still outstanding will be delayed but - // will not trigger a new fetch. - state = RequestMetadataState::NewInstance( - absl::OkStatus(), "authorization: foo", /*expect_delay=*/true); - state->RunRequestMetadataTest(creds_.get(), kTestUrlScheme, kTestAuthority, - kTestPath); - EXPECT_EQ(creds_->num_fetches(), 1); - // Now tick to finish the fetch. - event_engine_->TickUntilIdle(); - // Next request will be served from cache with no delay. - state = RequestMetadataState::NewInstance( - absl::OkStatus(), "authorization: foo", /*expect_delay=*/false); - state->RunRequestMetadataTest(creds_.get(), kTestUrlScheme, kTestAuthority, - kTestPath); - EXPECT_EQ(creds_->num_fetches(), 1); - // Advance time to expiration minus expiration adjustment and prefetch time. - exec_ctx.TestOnlySetNow(kExpirationTime - Duration::Seconds(90)); - // No new fetch yet. - EXPECT_EQ(creds_->num_fetches(), 1); - // Next request will trigger a new fetch but will still use the - // cached token. - creds_->AddResult(MakeToken("bar")); - state = RequestMetadataState::NewInstance( - absl::OkStatus(), "authorization: foo", /*expect_delay=*/false); - state->RunRequestMetadataTest(creds_.get(), kTestUrlScheme, kTestAuthority, - kTestPath); - EXPECT_EQ(creds_->num_fetches(), 2); - event_engine_->TickUntilIdle(); - // Next request will use the new data. - state = RequestMetadataState::NewInstance( - absl::OkStatus(), "authorization: bar", /*expect_delay=*/false); - state->RunRequestMetadataTest(creds_.get(), kTestUrlScheme, kTestAuthority, - kTestPath); - EXPECT_EQ(creds_->num_fetches(), 2); -} - -TEST_F(TokenFetcherCredentialsTest, Expires30SecondsEarly) { - const auto kExpirationTime = Timestamp::Now() + Duration::Hours(1); - ExecCtx exec_ctx; - creds_->AddResult(MakeToken("foo", kExpirationTime)); - // First request will trigger a fetch. - auto state = RequestMetadataState::NewInstance( - absl::OkStatus(), "authorization: foo", /*expect_delay=*/true); - state->RunRequestMetadataTest(creds_.get(), kTestUrlScheme, kTestAuthority, - kTestPath); - EXPECT_EQ(creds_->num_fetches(), 1); - event_engine_->TickUntilIdle(); - // Advance time to expiration minus 30 seconds. - exec_ctx.TestOnlySetNow(kExpirationTime - Duration::Seconds(30)); - // No new fetch yet. - EXPECT_EQ(creds_->num_fetches(), 1); - // Next request will trigger a new fetch and will delay the call until - // the fetch completes. - creds_->AddResult(MakeToken("bar")); - state = RequestMetadataState::NewInstance( - absl::OkStatus(), "authorization: bar", /*expect_delay=*/true); - state->RunRequestMetadataTest(creds_.get(), kTestUrlScheme, kTestAuthority, - kTestPath); - EXPECT_EQ(creds_->num_fetches(), 2); - event_engine_->TickUntilIdle(); -} - -TEST_F(TokenFetcherCredentialsTest, FetchFails) { - const absl::Status kExpectedError = absl::UnavailableError("bummer, dude"); - absl::optional run_after_duration; - event_engine_->SetRunAfterDurationCallback( - [&](FuzzingEventEngine::Duration duration) { - run_after_duration = duration; - }); - ExecCtx exec_ctx; - creds_->AddResult(kExpectedError); - // First request will trigger a fetch, which will fail. - auto state = RequestMetadataState::NewInstance(kExpectedError, "", - /*expect_delay=*/true); - state->RunRequestMetadataTest(creds_.get(), kTestUrlScheme, kTestAuthority, - kTestPath); - EXPECT_EQ(creds_->num_fetches(), 1); - while (!run_after_duration.has_value()) event_engine_->Tick(); - // Make sure backoff was set for the right period. - // This is 1 second (initial backoff) minus 1ms for the tick needed above. - EXPECT_EQ(run_after_duration, std::chrono::seconds(1)); - run_after_duration.reset(); - // Start a new call now, which will be queued and then eventually - // resumed when the next fetch happens. - state = RequestMetadataState::NewInstance( - absl::OkStatus(), "authorization: foo", /*expect_delay=*/true); - state->RunRequestMetadataTest(creds_.get(), kTestUrlScheme, kTestAuthority, - kTestPath); - // Tick until the next fetch starts. - creds_->AddResult(MakeToken("foo")); - event_engine_->TickUntilIdle(); - EXPECT_EQ(creds_->num_fetches(), 2); - // A call started now should use the new cached data. - state = RequestMetadataState::NewInstance( - absl::OkStatus(), "authorization: foo", /*expect_delay=*/false); - state->RunRequestMetadataTest(creds_.get(), kTestUrlScheme, kTestAuthority, - kTestPath); - EXPECT_EQ(creds_->num_fetches(), 2); -} - -TEST_F(TokenFetcherCredentialsTest, Backoff) { - const absl::Status kExpectedError = absl::UnavailableError("bummer, dude"); - absl::optional run_after_duration; - event_engine_->SetRunAfterDurationCallback( - [&](FuzzingEventEngine::Duration duration) { - run_after_duration = duration; - }); - ExecCtx exec_ctx; - creds_->AddResult(kExpectedError); - // First request will trigger a fetch, which will fail. - auto state = RequestMetadataState::NewInstance(kExpectedError, "", - /*expect_delay=*/true); - state->RunRequestMetadataTest(creds_.get(), kTestUrlScheme, kTestAuthority, - kTestPath); - EXPECT_EQ(creds_->num_fetches(), 1); - while (!run_after_duration.has_value()) event_engine_->Tick(); - // Make sure backoff was set for the right period. - EXPECT_EQ(run_after_duration, std::chrono::seconds(1)); - run_after_duration.reset(); - // Start a new call now, which will be queued and then eventually - // resumed when the next fetch happens. - state = RequestMetadataState::NewInstance(kExpectedError, "", - /*expect_delay=*/true); - state->RunRequestMetadataTest(creds_.get(), kTestUrlScheme, kTestAuthority, - kTestPath); - // Tick until the next fetch fails and the backoff timer starts again. - creds_->AddResult(kExpectedError); - while (!run_after_duration.has_value()) event_engine_->Tick(); - EXPECT_EQ(creds_->num_fetches(), 2); - // The backoff time should be longer now. We account for jitter here. - EXPECT_EQ(run_after_duration, std::chrono::milliseconds(1600)) - << "actual: " << run_after_duration->count(); - run_after_duration.reset(); - // Start another new call to trigger another new fetch once the - // backoff expires. - state = RequestMetadataState::NewInstance(kExpectedError, "", - /*expect_delay=*/true); - state->RunRequestMetadataTest(creds_.get(), kTestUrlScheme, kTestAuthority, - kTestPath); - // Tick until the next fetch starts. - creds_->AddResult(kExpectedError); - while (!run_after_duration.has_value()) event_engine_->Tick(); - EXPECT_EQ(creds_->num_fetches(), 3); - // Check backoff time again. - EXPECT_EQ(run_after_duration, std::chrono::milliseconds(2560)) - << "actual: " << run_after_duration->count(); -} - -TEST_F(TokenFetcherCredentialsTest, FetchNotStartedAfterBackoffWithoutRpc) { - const absl::Status kExpectedError = absl::UnavailableError("bummer, dude"); - absl::optional run_after_duration; - event_engine_->SetRunAfterDurationCallback( - [&](FuzzingEventEngine::Duration duration) { - run_after_duration = duration; - }); - ExecCtx exec_ctx; - creds_->AddResult(kExpectedError); - // First request will trigger a fetch, which will fail. - auto state = RequestMetadataState::NewInstance(kExpectedError, "", - /*expect_delay=*/true); - state->RunRequestMetadataTest(creds_.get(), kTestUrlScheme, kTestAuthority, - kTestPath); - EXPECT_EQ(creds_->num_fetches(), 1); - while (!run_after_duration.has_value()) event_engine_->Tick(); - // Make sure backoff was set for the right period. - EXPECT_EQ(run_after_duration, std::chrono::seconds(1)); - run_after_duration.reset(); - // Tick until the backoff expires. No new fetch should be started. - event_engine_->TickUntilIdle(); - EXPECT_EQ(creds_->num_fetches(), 1); - // Now start a new request, which will trigger a new fetch. - creds_->AddResult(MakeToken("foo")); - state = RequestMetadataState::NewInstance( - absl::OkStatus(), "authorization: foo", /*expect_delay=*/true); - state->RunRequestMetadataTest(creds_.get(), kTestUrlScheme, kTestAuthority, - kTestPath); - EXPECT_EQ(creds_->num_fetches(), 2); -} - -TEST_F(TokenFetcherCredentialsTest, ShutdownWhileBackoffTimerPending) { - const absl::Status kExpectedError = absl::UnavailableError("bummer, dude"); - absl::optional run_after_duration; - event_engine_->SetRunAfterDurationCallback( - [&](FuzzingEventEngine::Duration duration) { - run_after_duration = duration; - }); - ExecCtx exec_ctx; - creds_->AddResult(kExpectedError); - // First request will trigger a fetch, which will fail. - auto state = RequestMetadataState::NewInstance(kExpectedError, "", - /*expect_delay=*/true); - state->RunRequestMetadataTest(creds_.get(), kTestUrlScheme, kTestAuthority, - kTestPath); - EXPECT_EQ(creds_->num_fetches(), 1); - while (!run_after_duration.has_value()) event_engine_->Tick(); - // Make sure backoff was set for the right period. - EXPECT_EQ(run_after_duration, std::chrono::seconds(1)); - run_after_duration.reset(); - // Do nothing else. Make sure the creds shut down correctly. -} - // The subclass of ExternalAccountCredentials for testing. // ExternalAccountCredentials is an abstract class so we can't directly test // against it. @@ -2813,6 +2484,8 @@ TEST_F(CredentialsTest, grpc_version_string())); } +using grpc_event_engine::experimental::FuzzingEventEngine; + class ExternalAccountCredentialsTest : public ::testing::Test { protected: void SetUp() override { @@ -3066,8 +2739,7 @@ TEST_F(ExternalAccountCredentialsTest, FailureInvalidTokenUrl) { HttpRequest::SetOverride(httpcli_get_should_not_be_called, httpcli_post_should_not_be_called, httpcli_put_should_not_be_called); - // TODO(roth): This should return UNAUTHENTICATED. - grpc_error_handle expected_error = absl::UnknownError( + grpc_error_handle expected_error = GRPC_ERROR_CREATE( "error fetching oauth2 token: Invalid token url: " "invalid_token_url. Error: INVALID_ARGUMENT: Could not parse " "'scheme' from uri 'invalid_token_url'. Scheme not found."); @@ -3105,8 +2777,7 @@ TEST_F(ExternalAccountCredentialsTest, HttpRequest::SetOverride(httpcli_get_should_not_be_called, external_account_creds_httpcli_post_success, httpcli_put_should_not_be_called); - // TODO(roth): This should return UNAUTHENTICATED. - grpc_error_handle expected_error = absl::UnknownError( + grpc_error_handle expected_error = GRPC_ERROR_CREATE( "error fetching oauth2 token: Invalid service account impersonation url: " "invalid_service_account_impersonation_url. Error: INVALID_ARGUMENT: " "Could not parse 'scheme' from uri " @@ -3146,8 +2817,7 @@ TEST_F(ExternalAccountCredentialsTest, httpcli_get_should_not_be_called, external_account_creds_httpcli_post_failure_token_exchange_response_missing_access_token, httpcli_put_should_not_be_called); - // TODO(roth): This should return UNAUTHENTICATED. - grpc_error_handle expected_error = absl::UnknownError( + grpc_error_handle expected_error = GRPC_ERROR_CREATE( "error fetching oauth2 token: Missing or invalid access_token in " "{\"not_access_token\":\"not_access_token\",\"expires_in\":3599, " "\"token_type\":\"Bearer\"}."); @@ -3435,8 +3105,7 @@ TEST_F(ExternalAccountCredentialsTest, HttpRequest::SetOverride(httpcli_get_should_not_be_called, httpcli_post_should_not_be_called, httpcli_put_should_not_be_called); - // TODO(roth): This should return UNAVAILABLE. - grpc_error_handle expected_error = absl::InternalError( + grpc_error_handle expected_error = GRPC_ERROR_CREATE( "error fetching oauth2 token: Failed to load file: " "non_exisiting_file due to error(fdopen): No such file or directory"); auto state = RequestMetadataState::NewInstance(expected_error, {}); @@ -3486,8 +3155,7 @@ TEST_F(ExternalAccountCredentialsTest, HttpRequest::SetOverride(httpcli_get_should_not_be_called, httpcli_post_should_not_be_called, httpcli_put_should_not_be_called); - // TODO(roth): This should return UNAUTHENTICATED. - grpc_error_handle expected_error = absl::UnknownError( + grpc_error_handle expected_error = GRPC_ERROR_CREATE( "error fetching oauth2 token: The content of the file is not a " "valid json object."); auto state = RequestMetadataState::NewInstance(expected_error, {}); @@ -4153,8 +3821,7 @@ TEST_F(ExternalAccountCredentialsTest, ASSERT_TRUE(creds.ok()) << creds.status(); ASSERT_NE(*creds, nullptr); EXPECT_EQ((*creds)->min_security_level(), GRPC_PRIVACY_AND_INTEGRITY); - // TODO(roth): This should return UNAUTHENTICATED. - grpc_error_handle expected_error = absl::UnknownError( + grpc_error_handle expected_error = GRPC_ERROR_CREATE( "error fetching oauth2 token: Creating aws request signer failed."); auto state = RequestMetadataState::NewInstance(expected_error, {}); HttpRequest::SetOverride(aws_external_account_creds_httpcli_get_success, @@ -4195,8 +3862,7 @@ TEST_F(ExternalAccountCredentialsTest, ASSERT_TRUE(creds.ok()) << creds.status(); ASSERT_NE(*creds, nullptr); EXPECT_EQ((*creds)->min_security_level(), GRPC_PRIVACY_AND_INTEGRITY); - // TODO(roth): This should return UNAUTHENTICATED. - grpc_error_handle expected_error = absl::UnknownError( + grpc_error_handle expected_error = GRPC_ERROR_CREATE( "error fetching oauth2 token: " "Missing role name when retrieving signing keys."); auto state = RequestMetadataState::NewInstance(expected_error, {}); @@ -4488,190 +4154,6 @@ TEST_F(CredentialsTest, TestXdsCredentialsCompareFailure) { grpc_channel_credentials_release(xds_creds_2); } -class GcpServiceAccountIdentityCredentialsTest : public ::testing::Test { - protected: - void SetUp() override { - g_http_status = 200; - g_audience = ""; - g_token = nullptr; - g_on_http_request_error = nullptr; - HttpRequest::SetOverride(HttpGetOverride, httpcli_post_should_not_be_called, - httpcli_put_should_not_be_called); - } - - void TearDown() override { - HttpRequest::SetOverride(nullptr, nullptr, nullptr); - } - - static void ValidateHttpRequest(const grpc_http_request* request, - const URI& uri) { - EXPECT_EQ(uri.authority(), "metadata.google.internal."); - EXPECT_EQ(uri.path(), - "/computeMetadata/v1/instance/service-accounts/default/identity"); - EXPECT_THAT( - uri.query_parameter_map(), - ::testing::ElementsAre(::testing::Pair("audience", g_audience))); - ASSERT_EQ(request->hdr_count, 1); - EXPECT_EQ(absl::string_view(request->hdrs[0].key), "Metadata-Flavor"); - EXPECT_EQ(absl::string_view(request->hdrs[0].value), "Google"); - } - - static int HttpGetOverride(const grpc_http_request* request, const URI& uri, - Timestamp /*deadline*/, grpc_closure* on_done, - grpc_http_response* response) { - // Validate request. - ValidateHttpRequest(request, uri); - // Generate response. - *response = http_response(g_http_status, g_token == nullptr ? "" : g_token); - ExecCtx::Run(DEBUG_LOCATION, on_done, - g_on_http_request_error == nullptr ? absl::OkStatus() - : *g_on_http_request_error); - return 1; - } - - // Constructs a synthetic JWT token that's just valid enough for the - // call creds to extract the expiration date. - static std::string MakeToken(Timestamp expiration) { - gpr_timespec ts = expiration.as_timespec(GPR_CLOCK_REALTIME); - std::string json = absl::StrCat("{\"exp\":", ts.tv_sec, "}"); - return absl::StrCat("foo.", absl::WebSafeBase64Escape(json), ".bar"); - } - - static int g_http_status; - static absl::string_view g_audience; - static const char* g_token; - static absl::Status* g_on_http_request_error; -}; - -int GcpServiceAccountIdentityCredentialsTest::g_http_status; -absl::string_view GcpServiceAccountIdentityCredentialsTest::g_audience; -const char* GcpServiceAccountIdentityCredentialsTest::g_token; -absl::Status* GcpServiceAccountIdentityCredentialsTest::g_on_http_request_error; - -TEST_F(GcpServiceAccountIdentityCredentialsTest, Basic) { - g_audience = "CV-6"; - auto token = MakeToken(Timestamp::Now() + Duration::Hours(1)); - g_token = token.c_str(); - ExecCtx exec_ctx; - auto creds = - MakeRefCounted(g_audience); - CHECK_EQ(creds->min_security_level(), GRPC_PRIVACY_AND_INTEGRITY); - auto state = RequestMetadataState::NewInstance(absl::OkStatus(), g_token); - state->RunRequestMetadataTest(creds.get(), kTestUrlScheme, kTestAuthority, - kTestPath); - ExecCtx::Get()->Flush(); -} - -// HTTP status 429 is mapped to UNAVAILABLE as per -// https://github.com/grpc/grpc/blob/master/doc/http-grpc-status-mapping.md. -TEST_F(GcpServiceAccountIdentityCredentialsTest, FailsWithHttpStatus429) { - g_audience = "CV-5_Midway"; - g_http_status = 429; - ExecCtx exec_ctx; - auto creds = - MakeRefCounted(g_audience); - CHECK_EQ(creds->min_security_level(), GRPC_PRIVACY_AND_INTEGRITY); - auto state = RequestMetadataState::NewInstance( - absl::UnavailableError("JWT fetch failed with status 429"), ""); - state->RunRequestMetadataTest(creds.get(), kTestUrlScheme, kTestAuthority, - kTestPath); - ExecCtx::Get()->Flush(); -} - -// HTTP status 400 is mapped to INTERNAL as per -// https://github.com/grpc/grpc/blob/master/doc/http-grpc-status-mapping.md, -// so it should be rewritten as UNAUTHENTICATED. -TEST_F(GcpServiceAccountIdentityCredentialsTest, FailsWithHttpStatus400) { - g_audience = "CV-8_SantaCruzIslands"; - g_http_status = 400; - ExecCtx exec_ctx; - auto creds = - MakeRefCounted(g_audience); - CHECK_EQ(creds->min_security_level(), GRPC_PRIVACY_AND_INTEGRITY); - auto state = RequestMetadataState::NewInstance( - absl::UnauthenticatedError("JWT fetch failed with status 400"), ""); - state->RunRequestMetadataTest(creds.get(), kTestUrlScheme, kTestAuthority, - kTestPath); - ExecCtx::Get()->Flush(); -} - -TEST_F(GcpServiceAccountIdentityCredentialsTest, FailsWithHttpIOError) { - g_audience = "CV-2_CoralSea"; - absl::Status status = absl::InternalError("uh oh"); - g_on_http_request_error = &status; - ExecCtx exec_ctx; - auto creds = - MakeRefCounted(g_audience); - CHECK_EQ(creds->min_security_level(), GRPC_PRIVACY_AND_INTEGRITY); - auto state = RequestMetadataState::NewInstance( - absl::UnavailableError("INTERNAL:uh oh"), ""); - state->RunRequestMetadataTest(creds.get(), kTestUrlScheme, kTestAuthority, - kTestPath); - ExecCtx::Get()->Flush(); -} - -TEST_F(GcpServiceAccountIdentityCredentialsTest, TokenHasWrongNumberOfDots) { - g_audience = "CV-7_Guadalcanal"; - std::string bad_token = "foo.bar"; - g_token = bad_token.c_str(); - ExecCtx exec_ctx; - auto creds = - MakeRefCounted(g_audience); - CHECK_EQ(creds->min_security_level(), GRPC_PRIVACY_AND_INTEGRITY); - auto state = RequestMetadataState::NewInstance( - absl::UnauthenticatedError("error parsing JWT token"), ""); - state->RunRequestMetadataTest(creds.get(), kTestUrlScheme, kTestAuthority, - kTestPath); - ExecCtx::Get()->Flush(); -} - -TEST_F(GcpServiceAccountIdentityCredentialsTest, TokenPayloadNotBase64) { - g_audience = "CVE-56_Makin"; - std::string bad_token = "foo.&.bar"; - g_token = bad_token.c_str(); - ExecCtx exec_ctx; - auto creds = - MakeRefCounted(g_audience); - CHECK_EQ(creds->min_security_level(), GRPC_PRIVACY_AND_INTEGRITY); - auto state = RequestMetadataState::NewInstance( - absl::UnauthenticatedError("error parsing JWT token"), ""); - state->RunRequestMetadataTest(creds.get(), kTestUrlScheme, kTestAuthority, - kTestPath); - ExecCtx::Get()->Flush(); -} - -TEST_F(GcpServiceAccountIdentityCredentialsTest, TokenPayloadNotJson) { - g_audience = "CVE-73_Samar"; - std::string bad_token = - absl::StrCat("foo.", absl::WebSafeBase64Escape("xxx"), ".bar"); - g_token = bad_token.c_str(); - ExecCtx exec_ctx; - auto creds = - MakeRefCounted(g_audience); - CHECK_EQ(creds->min_security_level(), GRPC_PRIVACY_AND_INTEGRITY); - auto state = RequestMetadataState::NewInstance( - absl::UnauthenticatedError("error parsing JWT token"), ""); - state->RunRequestMetadataTest(creds.get(), kTestUrlScheme, kTestAuthority, - kTestPath); - ExecCtx::Get()->Flush(); -} - -TEST_F(GcpServiceAccountIdentityCredentialsTest, TokenInvalidExpiration) { - g_audience = "CVL-23_Leyte"; - std::string bad_token = absl::StrCat( - "foo.", absl::WebSafeBase64Escape("{\"exp\":\"foo\"}"), ".bar"); - g_token = bad_token.c_str(); - ExecCtx exec_ctx; - auto creds = - MakeRefCounted(g_audience); - CHECK_EQ(creds->min_security_level(), GRPC_PRIVACY_AND_INTEGRITY); - auto state = RequestMetadataState::NewInstance( - absl::UnauthenticatedError("error parsing JWT token"), ""); - state->RunRequestMetadataTest(creds.get(), kTestUrlScheme, kTestAuthority, - kTestPath); - ExecCtx::Get()->Flush(); -} - } // namespace } // namespace grpc_core