sample.ClientConfig.java Source code

Java tutorial

Introduction

Here is the source code for sample.ClientConfig.java

Source

/*
 * Copyright 2014-2016 the original author or authors.
 *
 * 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 sample;

import java.io.IOException;
import java.net.Socket;
import java.util.Collections;
import java.util.Properties;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;

import com.gemstone.gemfire.cache.client.Pool;
import com.gemstone.gemfire.management.membership.ClientMembership;
import com.gemstone.gemfire.management.membership.ClientMembershipEvent;
import com.gemstone.gemfire.management.membership.ClientMembershipListenerAdapter;

import org.springframework.beans.BeansException;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.beans.factory.config.BeanPostProcessor;
import org.springframework.context.annotation.Bean;
import org.springframework.context.support.PropertySourcesPlaceholderConfigurer;
import org.springframework.data.gemfire.client.ClientCacheFactoryBean;
import org.springframework.data.gemfire.client.PoolFactoryBean;
import org.springframework.data.gemfire.support.ConnectionEndpoint;
import org.springframework.session.data.gemfire.config.annotation.web.http.EnableGemFireHttpSession;
import org.springframework.session.data.gemfire.support.GemFireUtils;
import org.springframework.util.Assert;

// tag::class[]
@EnableGemFireHttpSession(maxInactiveIntervalInSeconds = 30) // <1>
public class ClientConfig {

    static final long DEFAULT_WAIT_DURATION = TimeUnit.SECONDS.toMillis(20);
    static final long DEFAULT_WAIT_INTERVAL = 500L;

    static final CountDownLatch latch = new CountDownLatch(1);

    static {
        System.setProperty("gemfire.log-level", logLevel());

        ClientMembership.registerClientMembershipListener(new ClientMembershipListenerAdapter() {
            public void memberJoined(ClientMembershipEvent event) {
                if (!event.isClient()) {
                    latch.countDown();
                }
            }
        });
    }

    private static String logLevel() {
        return System.getProperty("sample.httpsession.gemfire.log-level", "warning");
    }

    @Bean
    static PropertySourcesPlaceholderConfigurer propertySourcesPlaceholderConfigurer() {
        return new PropertySourcesPlaceholderConfigurer();
    }

    @Bean
    Properties gemfireProperties() { // <2>
        return new Properties();
    }

    @Bean
    ClientCacheFactoryBean gemfireCache() { // <4>
        ClientCacheFactoryBean clientCacheFactory = new ClientCacheFactoryBean();

        clientCacheFactory.setClose(true);
        clientCacheFactory.setProperties(gemfireProperties());

        return clientCacheFactory;
    }

    @Bean
    PoolFactoryBean gemfirePool(// <3>
            @Value("${spring.session.data.gemfire.port:" + ServerConfig.SERVER_PORT + "}") int port) {

        PoolFactoryBean poolFactory = new PoolFactoryBean();

        poolFactory.setFreeConnectionTimeout(5000); // 5 seconds
        poolFactory.setKeepAlive(false);
        poolFactory.setMaxConnections(ServerConfig.MAX_CONNECTIONS);
        poolFactory.setPingInterval(TimeUnit.SECONDS.toMillis(5));
        poolFactory.setReadTimeout(2000); // 2 seconds
        poolFactory.setRetryAttempts(2);
        poolFactory.setSubscriptionEnabled(true);
        poolFactory.setThreadLocalConnections(false);

        poolFactory
                .setServers(Collections.singletonList(new ConnectionEndpoint(ServerConfig.SERVER_HOSTNAME, port)));

        return poolFactory;
    }

    @Bean
    BeanPostProcessor gemfireCacheServerReadyBeanPostProcessor(// <5>
            @Value("${spring.session.data.gemfire.port:" + ServerConfig.SERVER_PORT + "}") final int port) {

        return new BeanPostProcessor() {

            public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
                if (bean instanceof PoolFactoryBean || bean instanceof Pool) {
                    Assert.isTrue(waitForCacheServerToStart(ServerConfig.SERVER_HOSTNAME, port),
                            String.format("GemFire Server failed to start [hostname: %1$s, port: %2$d]",
                                    ServerConfig.SERVER_HOSTNAME, port));
                }

                return bean;
            }

            public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
                if (bean instanceof PoolFactoryBean || bean instanceof Pool) {
                    try {
                        latch.await(DEFAULT_WAIT_DURATION, TimeUnit.MILLISECONDS);
                    } catch (InterruptedException e) {
                        Thread.currentThread().interrupt();
                    }
                }

                return bean;
            }
        };
    }
    // end::class[]

    interface Condition {
        boolean evaluate();
    }

    boolean waitForCacheServerToStart(String host, int port) {
        return waitForCacheServerToStart(host, port, DEFAULT_WAIT_DURATION);
    }

    /* (non-Javadoc) */
    boolean waitForCacheServerToStart(final String host, final int port, long duration) {
        return waitOnCondition(new Condition() {
            AtomicBoolean connected = new AtomicBoolean(false);

            public boolean evaluate() {
                Socket socket = null;

                try {
                    // NOTE: this code is not intended to be an atomic, compound action (a
                    // possible race condition);
                    // opening another connection (at the expense of using system
                    // resources) after connectivity
                    // has already been established is not detrimental in this use case
                    if (!connected.get()) {
                        socket = new Socket(host, port);
                        connected.set(true);
                    }
                } catch (IOException ignore) {
                } finally {
                    GemFireUtils.close(socket);
                }

                return connected.get();
            }
        }, duration);
    }

    boolean waitOnCondition(Condition condition) {
        return waitOnCondition(condition, DEFAULT_WAIT_DURATION);
    }

    @SuppressWarnings("all")
    boolean waitOnCondition(Condition condition, long duration) {
        final long timeout = (System.currentTimeMillis() + duration);

        try {
            while (!condition.evaluate() && System.currentTimeMillis() < timeout) {
                synchronized (condition) {
                    TimeUnit.MILLISECONDS.timedWait(condition, DEFAULT_WAIT_INTERVAL);
                }
            }
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
        }

        return condition.evaluate();
    }

}