All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.google.cloud.grpc.SharedResourceHolder Maven / Gradle / Ivy

There is a newer version: 2.49.0
Show newest version
/*
 * 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
 *
 *       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.
 */

package com.google.cloud.grpc;

import com.google.common.base.Preconditions;
import com.google.common.util.concurrent.ThreadFactoryBuilder;
import java.util.IdentityHashMap;
import java.util.concurrent.*;

/**
 * This class was copied from grpc-core to prevent dependence on an unstable API that may be subject
 * to changes
 * (https://github.com/grpc/grpc-java/blob/d07ecbe037d2705a1c9f4b6345581f860e505b56/core/src/main/java/io/grpc/internal/SharedResourceHolder.java)
 *
 * 

A holder for shared resource singletons. * *

Components like client channels and servers need certain resources, e.g. a thread pool, to * run. If the user has not provided such resources, these components will use a default one, which * is shared as a static resource. This class holds these default resources and manages their * life-cycles. * *

A resource is identified by the reference of a {@link Resource} object, which is typically a * singleton, provided to the get() and release() methods. Each Resource object (not its class) maps * to an object cached in the holder. * *

Resources are ref-counted and shut down after a delay when the ref-count reaches zero. */ final class SharedResourceHolder { static final long DESTROY_DELAY_SECONDS = 1; // The sole holder instance. private static final SharedResourceHolder holder = new SharedResourceHolder( new ScheduledExecutorFactory() { @Override public ScheduledExecutorService createScheduledExecutor() { return Executors.newSingleThreadScheduledExecutor( getThreadFactory("grpc-shared-destroyer-%d", true)); } }); private final IdentityHashMap, Instance> instances = new IdentityHashMap<>(); private final ScheduledExecutorFactory destroyerFactory; private ScheduledExecutorService destroyer; // Visible to tests that would need to create instances of the holder. SharedResourceHolder(ScheduledExecutorFactory destroyerFactory) { this.destroyerFactory = destroyerFactory; } private static ThreadFactory getThreadFactory(String nameFormat, boolean daemon) { return new ThreadFactoryBuilder().setDaemon(daemon).setNameFormat(nameFormat).build(); } /** * Try to get an existing instance of the given resource. If an instance does not exist, create a * new one with the given factory. * * @param resource the singleton object that identifies the requested static resource */ public static T get(Resource resource) { return holder.getInternal(resource); } /** * Releases an instance of the given resource. * *

The instance must have been obtained from {@link #get(Resource)}. Otherwise will throw * IllegalArgumentException. * *

Caller must not release a reference more than once. It's advisory that you clear the * reference to the instance with the null returned by this method. * * @param resource the singleton Resource object that identifies the released static resource * @param instance the released static resource * @return a null which the caller can use to clear the reference to that instance. */ public static T release(final Resource resource, final T instance) { return holder.releaseInternal(resource, instance); } /** * Visible to unit tests. * * @see #get(Resource) */ @SuppressWarnings("unchecked") synchronized T getInternal(Resource resource) { Instance instance = instances.get(resource); if (instance == null) { instance = new Instance(resource.create()); instances.put(resource, instance); } if (instance.destroyTask != null) { instance.destroyTask.cancel(false); instance.destroyTask = null; } instance.refcount++; return (T) instance.payload; } /** Visible to unit tests. */ synchronized T releaseInternal(final Resource resource, final T instance) { final Instance cached = instances.get(resource); if (cached == null) { throw new IllegalArgumentException("No cached instance found for " + resource); } Preconditions.checkArgument(instance == cached.payload, "Releasing the wrong instance"); Preconditions.checkState(cached.refcount > 0, "Refcount has already reached zero"); cached.refcount--; if (cached.refcount == 0) { Preconditions.checkState(cached.destroyTask == null, "Destroy task already scheduled"); // Schedule a delayed task to destroy the resource. if (destroyer == null) { destroyer = destroyerFactory.createScheduledExecutor(); } cached.destroyTask = destroyer.schedule( new LogExceptionRunnable( new Runnable() { @Override public void run() { synchronized (SharedResourceHolder.this) { // Refcount may have gone up since the task was scheduled. Re-check it. if (cached.refcount == 0) { try { resource.close(instance); } finally { instances.remove(resource); if (instances.isEmpty()) { destroyer.shutdown(); destroyer = null; } } } } } }), DESTROY_DELAY_SECONDS, TimeUnit.SECONDS); } // Always returning null return null; } /** Defines a resource, and the way to create and destroy instances of it. */ public interface Resource { /** Create a new instance of the resource. */ T create(); /** Destroy the given instance. */ void close(T instance); } interface ScheduledExecutorFactory { ScheduledExecutorService createScheduledExecutor(); } private static class Instance { final Object payload; int refcount; ScheduledFuture destroyTask; Instance(Object payload) { this.payload = payload; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy