Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance. Project price only 1 $
You can buy this project and download/modify it how often you want.
/*
* Copyright 2023 Google LLC
*
* 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
*
* https://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.
*/
package com.google.cloud.alloydb;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.base.Preconditions;
import com.google.common.util.concurrent.ListeningScheduledExecutorService;
import com.google.common.util.concurrent.MoreExecutors;
import com.google.errorprone.annotations.concurrent.GuardedBy;
import java.io.Closeable;
import java.io.IOException;
import java.net.Socket;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.Executors;
/**
* InternalConnectorRegistry is a singleton that creates a single Executor, KeyPair, and AlloyDB
* Admin Client for the lifetime of the SocketFactory. When callers are finished with the Connector,
* they should use the InternalConnectorRegistry to shut down all the associated resources.
*/
enum InternalConnectorRegistry implements Closeable {
INSTANCE;
@SuppressWarnings("ImmutableEnumChecker")
private final ListeningScheduledExecutorService executor;
@SuppressWarnings("ImmutableEnumChecker")
private CredentialFactoryProvider credentialFactoryProvider;
@SuppressWarnings("ImmutableEnumChecker")
private ConcurrentHashMap unnamedConnectors;
@SuppressWarnings("ImmutableEnumChecker")
private ConcurrentHashMap namedConnectors;
@SuppressWarnings("ImmutableEnumChecker")
private final Object shutdownGuard = new Object();
@SuppressWarnings("ImmutableEnumChecker")
@GuardedBy("shutdownGuard")
private boolean shutdown = false;
@SuppressWarnings("ImmutableEnumChecker")
private List userAgents = new ArrayList<>();
private static final String USER_AGENT = "alloydb-java-connector/" + Version.VERSION;
InternalConnectorRegistry() {
// During refresh, each instance consumes 2 threads from the thread pool. By using 8 threads,
// there should be enough free threads so that there will not be a deadlock. Most users
// configure 3 or fewer instances, requiring 6 threads during refresh. By setting
// this to 8, it's enough threads for most users, plus a safety factor of 2.
this.executor = MoreExecutors.listeningDecorator(Executors.newScheduledThreadPool(8));
this.unnamedConnectors = new ConcurrentHashMap<>();
this.namedConnectors = new ConcurrentHashMap<>();
this.credentialFactoryProvider = new CredentialFactoryProvider();
this.addArtifactId(USER_AGENT);
}
/** Test use only: Set a new CredentialFactoryProvider */
@VisibleForTesting
void setCredentialFactoryProvider(CredentialFactoryProvider credentialFactoryProvider) {
this.credentialFactoryProvider = credentialFactoryProvider;
}
/**
* Internal use only: Creates a socket representing a connection to a AlloyDB instance.
*
* @param config used to configure the connection.
* @return the newly created Socket.
* @throws IOException if error occurs during socket creation.
*/
public Socket connect(ConnectionConfig config) throws IOException {
synchronized (shutdownGuard) {
if (shutdown) {
throw new IllegalStateException("ConnectorRegistry was shut down.");
}
}
if (config.getNamedConnector() != null) {
Connector connector = getNamedConnector(config.getNamedConnector());
return connector.connect(config.withConnectorConfig(connector.getConfig()));
}
// Validate parameters
Preconditions.checkArgument(
config.getInstanceName() != null,
"alloydbInstance property not set. Please specify this property in the JDBC URL or the "
+ "connection Properties");
return getConnector(config).connect(config);
}
/** Register the configuration for a named connector. */
public void register(String name, ConnectorConfig config) {
synchronized (shutdownGuard) {
if (shutdown) {
throw new IllegalStateException("ConnectorRegistry was shut down.");
}
}
if (this.namedConnectors.containsKey(name)) {
throw new IllegalArgumentException("Named connection " + name + " exists.");
}
this.namedConnectors.put(name, createConnector(config));
}
/** Close a named connector, stopping the refresh process and removing it from the registry. */
public void close(String name) {
synchronized (shutdownGuard) {
if (shutdown) {
throw new IllegalStateException("ConnectorRegistry was shut down.");
}
}
Connector connector = namedConnectors.remove(name);
if (connector == null) {
throw new IllegalArgumentException("Named connection " + name + " does not exist.");
}
connector.close();
}
/** Shutdown all connectors. */
private void shutdownConnectors() {
this.unnamedConnectors.forEach((key, c) -> c.close());
this.unnamedConnectors.clear();
this.namedConnectors.forEach((key, c) -> c.close());
this.namedConnectors.clear();
}
@Override
public void close() {
shutdownInstance();
}
/** Calls shutdown on the singleton. */
public void resetInstance() {
shutdownConnectors();
}
/** Calls shutdown on the singleton. */
public void shutdownInstance() {
synchronized (shutdownGuard) {
shutdown = true;
shutdownConnectors();
this.executor.shutdown();
}
}
/**
* Sets the default string which is appended to the AlloyDB Admin API client User-Agent header.
*/
public void addArtifactId(String artifactId) {
if (!userAgents.contains(artifactId)) {
userAgents.add(artifactId);
}
}
/**
* Returns the default string which is appended to the AlloyDB Admin API client User-Agent header.
*/
@VisibleForTesting
String getUserAgents() {
return String.join(" ", userAgents);
}
private Connector getConnector(ConnectionConfig config) {
return unnamedConnectors.computeIfAbsent(
config.getConnectorConfig(), k -> createConnector(config.getConnectorConfig()));
}
private Connector createConnector(ConnectorConfig config) {
CredentialFactory instanceCredentialFactory =
credentialFactoryProvider.getInstanceCredentialFactory(config);
DefaultConnectionInfoRepositoryFactory connectionInfoRepositoryFactory =
new DefaultConnectionInfoRepositoryFactory(executor, getUserAgents());
DefaultConnectionInfoRepository connectionInfoRepository =
connectionInfoRepositoryFactory.create(instanceCredentialFactory, config);
AccessTokenSupplier accessTokenSupplier =
new DefaultAccessTokenSupplier(instanceCredentialFactory);
return new Connector(
config,
executor,
connectionInfoRepository,
RsaKeyPairGenerator.generateKeyPair(),
new DefaultConnectionInfoCacheFactory(),
new ConcurrentHashMap<>(),
accessTokenSupplier,
getUserAgents());
}
private Connector getNamedConnector(String name) {
Connector connector = namedConnectors.get(name);
if (connector == null) {
throw new IllegalArgumentException("Named connection " + name + " does not exist.");
}
return connector;
}
}