204 lines
6.5 KiB
Java
204 lines
6.5 KiB
Java
/*
|
|
* Copyright 2002-2015 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 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.config.GemfireConstants;
|
|
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;
|
|
|
|
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;
|
|
|
|
// 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",
|
|
System.getProperty("sample.httpsession.gemfire.log-level", "warning"));
|
|
|
|
ClientMembership.registerClientMembershipListener(
|
|
new ClientMembershipListenerAdapter() {
|
|
public void memberJoined(ClientMembershipEvent event) {
|
|
if (!event.isClient()) {
|
|
latch.countDown();
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
@Bean
|
|
PropertySourcesPlaceholderConfigurer propertySourcesPlaceholderConfigurer() {
|
|
return new PropertySourcesPlaceholderConfigurer();
|
|
}
|
|
|
|
@Bean
|
|
Properties gemfireProperties() { // <2>
|
|
return new Properties();
|
|
}
|
|
|
|
@Bean(name = GemfireConstants.DEFAULT_GEMFIRE_POOL_NAME)
|
|
PoolFactoryBean gemfirePool( // <3>
|
|
@Value("${spring.session.data.gemfire.port:"+ServerConfig.SERVER_PORT+"}") int port) {
|
|
|
|
PoolFactoryBean poolFactory = new PoolFactoryBean();
|
|
|
|
poolFactory.setName(GemfireConstants.DEFAULT_GEMFIRE_POOL_NAME);
|
|
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.setServerEndpoints(Collections.singletonList(new ConnectionEndpoint(
|
|
ServerConfig.SERVER_HOSTNAME, port)));
|
|
|
|
return poolFactory;
|
|
}
|
|
|
|
@Bean
|
|
ClientCacheFactoryBean gemfireCache(Pool gemfirePool) { // <4>
|
|
ClientCacheFactoryBean clientCacheFactory = new ClientCacheFactoryBean();
|
|
|
|
clientCacheFactory.setClose(true);
|
|
clientCacheFactory.setProperties(gemfireProperties());
|
|
clientCacheFactory.setPool(gemfirePool);
|
|
clientCacheFactory.setUseBeanFactoryLocator(false);
|
|
|
|
return clientCacheFactory;
|
|
}
|
|
|
|
@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();
|
|
}
|
|
}
|