| /* | 
 |  * | 
 |  * Copyright 2015-2016, Google Inc. | 
 |  * All rights reserved. | 
 |  * | 
 |  * Redistribution and use in source and binary forms, with or without | 
 |  * modification, are permitted provided that the following conditions are | 
 |  * met: | 
 |  * | 
 |  *     * Redistributions of source code must retain the above copyright | 
 |  * notice, this list of conditions and the following disclaimer. | 
 |  *     * Redistributions in binary form must reproduce the above | 
 |  * copyright notice, this list of conditions and the following disclaimer | 
 |  * in the documentation and/or other materials provided with the | 
 |  * distribution. | 
 |  *     * Neither the name of Google Inc. nor the names of its | 
 |  * contributors may be used to endorse or promote products derived from | 
 |  * this software without specific prior written permission. | 
 |  * | 
 |  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | 
 |  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | 
 |  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR | 
 |  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT | 
 |  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | 
 |  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | 
 |  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | 
 |  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | 
 |  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | 
 |  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | 
 |  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
 |  * | 
 |  */ | 
 |  | 
 | #include <grpc++/channel.h> | 
 | #include <grpc++/impl/grpc_library.h> | 
 | #include <grpc++/support/channel_arguments.h> | 
 | #include <grpc/support/log.h> | 
 | #include "src/cpp/client/create_channel_internal.h" | 
 | #include "src/cpp/client/secure_credentials.h" | 
 | #include "src/cpp/common/secure_auth_context.h" | 
 |  | 
 | namespace grpc { | 
 |  | 
 | static internal::GrpcLibraryInitializer g_gli_initializer; | 
 | SecureChannelCredentials::SecureChannelCredentials( | 
 |     grpc_channel_credentials* c_creds) | 
 |     : c_creds_(c_creds) { | 
 |   g_gli_initializer.summon(); | 
 | } | 
 |  | 
 | std::shared_ptr<grpc::Channel> SecureChannelCredentials::CreateChannel( | 
 |     const string& target, const grpc::ChannelArguments& args) { | 
 |   grpc_channel_args channel_args; | 
 |   args.SetChannelArgs(&channel_args); | 
 |   return CreateChannelInternal( | 
 |       args.GetSslTargetNameOverride(), | 
 |       grpc_secure_channel_create(c_creds_, target.c_str(), &channel_args, | 
 |                                  nullptr)); | 
 | } | 
 |  | 
 | SecureCallCredentials::SecureCallCredentials(grpc_call_credentials* c_creds) | 
 |     : c_creds_(c_creds) { | 
 |   internal::GrpcLibraryInitializer gli_initializer; | 
 |   gli_initializer.summon(); | 
 | } | 
 |  | 
 | bool SecureCallCredentials::ApplyToCall(grpc_call* call) { | 
 |   return grpc_call_set_credentials(call, c_creds_) == GRPC_CALL_OK; | 
 | } | 
 |  | 
 | namespace { | 
 | std::shared_ptr<ChannelCredentials> WrapChannelCredentials( | 
 |     grpc_channel_credentials* creds) { | 
 |   return creds == nullptr ? nullptr : std::shared_ptr<ChannelCredentials>( | 
 |                                           new SecureChannelCredentials(creds)); | 
 | } | 
 |  | 
 | std::shared_ptr<CallCredentials> WrapCallCredentials( | 
 |     grpc_call_credentials* creds) { | 
 |   return creds == nullptr ? nullptr : std::shared_ptr<CallCredentials>( | 
 |                                           new SecureCallCredentials(creds)); | 
 | } | 
 | }  // namespace | 
 |  | 
 | std::shared_ptr<ChannelCredentials> GoogleDefaultCredentials() { | 
 |   GrpcLibrary init;  // To call grpc_init(). | 
 |   return WrapChannelCredentials(grpc_google_default_credentials_create()); | 
 | } | 
 |  | 
 | // Builds SSL Credentials given SSL specific options | 
 | std::shared_ptr<ChannelCredentials> SslCredentials( | 
 |     const SslCredentialsOptions& options) { | 
 |   GrpcLibrary init;  // To call grpc_init(). | 
 |   grpc_ssl_pem_key_cert_pair pem_key_cert_pair = { | 
 |       options.pem_private_key.c_str(), options.pem_cert_chain.c_str()}; | 
 |  | 
 |   grpc_channel_credentials* c_creds = grpc_ssl_credentials_create( | 
 |       options.pem_root_certs.empty() ? nullptr : options.pem_root_certs.c_str(), | 
 |       options.pem_private_key.empty() ? nullptr : &pem_key_cert_pair, nullptr); | 
 |   return WrapChannelCredentials(c_creds); | 
 | } | 
 |  | 
 | // Builds credentials for use when running in GCE | 
 | std::shared_ptr<CallCredentials> GoogleComputeEngineCredentials() { | 
 |   GrpcLibrary init;  // To call grpc_init(). | 
 |   return WrapCallCredentials( | 
 |       grpc_google_compute_engine_credentials_create(nullptr)); | 
 | } | 
 |  | 
 | // Builds JWT credentials. | 
 | std::shared_ptr<CallCredentials> ServiceAccountJWTAccessCredentials( | 
 |     const grpc::string& json_key, long token_lifetime_seconds) { | 
 |   GrpcLibrary init;  // To call grpc_init(). | 
 |   if (token_lifetime_seconds <= 0) { | 
 |     gpr_log(GPR_ERROR, | 
 |             "Trying to create JWTCredentials with non-positive lifetime"); | 
 |     return WrapCallCredentials(nullptr); | 
 |   } | 
 |   gpr_timespec lifetime = | 
 |       gpr_time_from_seconds(token_lifetime_seconds, GPR_TIMESPAN); | 
 |   return WrapCallCredentials(grpc_service_account_jwt_access_credentials_create( | 
 |       json_key.c_str(), lifetime, nullptr)); | 
 | } | 
 |  | 
 | // Builds refresh token credentials. | 
 | std::shared_ptr<CallCredentials> GoogleRefreshTokenCredentials( | 
 |     const grpc::string& json_refresh_token) { | 
 |   GrpcLibrary init;  // To call grpc_init(). | 
 |   return WrapCallCredentials(grpc_google_refresh_token_credentials_create( | 
 |       json_refresh_token.c_str(), nullptr)); | 
 | } | 
 |  | 
 | // Builds access token credentials. | 
 | std::shared_ptr<CallCredentials> AccessTokenCredentials( | 
 |     const grpc::string& access_token) { | 
 |   GrpcLibrary init;  // To call grpc_init(). | 
 |   return WrapCallCredentials( | 
 |       grpc_access_token_credentials_create(access_token.c_str(), nullptr)); | 
 | } | 
 |  | 
 | // Builds IAM credentials. | 
 | std::shared_ptr<CallCredentials> GoogleIAMCredentials( | 
 |     const grpc::string& authorization_token, | 
 |     const grpc::string& authority_selector) { | 
 |   GrpcLibrary init;  // To call grpc_init(). | 
 |   return WrapCallCredentials(grpc_google_iam_credentials_create( | 
 |       authorization_token.c_str(), authority_selector.c_str(), nullptr)); | 
 | } | 
 |  | 
 | // Combines one channel credentials and one call credentials into a channel | 
 | // composite credentials. | 
 | std::shared_ptr<ChannelCredentials> CompositeChannelCredentials( | 
 |     const std::shared_ptr<ChannelCredentials>& channel_creds, | 
 |     const std::shared_ptr<CallCredentials>& call_creds) { | 
 |   // Note that we are not saving shared_ptrs to the two credentials passed in | 
 |   // here. This is OK because the underlying C objects (i.e., channel_creds and | 
 |   // call_creds) into grpc_composite_credentials_create will see their refcounts | 
 |   // incremented. | 
 |   SecureChannelCredentials* s_channel_creds = | 
 |       channel_creds->AsSecureCredentials(); | 
 |   SecureCallCredentials* s_call_creds = call_creds->AsSecureCredentials(); | 
 |   if (s_channel_creds && s_call_creds) { | 
 |     return WrapChannelCredentials(grpc_composite_channel_credentials_create( | 
 |         s_channel_creds->GetRawCreds(), s_call_creds->GetRawCreds(), nullptr)); | 
 |   } | 
 |   return nullptr; | 
 | } | 
 |  | 
 | void MetadataCredentialsPluginWrapper::Destroy(void* wrapper) { | 
 |   if (wrapper == nullptr) return; | 
 |   MetadataCredentialsPluginWrapper* w = | 
 |       reinterpret_cast<MetadataCredentialsPluginWrapper*>(wrapper); | 
 |   delete w; | 
 | } | 
 |  | 
 | void MetadataCredentialsPluginWrapper::GetMetadata( | 
 |     void* wrapper, grpc_auth_metadata_context context, | 
 |     grpc_credentials_plugin_metadata_cb cb, void* user_data) { | 
 |   GPR_ASSERT(wrapper); | 
 |   MetadataCredentialsPluginWrapper* w = | 
 |       reinterpret_cast<MetadataCredentialsPluginWrapper*>(wrapper); | 
 |   if (!w->plugin_) { | 
 |     cb(user_data, NULL, 0, GRPC_STATUS_OK, NULL); | 
 |     return; | 
 |   } | 
 |   if (w->plugin_->IsBlocking()) { | 
 |     w->thread_pool_->Add( | 
 |         std::bind(&MetadataCredentialsPluginWrapper::InvokePlugin, w, context, | 
 |                   cb, user_data)); | 
 |   } else { | 
 |     w->InvokePlugin(context, cb, user_data); | 
 |   } | 
 | } | 
 |  | 
 | void MetadataCredentialsPluginWrapper::InvokePlugin( | 
 |     grpc_auth_metadata_context context, grpc_credentials_plugin_metadata_cb cb, | 
 |     void* user_data) { | 
 |   std::multimap<grpc::string, grpc::string> metadata; | 
 |  | 
 |   // const_cast is safe since the SecureAuthContext does not take owndership and | 
 |   // the object is passed as a const ref to plugin_->GetMetadata. | 
 |   SecureAuthContext cpp_channel_auth_context( | 
 |       const_cast<grpc_auth_context*>(context.channel_auth_context), false); | 
 |  | 
 |   Status status = plugin_->GetMetadata(context.service_url, context.method_name, | 
 |                                        cpp_channel_auth_context, &metadata); | 
 |   std::vector<grpc_metadata> md; | 
 |   for (auto it = metadata.begin(); it != metadata.end(); ++it) { | 
 |     grpc_metadata md_entry; | 
 |     md_entry.key = it->first.c_str(); | 
 |     md_entry.value = it->second.data(); | 
 |     md_entry.value_length = it->second.size(); | 
 |     md_entry.flags = 0; | 
 |     md.push_back(md_entry); | 
 |   } | 
 |   cb(user_data, md.empty() ? nullptr : &md[0], md.size(), | 
 |      static_cast<grpc_status_code>(status.error_code()), | 
 |      status.error_message().c_str()); | 
 | } | 
 |  | 
 | MetadataCredentialsPluginWrapper::MetadataCredentialsPluginWrapper( | 
 |     std::unique_ptr<MetadataCredentialsPlugin> plugin) | 
 |     : thread_pool_(CreateDefaultThreadPool()), plugin_(std::move(plugin)) {} | 
 |  | 
 | std::shared_ptr<CallCredentials> MetadataCredentialsFromPlugin( | 
 |     std::unique_ptr<MetadataCredentialsPlugin> plugin) { | 
 |   GrpcLibrary init;  // To call grpc_init(). | 
 |   const char* type = plugin->GetType(); | 
 |   MetadataCredentialsPluginWrapper* wrapper = | 
 |       new MetadataCredentialsPluginWrapper(std::move(plugin)); | 
 |   grpc_metadata_credentials_plugin c_plugin = { | 
 |       MetadataCredentialsPluginWrapper::GetMetadata, | 
 |       MetadataCredentialsPluginWrapper::Destroy, wrapper, type}; | 
 |   return WrapCallCredentials( | 
 |       grpc_metadata_credentials_create_from_plugin(c_plugin, nullptr)); | 
 | } | 
 |  | 
 | }  // namespace grpc |