|
|
|
@ -26,12 +26,15 @@ |
|
|
|
|
#include <grpc/support/log.h> |
|
|
|
|
#include <grpc/support/sync.h> |
|
|
|
|
|
|
|
|
|
#include "src/core/ext/filters/client_channel/lb_policy/grpclb/grpclb.h" |
|
|
|
|
#include "src/core/lib/channel/channel_args.h" |
|
|
|
|
#include "src/core/lib/gpr/env.h" |
|
|
|
|
#include "src/core/lib/gpr/string.h" |
|
|
|
|
#include "src/core/lib/http/httpcli.h" |
|
|
|
|
#include "src/core/lib/http/parser.h" |
|
|
|
|
#include "src/core/lib/iomgr/load_file.h" |
|
|
|
|
#include "src/core/lib/iomgr/polling_entity.h" |
|
|
|
|
#include "src/core/lib/security/credentials/alts/alts_credentials.h" |
|
|
|
|
#include "src/core/lib/security/credentials/google_default/google_default_credentials.h" |
|
|
|
|
#include "src/core/lib/security/credentials/jwt/jwt_credentials.h" |
|
|
|
|
#include "src/core/lib/security/credentials/oauth2/oauth2_credentials.h" |
|
|
|
@ -45,8 +48,8 @@ |
|
|
|
|
|
|
|
|
|
/* -- Default credentials. -- */ |
|
|
|
|
|
|
|
|
|
static grpc_channel_credentials* default_credentials = nullptr; |
|
|
|
|
static int compute_engine_detection_done = 0; |
|
|
|
|
static grpc_channel_credentials* g_default_credentials = nullptr; |
|
|
|
|
static int g_compute_engine_detection_done = 0; |
|
|
|
|
static gpr_mu g_state_mu; |
|
|
|
|
static gpr_mu* g_polling_mu; |
|
|
|
|
static gpr_once g_once = GPR_ONCE_INIT; |
|
|
|
@ -60,6 +63,52 @@ typedef struct { |
|
|
|
|
grpc_http_response response; |
|
|
|
|
} compute_engine_detector; |
|
|
|
|
|
|
|
|
|
static void google_default_credentials_destruct( |
|
|
|
|
grpc_channel_credentials* creds) { |
|
|
|
|
grpc_google_default_channel_credentials* c = |
|
|
|
|
reinterpret_cast<grpc_google_default_channel_credentials*>(creds); |
|
|
|
|
grpc_channel_credentials_unref(c->alts_creds); |
|
|
|
|
grpc_channel_credentials_unref(c->ssl_creds); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static grpc_security_status google_default_create_security_connector( |
|
|
|
|
grpc_channel_credentials* creds, grpc_call_credentials* call_creds, |
|
|
|
|
const char* target, const grpc_channel_args* args, |
|
|
|
|
grpc_channel_security_connector** sc, grpc_channel_args** new_args) { |
|
|
|
|
grpc_google_default_channel_credentials* c = |
|
|
|
|
reinterpret_cast<grpc_google_default_channel_credentials*>(creds); |
|
|
|
|
bool is_grpclb_load_balancer = grpc_channel_arg_get_bool( |
|
|
|
|
grpc_channel_args_find(args, GRPC_ARG_ADDRESS_IS_GRPCLB_LOAD_BALANCER), |
|
|
|
|
false); |
|
|
|
|
bool is_backend_from_grpclb_load_balancer = grpc_channel_arg_get_bool( |
|
|
|
|
grpc_channel_args_find( |
|
|
|
|
args, GRPC_ARG_ADDRESS_IS_BACKEND_FROM_GRPCLB_LOAD_BALANCER), |
|
|
|
|
false); |
|
|
|
|
bool use_alts = |
|
|
|
|
is_grpclb_load_balancer || is_backend_from_grpclb_load_balancer; |
|
|
|
|
grpc_security_status status = GRPC_SECURITY_ERROR; |
|
|
|
|
status = use_alts ? c->alts_creds->vtable->create_security_connector( |
|
|
|
|
c->alts_creds, call_creds, target, args, sc, new_args) |
|
|
|
|
: c->ssl_creds->vtable->create_security_connector( |
|
|
|
|
c->ssl_creds, call_creds, target, args, sc, new_args); |
|
|
|
|
/* grpclb-specific channel args are removed from the channel args set
|
|
|
|
|
* to ensure backends and fallback adresses will have the same set of channel |
|
|
|
|
* args. By doing that, it guarantees the connections to backends will not be |
|
|
|
|
* torn down and re-connected when switching in and out of fallback mode. |
|
|
|
|
*/ |
|
|
|
|
static const char* args_to_remove[] = { |
|
|
|
|
GRPC_ARG_ADDRESS_IS_GRPCLB_LOAD_BALANCER, |
|
|
|
|
GRPC_ARG_ADDRESS_IS_BACKEND_FROM_GRPCLB_LOAD_BALANCER, |
|
|
|
|
}; |
|
|
|
|
*new_args = grpc_channel_args_copy_and_add_and_remove( |
|
|
|
|
args, args_to_remove, GPR_ARRAY_SIZE(args_to_remove), nullptr, 0); |
|
|
|
|
return status; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
static grpc_channel_credentials_vtable google_default_credentials_vtable = { |
|
|
|
|
google_default_credentials_destruct, |
|
|
|
|
google_default_create_security_connector, nullptr}; |
|
|
|
|
|
|
|
|
|
static void on_compute_engine_detection_http_response(void* user_data, |
|
|
|
|
grpc_error* error) { |
|
|
|
|
compute_engine_detector* detector = |
|
|
|
@ -234,8 +283,8 @@ grpc_channel_credentials* grpc_google_default_credentials_create(void) { |
|
|
|
|
|
|
|
|
|
gpr_mu_lock(&g_state_mu); |
|
|
|
|
|
|
|
|
|
if (default_credentials != nullptr) { |
|
|
|
|
result = grpc_channel_credentials_ref(default_credentials); |
|
|
|
|
if (g_default_credentials != nullptr) { |
|
|
|
|
result = grpc_channel_credentials_ref(g_default_credentials); |
|
|
|
|
goto end; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
@ -253,9 +302,9 @@ grpc_channel_credentials* grpc_google_default_credentials_create(void) { |
|
|
|
|
|
|
|
|
|
/* At last try to see if we're on compute engine (do the detection only once
|
|
|
|
|
since it requires a network test). */ |
|
|
|
|
if (!compute_engine_detection_done) { |
|
|
|
|
if (!g_compute_engine_detection_done) { |
|
|
|
|
int need_compute_engine_creds = is_stack_running_on_compute_engine(); |
|
|
|
|
compute_engine_detection_done = 1; |
|
|
|
|
g_compute_engine_detection_done = 1; |
|
|
|
|
if (need_compute_engine_creds) { |
|
|
|
|
call_creds = grpc_google_compute_engine_credentials_create(nullptr); |
|
|
|
|
if (call_creds == nullptr) { |
|
|
|
@ -269,18 +318,25 @@ grpc_channel_credentials* grpc_google_default_credentials_create(void) { |
|
|
|
|
end: |
|
|
|
|
if (result == nullptr) { |
|
|
|
|
if (call_creds != nullptr) { |
|
|
|
|
/* Blend with default ssl credentials and add a global reference so that
|
|
|
|
|
it |
|
|
|
|
can be cached and re-served. */ |
|
|
|
|
grpc_channel_credentials* ssl_creds = |
|
|
|
|
grpc_ssl_credentials_create(nullptr, nullptr, nullptr); |
|
|
|
|
default_credentials = grpc_channel_credentials_ref( |
|
|
|
|
grpc_composite_channel_credentials_create(ssl_creds, call_creds, |
|
|
|
|
nullptr)); |
|
|
|
|
GPR_ASSERT(default_credentials != nullptr); |
|
|
|
|
grpc_channel_credentials_unref(ssl_creds); |
|
|
|
|
/* Create google default credentials. */ |
|
|
|
|
auto creds = static_cast<grpc_google_default_channel_credentials*>( |
|
|
|
|
gpr_zalloc(sizeof(grpc_google_default_channel_credentials))); |
|
|
|
|
creds->base.vtable = &google_default_credentials_vtable; |
|
|
|
|
creds->base.type = GRPC_CHANNEL_CREDENTIALS_TYPE_GOOGLE_DEFAULT; |
|
|
|
|
gpr_ref_init(&creds->base.refcount, 1); |
|
|
|
|
creds->ssl_creds = grpc_ssl_credentials_create(nullptr, nullptr, nullptr); |
|
|
|
|
GPR_ASSERT(creds->ssl_creds != nullptr); |
|
|
|
|
grpc_alts_credentials_options* options = |
|
|
|
|
grpc_alts_credentials_client_options_create(); |
|
|
|
|
creds->alts_creds = grpc_alts_credentials_create(options); |
|
|
|
|
grpc_alts_credentials_options_destroy(options); |
|
|
|
|
/* Add a global reference so that it can be cached and re-served. */ |
|
|
|
|
g_default_credentials = grpc_composite_channel_credentials_create( |
|
|
|
|
&creds->base, call_creds, nullptr); |
|
|
|
|
GPR_ASSERT(g_default_credentials != nullptr); |
|
|
|
|
grpc_channel_credentials_unref(&creds->base); |
|
|
|
|
grpc_call_credentials_unref(call_creds); |
|
|
|
|
result = default_credentials; |
|
|
|
|
result = grpc_channel_credentials_ref(g_default_credentials); |
|
|
|
|
} else { |
|
|
|
|
gpr_log(GPR_ERROR, "Could not create google default credentials."); |
|
|
|
|
} |
|
|
|
@ -299,11 +355,11 @@ void grpc_flush_cached_google_default_credentials(void) { |
|
|
|
|
grpc_core::ExecCtx exec_ctx; |
|
|
|
|
gpr_once_init(&g_once, init_default_credentials); |
|
|
|
|
gpr_mu_lock(&g_state_mu); |
|
|
|
|
if (default_credentials != nullptr) { |
|
|
|
|
grpc_channel_credentials_unref(default_credentials); |
|
|
|
|
default_credentials = nullptr; |
|
|
|
|
if (g_default_credentials != nullptr) { |
|
|
|
|
grpc_channel_credentials_unref(g_default_credentials); |
|
|
|
|
g_default_credentials = nullptr; |
|
|
|
|
} |
|
|
|
|
compute_engine_detection_done = 0; |
|
|
|
|
g_compute_engine_detection_done = 0; |
|
|
|
|
gpr_mu_unlock(&g_state_mu); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|