Package org.apache.http.impl.client.cache

Source Code of org.apache.http.impl.client.cache.TestStaleWhileRevalidationReleasesConnection

/*
* ====================================================================
* 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.
* ====================================================================
*
* This software consists of voluntary contributions made by many
* individuals on behalf of the Apache Software Foundation.  For more
* information on the Apache Software Foundation, please see
* <http://www.apache.org/>.
*
*/
package org.apache.http.impl.client.cache;

import org.apache.http.HttpEntity;
import org.apache.http.HttpException;
import org.apache.http.HttpRequest;
import org.apache.http.HttpResponse;
import org.apache.http.MethodNotSupportedException;
import org.apache.http.Header;
import org.apache.http.client.ClientProtocolException;
import org.apache.http.client.HttpClient;
import org.apache.http.client.cache.CacheResponseStatus;
import org.apache.http.client.cache.HttpCacheContext;
import org.apache.http.client.config.RequestConfig;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.entity.ByteArrayEntity;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClientBuilder;
import org.apache.http.protocol.BasicHttpContext;
import org.apache.http.protocol.HttpContext;
import org.apache.http.protocol.HttpRequestHandler;
import org.apache.http.util.EntityUtils;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;

import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.util.Locale;


import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertNull;
import org.apache.http.localserver.LocalTestServer;

/**
* Test that after background validation that a subsequent request for non cached
* conent can be made.  This verifies that the connection has been release back to
* the pool by the AsynchronousValidationRequest.
*/
public class TestStaleWhileRevalidationReleasesConnection {

    private static final EchoViaHeaderHandler cacheHandler = new EchoViaHeaderHandler();

    protected LocalTestServer localServer;
    private int port;
    private CloseableHttpClient client;
    private final String url = "/static/dom";
    private final String url2 = "2";


    @Before
    public void start() throws Exception  {
        this.localServer = new LocalTestServer(null, null);
        this.localServer.register(url +"*", new EchoViaHeaderHandler());
        localServer.setTimeout(5000);
        this.localServer.start();

        port = this.localServer.getServiceAddress().getPort();

        final CacheConfig cacheConfig = CacheConfig.custom()
                .setMaxCacheEntries(100)
                .setMaxObjectSize(15) //1574
                .setAsynchronousWorkerIdleLifetimeSecs(60)
                .setAsynchronousWorkersMax(1)
                .setAsynchronousWorkersCore(1)
                .setRevalidationQueueSize(100)
                .setSharedCache(true)
                .build();

        final HttpClientBuilder clientBuilder = CachingHttpClientBuilder.create().setCacheConfig(cacheConfig);
        clientBuilder.setMaxConnTotal(1);
        clientBuilder.setMaxConnPerRoute(1);

        final RequestConfig config = RequestConfig.custom()
                .setSocketTimeout(10000)
                .setConnectTimeout(10000)
                .setConnectionRequestTimeout(1000)
                .build();

        clientBuilder.setDefaultRequestConfig(config);


        client = clientBuilder.build();
    }

    @After
    public void stop() {
        if (this.localServer != null) {
            try {
                this.localServer.stop();
            } catch(final Exception e) {
                e.printStackTrace();
            }
        }

        try {
            client.close();
        } catch(final IOException e) {
            e.printStackTrace();
        }
    }

    @Test
    public void testStaleWhileRevalidate() {
        final String urlToCall = "http://localhost:"+port + url;
        final HttpContext localContext = new BasicHttpContext();
        Exception requestException = null;

        // This will fetch from backend.
        requestException = sendRequest(client, localContext,urlToCall,null);
        assertNull(requestException);

        CacheResponseStatus responseStatus = (CacheResponseStatus) localContext.getAttribute(HttpCacheContext.CACHE_RESPONSE_STATUS);
        assertEquals(CacheResponseStatus.CACHE_MISS,responseStatus);

        try {
            Thread.sleep(1000);
        } catch (final Exception e) {

        }
        // These will be cached
        requestException = sendRequest(client, localContext,urlToCall,null);
        assertNull(requestException);

        responseStatus = (CacheResponseStatus) localContext.getAttribute(HttpCacheContext.CACHE_RESPONSE_STATUS);
        assertEquals(CacheResponseStatus.CACHE_HIT,responseStatus);

        requestException = sendRequest(client, localContext,urlToCall,null);
        assertNull(requestException);

        responseStatus = (CacheResponseStatus) localContext.getAttribute(HttpCacheContext.CACHE_RESPONSE_STATUS);
        assertEquals(CacheResponseStatus.CACHE_HIT,responseStatus);

        // wait, so that max-age is expired
        try {
            Thread.sleep(4000);
        } catch (final Exception e) {

        }

        // This will cause a revalidation to occur
        requestException = sendRequest(client, localContext,urlToCall,"This is new content that is bigger than cache limit");
        assertNull(requestException);

        responseStatus = (CacheResponseStatus) localContext.getAttribute(HttpCacheContext.CACHE_RESPONSE_STATUS);
        assertEquals(CacheResponseStatus.CACHE_HIT,responseStatus);

        try {
            Thread.sleep(1000);
        } catch (final Exception e) {

        }

        // fetch a different content This will hang due to connection leak in revalidation
        requestException = sendRequest(client, localContext,urlToCall+url2,null);
        if(requestException!=null) {
            requestException.printStackTrace();
        }
        assertNull(requestException);


    }

    static Exception sendRequest(final HttpClient cachingClient, final HttpContext localContext , final String url, final String content) {
        final HttpGet httpget = new HttpGet(url);
        if(content!=null) {
            httpget.setHeader(cacheHandler.getUserContentHeader(),content);
        }

        HttpResponse response = null;
        try {
            response = cachingClient.execute(httpget, localContext);
            return null;
        } catch (final ClientProtocolException e1) {
            return e1;
        } catch (final IOException e1) {
            return e1;
        } finally {
            if(response!=null) {
                final HttpEntity entity = response.getEntity();
                try {
                    EntityUtils.consume(entity);
                } catch (final IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    public static class EchoViaHeaderHandler
            implements HttpRequestHandler {

        private final String CACHE_CONTROL_HEADER = "Cache-Control";

        private final byte[] DEFAULT_CONTENT;
        private final String DEFAULT_CLIENT_CONTROLLED_CONTENT_HEADER;
        private final String DEFAULT_RESPONSE_CACHE_HEADER;

        // public default constructor
        public EchoViaHeaderHandler() {
            this("ECHO-CONTENT","abc".getBytes(), "public, max-age=3, stale-while-revalidate=5");
        }

        public EchoViaHeaderHandler(final String contentHeader,final byte[] content,
                                    final String defaultResponseCacheHeader) {
            DEFAULT_CLIENT_CONTROLLED_CONTENT_HEADER = contentHeader;
            DEFAULT_CONTENT = content;
            DEFAULT_RESPONSE_CACHE_HEADER = defaultResponseCacheHeader;
        }


        /**
         * Return the header the user can set the content that will be returned by the server
         * @return
         */
        public String getUserContentHeader() {
            return DEFAULT_CLIENT_CONTROLLED_CONTENT_HEADER;
        }

        /**
         * Handles a request by echoing the incoming request entity.
         * If there is no request entity, an empty document is returned.
         *
         * @param request   the request
         * @param response  the response
         * @param context   the context
         *
         * @throws org.apache.http.HttpException    in case of a problem
         * @throws java.io.IOException      in case of an IO problem
         */
        public void handle(final HttpRequest request,
                           final HttpResponse response,
                           final HttpContext context)
                throws HttpException, IOException {

            final String method = request.getRequestLine().getMethod().toUpperCase(Locale.ENGLISH);
            if (!"GET".equals(method) &&
                    !"POST".equals(method) &&
                    !"PUT".equals(method)
                    ) {
                throw new MethodNotSupportedException
                        (method + " not supported by " + getClass().getName());
            }

            response.setStatusCode(org.apache.http.HttpStatus.SC_OK);
            response.addHeader("Cache-Control",getCacheContent(request));
            final byte[] content = getHeaderContent(request);
            final ByteArrayEntity bae = new ByteArrayEntity(content);
            response.setHeader("Connection","keep-alive");

            response.setEntity(bae);

        } // handle


        public byte[] getHeaderContent(final HttpRequest request) {
            final Header contentHeader = request.getFirstHeader(DEFAULT_CLIENT_CONTROLLED_CONTENT_HEADER);
            if(contentHeader!=null) {
                try {
                    return contentHeader.getValue().getBytes("UTF-8");
                } catch(final UnsupportedEncodingException e) {
                    return contentHeader.getValue().getBytes();
                }
            } else {
                return DEFAULT_CONTENT;
            }
        }

        public String getCacheContent(final HttpRequest request) {
            final Header contentHeader = request.getFirstHeader(CACHE_CONTROL_HEADER);
            if(contentHeader!=null) {
                return contentHeader.getValue();
            } else {
                return DEFAULT_RESPONSE_CACHE_HEADER;
            }
        }

    } // class EchoHandler

  }
TOP

Related Classes of org.apache.http.impl.client.cache.TestStaleWhileRevalidationReleasesConnection

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.