예제 #1
0
        /// <summary>
        /// Allocates a session from the pool if possible, or creates a new Spanner Session.
        /// </summary>
        /// <param name="spannerClient">The client to associate with the session. Must not be null.</param>
        /// <param name="project">The project of the session. Must not be null.</param>
        /// <param name="spannerInstance">The instance of the session. Must not be null.</param>
        /// <param name="spannerDatabase">The database of the session. Must not be null.</param>
        /// <param name="options">The transaction options for the session. Must not be null.</param>
        /// <param name="cancellationToken">A cancellation token for the asynchronous operation.</param>
        /// <returns>A task which, when completed, will return the session.</returns>
        public async Task <Session> CreateSessionFromPoolAsync(
            SpannerClient spannerClient,
            string project,
            string spannerInstance,
            string spannerDatabase,
            TransactionOptions options,
            CancellationToken cancellationToken)
        {
            GaxPreconditions.CheckNotNull(spannerClient, nameof(spannerClient));
            GaxPreconditions.CheckNotNull(project, nameof(project));
            GaxPreconditions.CheckNotNull(spannerInstance, nameof(spannerInstance));
            GaxPreconditions.CheckNotNull(spannerDatabase, nameof(spannerDatabase));
            GaxPreconditions.CheckNotNull(options, nameof(options));

            await StartSessionCreatingAsync(cancellationToken).ConfigureAwait(false);

            Session        sessionResult  = null;
            SessionPoolKey sessionPoolKey = default;

            try
            {
                sessionPoolKey = new SessionPoolKey(spannerClient, project, spannerInstance, spannerDatabase);
                SessionPoolImpl targetPool = _poolByClientAndDatabase.GetOrAdd(sessionPoolKey, key => new SessionPoolImpl(key, Options));

                sessionResult = await targetPool.AcquireSessionAsync(options, cancellationToken).ConfigureAwait(false);

                // Refresh the MRU list which tells us what sessions need to be trimmed from the pool when we want
                // to add another poolEntry.
                return(sessionResult);
            }
            finally
            {
                EndSessionCreating(sessionResult, sessionPoolKey);
            }
        }
예제 #2
0
        // Only present for testing.
        internal int GetPoolSize(SpannerClient spannerClient, string project, string spannerInstance, string spannerDatabase)
        {
            GaxPreconditions.CheckNotNull(spannerClient, nameof(spannerClient));
            GaxPreconditions.CheckNotNull(project, nameof(project));
            GaxPreconditions.CheckNotNull(spannerInstance, nameof(spannerInstance));
            GaxPreconditions.CheckNotNull(spannerDatabase, nameof(spannerDatabase));
            var             sessionPoolKey = new SessionPoolKey(spannerClient, project, spannerInstance, spannerDatabase);
            SessionPoolImpl targetPool     = _poolByClientAndDatabase.GetOrAdd(sessionPoolKey, key => new SessionPoolImpl(key, Options));

            return(targetPool.GetPoolSize());
        }
예제 #3
0
        private void EndSessionCreating(Session sessionResult, SessionPoolKey sessionPoolKey)
        {
            lock (_waitSync)
            {
                if (sessionResult != null)
                {
                    _sessionsInUse.AddOrUpdate(sessionResult,
                                               x => sessionPoolKey,
                                               (x, y) => sessionPoolKey);
                }

                _sessionsCreating--;
                LogSessionsInUse();

                //signal more queue entries (in fair order) if we still have room.
                SignalAnyWaitingRequests();
            }
        }
예제 #4
0
        internal int GetPoolSize(
            SpannerClient spannerClient,
            string project,
            string spannerInstance,
            string spannerDatabase)
        {
            project.ThrowIfNullOrEmpty(nameof(project));
            spannerInstance.ThrowIfNullOrEmpty(nameof(spannerInstance));
            spannerDatabase.ThrowIfNullOrEmpty(nameof(spannerDatabase));
            var sessionPoolKey = new SessionPoolKey(spannerClient,
                                                    project,
                                                    spannerInstance,
                                                    spannerDatabase);
            SessionPoolImpl targetPool = _poolByClientAndDatabase.GetOrAdd(sessionPoolKey,
                                                                           key => new SessionPoolImpl(key, Options));

            return(targetPool.GetPoolSize());
        }
예제 #5
0
        /// <summary>
        /// Allocates a session from the pool if possible, or creates a new Spanner Session.
        /// </summary>
        /// <param name="spannerClient"></param>
        /// <param name="project"></param>
        /// <param name="spannerInstance"></param>
        /// <param name="spannerDatabase"></param>
        /// <param name="options"></param>
        /// <param name="cancellationToken"></param>
        /// <returns></returns>
        public static async Task <Session> CreateSessionFromPoolAsync(
            this SpannerClient spannerClient,
            string project,
            string spannerInstance,
            string spannerDatabase,
            TransactionOptions options,
            CancellationToken cancellationToken)
        {
            project.ThrowIfNullOrEmpty(nameof(project));
            spannerInstance.ThrowIfNullOrEmpty(nameof(spannerInstance));
            spannerDatabase.ThrowIfNullOrEmpty(nameof(spannerDatabase));

            await StartSessionCreatingAsync(cancellationToken).ConfigureAwait(false);

            Session        sessionResult  = null;
            SessionPoolKey sessionPoolKey = default(SessionPoolKey);

            try
            {
                sessionPoolKey = new SessionPoolKey(spannerClient,
                                                    project,
                                                    spannerInstance,
                                                    spannerDatabase);
                SessionPoolImpl targetPool = s_poolByClientAndDatabase.GetOrAdd(sessionPoolKey,
                                                                                key => new SessionPoolImpl(key));

                sessionResult = await targetPool.AcquireSessionAsync(options, cancellationToken).ConfigureAwait(false);

                //refresh the mru list which tells us what sessions need to be trimmed from the pool when we want
                // to add another poolEntry.
                ReSortMru(targetPool);
                return(sessionResult);
            } finally
            {
                EndSessionCreating(sessionResult, sessionPoolKey);
            }
        }
예제 #6
0
 internal SessionPoolImpl(SessionPoolKey key)
 {
     Key = key;
 }
예제 #7
0
 internal SessionPoolImpl(SessionPoolKey key, SessionPoolOptions options)
 {
     _options = GaxPreconditions.CheckNotNull(options, nameof(options));
     _sessionCreateThrottle = new SemaphoreSlim(_options.MaximumConcurrentSessionCreates, _options.MaximumConcurrentSessionCreates);
     Key = key;
 }