/* * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER. * * Copyright (c) 2013-2015 Oracle and/or its affiliates. All rights reserved. * * The contents of this file are subject to the terms of either the GNU * General Public License Version 2 only ("GPL") or the Common Development * and Distribution License("CDDL") (collectively, the "License"). You * may not use this file except in compliance with the License. You can * obtain a copy of the License at * http://glassfish.java.net/public/CDDL+GPL_1_1.html * or packager/legal/LICENSE.txt. See the License for the specific * language governing permissions and limitations under the License. * * When distributing the software, include this License Header Notice in each * file and include the License file at packager/legal/LICENSE.txt. * * GPL Classpath Exception: * Oracle designates this particular file as subject to the "Classpath" * exception as provided by Oracle in the GPL Version 2 section of the License * file that accompanied this code. * * Modifications: * If applicable, add the following below the License Header, with the fields * enclosed by brackets [] replaced by your own identifying information: * "Portions Copyright [year] [name of copyright owner]" * * Contributor(s): * If you wish your version of this file to be governed by only the CDDL or * only the GPL Version 2, indicate your decision by adding "[Contributor] * elects to include this software in this distribution under the [CDDL or GPL * Version 2] license." If you don't indicate a single choice of license, a * recipient has the option to distribute your version of this file under * either the CDDL, the GPL Version 2 or to extend the choice of license to * its licensees as provided above. However, if you add GPL Version 2 code * and therefore, elected the GPL Version 2 license, then the option applies * only if the new code is made subject to such option by the copyright * holder. */ package org.glassfish.jersey.server.internal.monitoring; import java.util.concurrent.TimeUnit; import org.junit.Test; import static org.junit.Assert.assertEquals; /** * Tests of {@link TimeWindowStatisticsImpl}. * * @author Miroslav Fuksa * @author Stepan Vavra (stepan.vavra at oracle.com) */ public class TimeWindowStatisticsImplTest { private static final double DELTA = 0.0001; @Test public void test() { final long now = System.currentTimeMillis(); final TimeWindowStatisticsImpl.Builder<Long> builder = new TimeWindowStatisticsImpl.Builder<>( new SlidingWindowTimeReservoir(1000L, TimeUnit.MILLISECONDS, now, TimeUnit.MILLISECONDS)); builder.addRequest(now, 30L); builder.addRequest(now + 300, 100L); builder.addRequest(now + 600, 150L); builder.addRequest(now + 800, 15L); builder.addRequest(now + 999, 60L); builder.addRequest(now + 1000, 95L); check(builder, now + 1000, 6, 15, 150, 75, 6); builder.addRequest(now + 1001, 999L); // the original implementation was supposed to trim the first request, we can only guess why it didn't ... check(builder, now + 1001, 6, 15, 999, 236, 6); } @Test public void test10() { final long now = 0; final TimeWindowStatisticsImpl.Builder<Long> builder = new TimeWindowStatisticsImpl.Builder<>( new SlidingWindowTimeReservoir(10000, TimeUnit.MILLISECONDS, now, TimeUnit.MILLISECONDS)); builder.addRequest(now, 30L); builder.addRequest(now + 300, 100L); builder.addRequest(now + 600, 150L); builder.addRequest(now + 800, 15L); builder.addRequest(now + 999, 60L); builder.addRequest(now + 1000, 95L); builder.addRequest(now + 8001, 600L); // check unfinished interval check(builder, now + 8001, 7, 15, 600, 150, 0.8748906); // the original implementation used chunks for time units and the metrics calculation // was accurate only when aligned with the chunks; now, we're accurate as possible which // is why we don't need to use ratio (to adjust the count of the request in the last chunk check(builder, now + 10900, 3, 60, 600, 251, 0.3); // the original calculation left minimum as '15' which collides with the api doc check(builder, now + 11000, 2, 95, 600, 347, 0.2); } /** * This test shows that current implementation of {@link org.glassfish.jersey.server.monitoring.TimeWindowStatistics} is able * to process information that happened before the last update time. */ @Test public void testRequestInPast() { final long now = 0; final TimeWindowStatisticsImpl.Builder<Long> builder = new TimeWindowStatisticsImpl.Builder<>( new SlidingWindowTimeReservoir(1000, TimeUnit.MILLISECONDS, now, TimeUnit.MILLISECONDS)); builder.addRequest(now, 40L); builder.addRequest(now + 1000, 30L); // this is a request in past which will actually reuse the time 'now + 1000' builder.addRequest(now + 100, 10L); check(builder, now + 1000, 3, 10, 40, 26, 3); // this request in past is so old that it doesn't even fit into the window builder.addRequest(now + 100, 0L); builder.addRequest(now + 1200, 20L); check(builder, now + 1201, 2, 20, 30, 25, 2); // snapshot retrieval in past does return values in past; in fact, time 'now + 1201' is used check(builder, now + 1000, 2, 20, 30, 25, 2); } @Test public void test3s() { final long now = 0; final TimeWindowStatisticsImpl.Builder<Long> builder = new TimeWindowStatisticsImpl.Builder<>( new SlidingWindowTimeReservoir(3000, TimeUnit.MILLISECONDS, now, TimeUnit.MILLISECONDS)); builder.addRequest(now, 99L); builder.addRequest(now + 300, 98L); builder.addRequest(now + 600, 1L); builder.addRequest(now + 1000, 96L); builder.addRequest(now + 1500, 95L); builder.addRequest(now + 2500, 3L); // ... above should be ignored builder.addRequest(now + 3500, 90L); builder.addRequest(now + 3900, 4L); builder.addRequest(now + 3900, 80L); builder.addRequest(now + 4200, 92L); builder.addRequest(now + 4900, 15L); builder.addRequest(now + 5300, 8L); builder.addRequest(now + 5600, 50L); check(builder, now + 6001, 7, 4, 92, 48, 2.333333); } @Test public void testLongPause() { final long now = 0; final TimeWindowStatisticsImpl.Builder<Long> builder = new TimeWindowStatisticsImpl.Builder<>( new SlidingWindowTimeReservoir(60, TimeUnit.SECONDS, now, TimeUnit.MILLISECONDS)); builder.addRequest(now, 99L); final long time = now + 1000 * 60 * 60 * 23; builder.addRequest(time, 95L); builder.addRequest(time + 5, 5L); check(builder, time + 20000, 2, 5, 95, 50, 0.03333); } @Test public void testMultipleRequestsAtTheSameTime() { final long now = 0; final TimeWindowStatisticsImpl.Builder<Long> builder = new TimeWindowStatisticsImpl.Builder<>( new SlidingWindowTimeReservoir(1, TimeUnit.SECONDS, now, TimeUnit.MILLISECONDS)); // put multiple requests at the beginning so that even the COLLISION_BUFFER bounds is tested builder.addRequest(now, 10L); builder.addRequest(now, 20L); builder.addRequest(now, 30L); builder.addRequest(now, 40L); builder.addRequest(now + 1, 50L); // put multiple requests in the middle of the window builder.addRequest(now + 500, 60L); builder.addRequest(now + 500, 70L); check(builder, now + 500, 7, 10, 70, 40, 14); // put multiple requests at the end of the window builder.addRequest(now + 1000, 80L); builder.addRequest(now + 1000, 90L); check(builder, now + 1000, 9, 10, 90, 50, 9); // at 'now + 1001' all the requests from 'now' should be gone check(builder, now + 1001, 5, 50, 90, 70, 5); } @Test public void testExhaustiveRequestsAtTheSameTime() { final long now = 0; final TimeWindowStatisticsImpl.Builder<Long> builder = new TimeWindowStatisticsImpl.Builder<>( new SlidingWindowTimeReservoir(1, TimeUnit.SECONDS, now, TimeUnit.MILLISECONDS)); // put multiple requests at the beginning so that even the COLLISION_BUFFER bounds is tested for (int i = 0; i < 256; ++i) { builder.addRequest(now, 10L); } // add one more request which should be visible at 'now + 1001' builder.addRequest(now + 1, 10L); // put multiple requests in the middle of the window for (int i = 0; i < 256; ++i) { builder.addRequest(now + 500, 10L); } check(builder, now + 500, 256 * 2 + 1, 10, 10, 10, 256 * 2 * 2 + 1 * 2); // put multiple requests at the end of the window for (int i = 0; i < 256; ++i) { builder.addRequest(now + 1000, 10L); } check(builder, now + 1000, 256 * 3 + 1, 10, 10, 10, 256 * 3 + 1); // at 'now + 1001' all the requests from 'now' should be gone check(builder, now + 1001, 256 * 2 + 1, 10, 10, 10, 256 * 2 + 1); // at 'now + 1002' the one additional request we added is gone check(builder, now + 1002, 256 * 2, 10, 10, 10, 256 * 2); } /** * Tests JERSEY-2848 */ @Test public void testGapGreaterThanTimeWindowPause() { final long now = 0; final TimeWindowStatisticsImpl.Builder<Long> builder = new TimeWindowStatisticsImpl.Builder<>( new SlidingWindowTimeReservoir(10, TimeUnit.SECONDS, now, TimeUnit.MILLISECONDS)); builder.addRequest(now, 91L); builder.addRequest(now + 1000, 92L); builder.addRequest(now + 2000, 93L); // we need to add the time of last request + the whole time windows pause + additional time that is greater than unit time // which is 1000 final long time = now + 2000 + 10000 + 1001; // this request addition causes the queue to reset; however, the original implementation didn't reset the total count // and total duration; as a result, the stats in that window became corrupted builder.addRequest(time, 94L); builder.addRequest(time + 1000, 95L); builder.addRequest(time + 2000, 96L); check(builder, time + 3000, 3, 94, 96, 95, 0.3); // this line would pass before JERSEY-2848 was fixed; apparently, the values in this window became corrupted // check(builder, time + 3000, 5, 94, 96, 93, 0.5); } private void check(final TimeWindowStatisticsImpl.Builder builder, final long buildTime, final int totalCount, final int minimumExecTime, final int maximumExecTime, final long average, final double requestsPerSecond) { final TimeWindowStatisticsImpl stat = builder.build(buildTime); assertEquals("Total count does not match!", totalCount, stat.getRequestCount()); assertEquals("Min exec time does not match!", minimumExecTime, stat.getMinimumDuration()); assertEquals("Max exec time does not match!", maximumExecTime, stat.getMaximumDuration()); assertEquals("Average exec time does not match!", average, stat.getAverageDuration()); assertEquals("Requests per seconds does not match!", requestsPerSecond, stat.getRequestsPerSecond(), DELTA); } @Test public void testGeneric() { final TimeWindowStatisticsImpl.Builder<Long> builder = new TimeWindowStatisticsImpl.Builder<>( new SlidingWindowTimeReservoir(10, TimeUnit.SECONDS, 0, TimeUnit.MILLISECONDS)); for (int i = 0; i < 100; i++) { final int requestTime = i * 10000; builder.addRequest(requestTime + 1, (long) i); for (int j = 11; j < 100; j++) { try { final TimeWindowStatisticsImpl stat = builder.build(requestTime + j * 100); assertEquals(1, stat.getRequestCount()); assertEquals(i, stat.getMinimumDuration()); assertEquals(i, stat.getMaximumDuration()); } catch (final AssertionError e) { throw new AssertionError("i=" + i + ", j=" + j, e); } } } } @Test public void testUnlimited() { final TimeWindowStatisticsImpl.Builder<Long> builder = new TimeWindowStatisticsImpl.Builder<>( new UniformTimeReservoir(0, TimeUnit.MILLISECONDS)); check(builder, 0, 0, 0, 0, 0, 0); check(builder, 10000, 0, 0, 0, 0, 0); builder.addRequest(0, 10L); check(builder, 50, 1, 10, 10, 10, 20.0); builder.addRequest(100 + 300, 20L); builder.addRequest(1000 + 600, 30L); builder.addRequest(1587 + 800, 40L); builder.addRequest(5544 + 999, 60L); builder.addRequest(9998 + 1000, 50L); check(builder, 10000, 6, 10, 60, 35, 0.6); } }