org.apache.geode.modules.session.TestSessionsBase.java Source code

Java tutorial

Introduction

Here is the source code for org.apache.geode.modules.session.TestSessionsBase.java

Source

/*
 * Licensed to the Apache Software Foundation (ASF) under one or more contributor license
 * agreements. See the NOTICE file distributed with this work for additional information regarding
 * copyright ownership. The ASF licenses this file to You 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 org.apache.geode.modules.session;

import static org.apache.geode.distributed.ConfigurationProperties.*;
import static org.junit.Assert.*;

import java.beans.PropertyChangeEvent;
import java.io.File;
import java.io.IOException;
import java.io.PrintWriter;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;

import com.meterware.httpunit.GetMethodWebRequest;
import com.meterware.httpunit.WebConversation;
import com.meterware.httpunit.WebRequest;
import com.meterware.httpunit.WebResponse;
import org.apache.catalina.core.StandardWrapper;
import org.apache.commons.io.FileUtils;
import org.junit.AfterClass;
import org.junit.Before;
import org.junit.Test;

import org.apache.geode.cache.Region;
import org.apache.geode.internal.AvailablePortHelper;
import org.apache.geode.modules.session.catalina.DeltaSessionManager;
import org.apache.geode.modules.session.catalina.PeerToPeerCacheLifecycleListener;

public abstract class TestSessionsBase {

    private static EmbeddedTomcat server;

    private static Region<String, HttpSession> region;

    private static StandardWrapper servlet;

    protected static DeltaSessionManager sessionManager;

    protected static int port;

    // Set up the servers we need
    public static void setupServer(DeltaSessionManager manager) throws Exception {
        FileUtils.copyDirectory(new File("../resources/test/tomcat"), new File("./tomcat"));
        port = AvailablePortHelper.getRandomAvailableTCPPort();
        server = new EmbeddedTomcat("/test", port, "JVM-1");

        PeerToPeerCacheLifecycleListener p2pListener = new PeerToPeerCacheLifecycleListener();
        p2pListener.setProperty(MCAST_PORT, "0");
        p2pListener.setProperty(LOG_LEVEL, "config");
        server.getEmbedded().addLifecycleListener(p2pListener);
        sessionManager = manager;
        sessionManager.setEnableCommitValve(true);
        server.getRootContext().setManager(sessionManager);

        servlet = server.addServlet("/test/*", "default", CommandServlet.class.getName());
        server.startContainer();

        /*
         * Can only retrieve the region once the container has started up (and the cache has started
         * too).
         */
        region = sessionManager.getSessionCache().getSessionRegion();
    }

    @AfterClass
    public static void teardownClass() throws Exception {
        server.stopContainer();
    }

    /**
     * Reset some data
     */
    @Before
    public void setup() throws Exception {
        sessionManager.setMaxInactiveInterval(30);
        region.clear();
    }

    /**
     * Check that the basics are working
     */
    @Test
    public void testSanity() throws Exception {
        WebConversation wc = new WebConversation();
        WebRequest req = new GetMethodWebRequest(String.format("http://localhost:%d/test", port));
        req.setParameter("cmd", QueryCommand.GET.name());
        req.setParameter("param", "null");
        WebResponse response = wc.getResponse(req);

        assertEquals("JSESSIONID", response.getNewCookieNames()[0]);
    }

    /**
     * Test callback functionality. This is here really just as an example. Callbacks are useful to
     * implement per test actions which can be defined within the actual test method instead of in a
     * separate servlet class.
     */
    @Test
    public void testCallback() throws Exception {
        final String helloWorld = "Hello World";
        Callback c = new Callback() {

            @Override
            public void call(HttpServletRequest request, HttpServletResponse response) throws IOException {
                PrintWriter out = response.getWriter();
                out.write(helloWorld);
            }
        };
        servlet.getServletContext().setAttribute("callback", c);

        WebConversation wc = new WebConversation();
        WebRequest req = new GetMethodWebRequest(String.format("http://localhost:%d/test", port));

        req.setParameter("cmd", QueryCommand.CALLBACK.name());
        req.setParameter("param", "callback");
        WebResponse response = wc.getResponse(req);

        assertEquals(helloWorld, response.getText());
    }

    /**
     * Test that calling session.isNew() works for the initial as well as subsequent requests.
     */
    @Test
    public void testIsNew() throws Exception {
        Callback c = new Callback() {

            @Override
            public void call(HttpServletRequest request, HttpServletResponse response) throws IOException {
                HttpSession session = request.getSession();
                response.getWriter().write(Boolean.toString(session.isNew()));
            }
        };
        servlet.getServletContext().setAttribute("callback", c);

        WebConversation wc = new WebConversation();
        WebRequest req = new GetMethodWebRequest(String.format("http://localhost:%d/test", port));

        req.setParameter("cmd", QueryCommand.CALLBACK.name());
        req.setParameter("param", "callback");
        WebResponse response = wc.getResponse(req);

        assertEquals("true", response.getText());
        response = wc.getResponse(req);

        assertEquals("false", response.getText());
    }

    /**
     * Check that our session persists. The values we pass in as query params are used to set
     * attributes on the session.
     */
    @Test
    public void testSessionPersists1() throws Exception {
        String key = "value_testSessionPersists1";
        String value = "Foo";

        WebConversation wc = new WebConversation();
        WebRequest req = new GetMethodWebRequest(String.format("http://localhost:%d/test", port));
        req.setParameter("cmd", QueryCommand.SET.name());
        req.setParameter("param", key);
        req.setParameter("value", value);
        WebResponse response = wc.getResponse(req);
        String sessionId = response.getNewCookieValue("JSESSIONID");

        assertNotNull("No apparent session cookie", sessionId);

        // The request retains the cookie from the prior response...
        req.setParameter("cmd", QueryCommand.GET.name());
        req.setParameter("param", key);
        req.removeParameter("value");
        response = wc.getResponse(req);

        assertEquals(value, response.getText());
    }

    /**
     * Test that invalidating a session makes it's attributes inaccessible.
     */
    @Test
    public void testInvalidate() throws Exception {
        String key = "value_testInvalidate";
        String value = "Foo";

        WebConversation wc = new WebConversation();
        WebRequest req = new GetMethodWebRequest(String.format("http://localhost:%d/test", port));

        // Set an attribute
        req.setParameter("cmd", QueryCommand.SET.name());
        req.setParameter("param", key);
        req.setParameter("value", value);
        WebResponse response = wc.getResponse(req);

        // Invalidate the session
        req.removeParameter("param");
        req.removeParameter("value");
        req.setParameter("cmd", QueryCommand.INVALIDATE.name());
        wc.getResponse(req);

        // The attribute should not be accessible now...
        req.setParameter("cmd", QueryCommand.GET.name());
        req.setParameter("param", key);
        response = wc.getResponse(req);

        assertEquals("", response.getText());
    }

    /**
     * Test setting the session expiration
     */
    @Test
    public void testSessionExpiration1() throws Exception {
        // TestSessions only live for a second
        sessionManager.setMaxInactiveInterval(1);

        String key = "value_testSessionExpiration1";
        String value = "Foo";

        WebConversation wc = new WebConversation();
        WebRequest req = new GetMethodWebRequest(String.format("http://localhost:%d/test", port));

        // Set an attribute
        req.setParameter("cmd", QueryCommand.SET.name());
        req.setParameter("param", key);
        req.setParameter("value", value);
        WebResponse response = wc.getResponse(req);

        // Sleep a while
        Thread.sleep(2000);

        // The attribute should not be accessible now...
        req.setParameter("cmd", QueryCommand.GET.name());
        req.setParameter("param", key);
        response = wc.getResponse(req);

        assertEquals("", response.getText());
    }

    /**
     * Test setting the session expiration via a property change as would happen under normal
     * deployment conditions.
     */
    @Test
    public void testSessionExpiration2() throws Exception {
        // TestSessions only live for a minute
        sessionManager.propertyChange(new PropertyChangeEvent(server.getRootContext(), "sessionTimeout",
                new Integer(30), new Integer(1)));

        // Check that the value has been set to 60 seconds
        assertEquals(60, sessionManager.getMaxInactiveInterval());
    }

    /**
     * Test that removing a session attribute also removes it from the region
     */
    @Test
    public void testRemoveAttribute() throws Exception {
        String key = "value_testRemoveAttribute";
        String value = "Foo";

        WebConversation wc = new WebConversation();
        WebRequest req = new GetMethodWebRequest(String.format("http://localhost:%d/test", port));

        // Set an attribute
        req.setParameter("cmd", QueryCommand.SET.name());
        req.setParameter("param", key);
        req.setParameter("value", value);
        WebResponse response = wc.getResponse(req);
        String sessionId = response.getNewCookieValue("JSESSIONID");

        // Implicitly remove the attribute
        req.removeParameter("value");
        wc.getResponse(req);

        // The attribute should not be accessible now...
        req.setParameter("cmd", QueryCommand.GET.name());
        req.setParameter("param", key);
        response = wc.getResponse(req);

        assertEquals("", response.getText());
        assertNull(region.get(sessionId).getAttribute(key));
    }

    /**
     * Test that a session attribute gets set into the region too.
     */
    @Test
    public void testBasicRegion() throws Exception {
        String key = "value_testBasicRegion";
        String value = "Foo";

        WebConversation wc = new WebConversation();
        WebRequest req = new GetMethodWebRequest(String.format("http://localhost:%d/test", port));

        // Set an attribute
        req.setParameter("cmd", QueryCommand.SET.name());
        req.setParameter("param", key);
        req.setParameter("value", value);
        WebResponse response = wc.getResponse(req);
        String sessionId = response.getNewCookieValue("JSESSIONID");

        assertEquals(value, region.get(sessionId).getAttribute(key));
    }

    /**
     * Test that a session attribute gets removed from the region when the session is invalidated.
     */
    @Test
    public void testRegionInvalidate() throws Exception {
        String key = "value_testRegionInvalidate";
        String value = "Foo";

        WebConversation wc = new WebConversation();
        WebRequest req = new GetMethodWebRequest(String.format("http://localhost:%d/test", port));

        // Set an attribute
        req.setParameter("cmd", QueryCommand.SET.name());
        req.setParameter("param", key);
        req.setParameter("value", value);
        WebResponse response = wc.getResponse(req);
        String sessionId = response.getNewCookieValue("JSESSIONID");

        // Invalidate the session
        req.removeParameter("param");
        req.removeParameter("value");
        req.setParameter("cmd", QueryCommand.INVALIDATE.name());
        wc.getResponse(req);

        assertNull("The region should not have an entry for this session", region.get(sessionId));
    }

    /**
     * Test that multiple attribute updates, within the same request result in only the latest one
     * being effective.
     */
    @Test
    public void testMultipleAttributeUpdates() throws Exception {
        final String key = "value_testMultipleAttributeUpdates";
        Callback c = new Callback() {

            @Override
            public void call(HttpServletRequest request, HttpServletResponse response) throws IOException {
                HttpSession session = request.getSession();
                for (int i = 0; i < 1000; i++) {
                    session.setAttribute(key, Integer.toString(i));
                }
            }
        };
        servlet.getServletContext().setAttribute("callback", c);

        WebConversation wc = new WebConversation();
        WebRequest req = new GetMethodWebRequest(String.format("http://localhost:%d/test", port));

        // Execute the callback
        req.setParameter("cmd", QueryCommand.CALLBACK.name());
        req.setParameter("param", "callback");
        WebResponse response = wc.getResponse(req);

        String sessionId = response.getNewCookieValue("JSESSIONID");

        assertEquals("999", region.get(sessionId).getAttribute(key));
    }

    /*
     * Test for issue #38 CommitSessionValve throws exception on invalidated sessions
     */
    @Test
    public void testCommitSessionValveInvalidSession() throws Exception {
        Callback c = new Callback() {
            @Override
            public void call(HttpServletRequest request, HttpServletResponse response) throws IOException {
                HttpSession session = request.getSession();
                session.invalidate();
                response.getWriter().write("done");
            }
        };
        servlet.getServletContext().setAttribute("callback", c);

        WebConversation wc = new WebConversation();
        WebRequest req = new GetMethodWebRequest(String.format("http://localhost:%d/test", port));

        // Execute the callback
        req.setParameter("cmd", QueryCommand.CALLBACK.name());
        req.setParameter("param", "callback");
        WebResponse response = wc.getResponse(req);

        assertEquals("done", response.getText());
    }

    /**
     * Test for issue #45 Sessions are being created for every request
     */
    @Test
    public void testExtraSessionsNotCreated() throws Exception {
        Callback c = new Callback() {
            @Override
            public void call(HttpServletRequest request, HttpServletResponse response) throws IOException {
                // Do nothing with sessions
                response.getWriter().write("done");
            }
        };
        servlet.getServletContext().setAttribute("callback", c);

        WebConversation wc = new WebConversation();
        WebRequest req = new GetMethodWebRequest(String.format("http://localhost:%d/test", port));

        // Execute the callback
        req.setParameter("cmd", QueryCommand.CALLBACK.name());
        req.setParameter("param", "callback");
        WebResponse response = wc.getResponse(req);

        assertEquals("done", response.getText());
        assertEquals("The region should be empty", 0, region.size());
    }

    /**
     * Test for issue #46 lastAccessedTime is not updated at the start of the request, but only at the
     * end.
     */
    @Test
    public void testLastAccessedTime() throws Exception {
        Callback c = new Callback() {
            @Override
            public void call(HttpServletRequest request, HttpServletResponse response) throws IOException {
                HttpSession session = request.getSession();
                // Hack to expose the session to our test context
                session.getServletContext().setAttribute("session", session);
                session.setAttribute("lastAccessTime", session.getLastAccessedTime());
                try {
                    Thread.sleep(100);
                } catch (InterruptedException ex) {
                }
                session.setAttribute("somethingElse", 1);
                request.getSession();
                response.getWriter().write("done");
            }
        };
        servlet.getServletContext().setAttribute("callback", c);

        WebConversation wc = new WebConversation();
        WebRequest req = new GetMethodWebRequest(String.format("http://localhost:%d/test", port));

        // Execute the callback
        req.setParameter("cmd", QueryCommand.CALLBACK.name());
        req.setParameter("param", "callback");
        WebResponse response = wc.getResponse(req);

        HttpSession session = (HttpSession) servlet.getServletContext().getAttribute("session");
        Long lastAccess = (Long) session.getAttribute("lastAccessTime");

        assertTrue("Last access time not set correctly: " + lastAccess.longValue() + " not <= "
                + session.getLastAccessedTime(), lastAccess.longValue() <= session.getLastAccessedTime());
    }
}