/** * Copyright (c) 2012, Andy Janata * All rights reserved. * * Redistribution and use in source and binary forms, with or without modification, are permitted * provided that the following conditions are met: * * * Redistributions of source code must retain the above copyright notice, this list of conditions * and the following disclaimer. * * Redistributions in binary form must reproduce the above copyright notice, this list of * conditions and the following disclaimer in the documentation and/or other materials provided * with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND * FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY * WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ package net.socialgamer.cah; import java.io.File; import java.io.FileReader; import java.util.Date; import java.util.Properties; import java.util.concurrent.ScheduledThreadPoolExecutor; import java.util.concurrent.TimeUnit; import javax.servlet.ServletContext; import javax.servlet.ServletContextEvent; import net.socialgamer.cah.cardcast.CardcastModule; import net.socialgamer.cah.cardcast.CardcastService; import net.socialgamer.cah.task.BroadcastGameListUpdateTask; import net.socialgamer.cah.task.UserPingTask; import org.apache.log4j.Logger; import org.apache.log4j.PropertyConfigurator; import com.google.inject.Guice; import com.google.inject.Injector; import com.google.inject.servlet.GuiceServletContextListener; /** * Class with things that need to be done when the servlet context is created and destroyed. Creates * and stores a Guice injector, stores the time the server was started, and creates a thread to * check for any clients which have stopped responding. * * @author Andy Janata (ajanata@socialgamer.net) */ public class StartupUtils extends GuiceServletContextListener { private static final Logger LOG = Logger.getLogger(StartupUtils.class); /** * Context attribute key name for the Guice injector. */ public static final String INJECTOR = "injector"; /** * Delay before the disconnected client timer is started when the server starts, in milliseconds. */ private static final long PING_START_DELAY = 60 * 1000; /** * Delay between invocations of the disconnected client timer, in milliseconds. */ private static final long PING_CHECK_DELAY = 5 * 1000; /** * Delay before the "update game list" broadcast timer is started, in milliseconds. */ private static final long BROADCAST_UPDATE_START_DELAY = TimeUnit.SECONDS.toMillis(60); /** * Delay between invocations of the "update game list" broadcast timer, in milliseconds. */ private static final long BROADCAST_UPDATE_DELAY = TimeUnit.SECONDS.toMillis(60); /** * Context attribute key name for the time the server was started. */ public static final String DATE_NAME = "started_at"; /** * Context attribute key name for whether verbose request and response logging is enabled. */ public static final String VERBOSE_DEBUG = "verbose_debug"; /** * The time the server was started. */ private Date serverStarted; @Override public void contextDestroyed(final ServletContextEvent contextEvent) { final ServletContext context = contextEvent.getServletContext(); final Injector injector = (Injector) context.getAttribute(INJECTOR); final ScheduledThreadPoolExecutor timer = injector .getInstance(ScheduledThreadPoolExecutor.class); timer.shutdownNow(); context.removeAttribute(INJECTOR); context.removeAttribute(DATE_NAME); super.contextDestroyed(contextEvent); } @Override public void contextInitialized(final ServletContextEvent contextEvent) { final ServletContext context = contextEvent.getServletContext(); final Injector injector = getInjector(); final ScheduledThreadPoolExecutor timer = injector .getInstance(ScheduledThreadPoolExecutor.class); final UserPingTask ping = injector.getInstance(UserPingTask.class); timer.scheduleAtFixedRate(ping, PING_START_DELAY, PING_CHECK_DELAY, TimeUnit.MILLISECONDS); final BroadcastGameListUpdateTask broadcastUpdate = injector .getInstance(BroadcastGameListUpdateTask.class); timer.scheduleAtFixedRate(broadcastUpdate, BROADCAST_UPDATE_START_DELAY, BROADCAST_UPDATE_DELAY, TimeUnit.MILLISECONDS); serverStarted = new Date(); context.setAttribute(INJECTOR, injector); context.setAttribute(DATE_NAME, serverStarted); reconfigureLogging(contextEvent.getServletContext()); reloadProperties(contextEvent.getServletContext()); CardcastService.hackSslVerifier(); } public static void reloadProperties(final ServletContext context) { LOG.info("Reloading pyx.properties"); final Injector injector = (Injector) context.getAttribute(INJECTOR); final Properties props = injector.getInstance(Properties.class); final File propsFile = new File(context.getRealPath("/WEB-INF/pyx.properties")); try { synchronized (props) { props.clear(); props.load(new FileReader(propsFile)); } } catch (final Exception e) { // we should probably do something? e.printStackTrace(); } } public static void reconfigureLogging(final ServletContext context) { LOG.info("Reloading log4j.properties"); PropertyConfigurator.configure(context.getRealPath( "/WEB-INF/log4j.properties")); } @Override protected Injector getInjector() { return Guice.createInjector(new CahModule(), new CardcastModule()); } }