All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.wicket.protocol.http.AbstractRequestLogger Maven / Gradle / Ivy

Go to download

Pax Wicket Service is an OSGi extension of the Wicket framework, allowing for dynamic loading and unloading of Wicket components and pageSources.

There is a newer version: 5.0.0
Show newest version
/*
 * 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.wicket.protocol.http;

import static java.lang.System.arraycopy;

import java.util.Arrays;
import java.util.Calendar;
import java.util.Date;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.atomic.AtomicInteger;

import org.apache.wicket.Application;
import org.apache.wicket.MetaDataKey;
import org.apache.wicket.Session;
import org.apache.wicket.request.IRequestHandler;
import org.apache.wicket.request.cycle.RequestCycle;
import org.apache.wicket.settings.IRequestLoggerSettings;
import org.apache.wicket.util.lang.Args;
import org.apache.wicket.util.time.Time;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * Base class that collects request and session information for request logging to enable rich
 * information about the events that transpired during a single request. Typical HTTPD and/or
 * Servlet container log files are unusable for determining what happened in the application since
 * they contain the requested URLs of the form http://example.com/app?wicket:interface:0:0:0, which
 * doesn't convey any useful information. Requestloggers can show which page was the target of the
 * request, and which page was rendered as a response, and anything else: resources, Ajax request,
 * etc.
 * 

* The information in the log files can take any format, depending on the request logger * implementation: currently Wicket supports two formats: a {@link RequestLogger legacy, log4j * compatible format}, and a {@link JsonRequestLogger JSON format}. */ public abstract class AbstractRequestLogger implements IStagedRequestLogger { private static final Logger LOG = LoggerFactory.getLogger(AbstractRequestLogger.class); /** * Key for storing request data in the request cycle's meta data. */ private static MetaDataKey REQUEST_DATA = new MetaDataKey() { private static final long serialVersionUID = 1L; }; /** * Key for storing session data in the request cycle's meta data. */ private static MetaDataKey SESSION_DATA = new MetaDataKey() { private static final long serialVersionUID = 1L; }; private final AtomicInteger totalCreatedSessions = new AtomicInteger(); private final AtomicInteger peakSessions = new AtomicInteger(); private final Map liveSessions; private final AtomicInteger activeRequests = new AtomicInteger(); private final AtomicInteger peakActiveRequests = new AtomicInteger(); /** * Rounded request buffer that keeps the request data for the last N requests in the buffer. */ private RequestData[] requestWindow; /** * Cursor pointing to the current writable location in the buffer. Points to the first empty * slot or if the buffer has been filled completely to the oldest request in the buffer. */ private int indexInWindow = 0; /** * records the total request time across the sliding request window so that it can be used to * calculate the average request time across the window duration. */ private long totalRequestTime = 0l; /** * records the start time of the oldest request across the sliding window so that it can be used * to calculate the average request time across the window duration. */ private Date startTimeOfOldestRequest; /** * Construct. */ public AbstractRequestLogger() { int requestsWindowSize = Application.get() .getRequestLoggerSettings() .getRequestsWindowSize(); requestWindow = new RequestData[requestsWindowSize]; liveSessions = new ConcurrentHashMap(); } public int getCurrentActiveRequestCount() { return activeRequests.get(); } public int getPeakActiveRequestCount() { return peakActiveRequests.get(); } public SessionData[] getLiveSessions() { final SessionData[] sessions = liveSessions.values().toArray( new SessionData[liveSessions.values().size()]); Arrays.sort(sessions); return sessions; } public int getPeakSessions() { return peakSessions.get(); } public List getRequests() { synchronized (requestWindow) { RequestData[] result = new RequestData[hasBufferRolledOver() ? requestWindow.length : indexInWindow]; copyRequestsInOrder(result); return Arrays.asList(result); } } /** * Copies all request data into {@code copy} such that the oldest request is in slot 0 and the * most recent request is in slot {@code copy.length} * * @param copy * the target, has to have a capacity of at least {@code requestWindow.length} */ private void copyRequestsInOrder(RequestData[] copy) { if (hasBufferRolledOver()) { // first copy the oldest requests stored behind the cursor into the copy int oldestPos = indexInWindow + 1; if (oldestPos < requestWindow.length) arraycopy(requestWindow, oldestPos, copy, 0, requestWindow.length - oldestPos); // then append the newer requests stored from index 0 til the cursor position. arraycopy(requestWindow, 0, copy, requestWindow.length - oldestPos, indexInWindow); } else { arraycopy(requestWindow, 0, copy, 0, indexInWindow); } } /** * @return whether the buffer has been filled to capacity at least once */ private boolean hasBufferRolledOver() { return requestWindow[requestWindow.length - 1] != null; } public int getTotalCreatedSessions() { return totalCreatedSessions.get(); } public void objectCreated(Object value) { } public void objectRemoved(Object value) { } public void objectUpdated(Object value) { } public void requestTime(long timeTaken) { RequestData requestdata = RequestCycle.get().getMetaData(REQUEST_DATA); if (requestdata != null) { if (activeRequests.get() > 0) { requestdata.setActiveRequest(activeRequests.decrementAndGet()); } Session session = Session.get(); String sessionId = session.getId(); requestdata.setSessionId(sessionId); Object sessionInfo = getSessionInfo(session); requestdata.setSessionInfo(sessionInfo); long sizeInBytes = -1; if (Application.get().getRequestLoggerSettings().getRecordSessionSize()) { try { sizeInBytes = session.getSizeInBytes(); } catch (Exception e) { // log the error and let the request logging continue (this is what happens in // the detach phase of the request cycle anyway. This provides better // diagnostics). LOG.error( "Exception while determining the size of the session in the request logger: " + e.getMessage(), e); } } requestdata.setSessionSize(sizeInBytes); requestdata.setTimeTaken(timeTaken); addRequest(requestdata); SessionData sessiondata = null; if (sessionId != null) { sessiondata = liveSessions.get(sessionId); if (sessiondata == null) { // if the session has been destroyed during the request by // Session#invalidateNow, retrieve the old session data from the RequestCycle. sessiondata = RequestCycle.get().getMetaData(SESSION_DATA); } if (sessiondata == null) { // passivated session or logger only started after it. sessionCreated(sessionId); sessiondata = liveSessions.get(sessionId); } if (sessiondata != null) { sessiondata.setSessionInfo(sessionInfo); sessiondata.setSessionSize(sizeInBytes); sessiondata.addTimeTaken(timeTaken); RequestCycle.get().setMetaData(SESSION_DATA, sessiondata); } } } } public void sessionCreated(String sessionId) { liveSessions.put(sessionId, new SessionData(sessionId)); if (liveSessions.size() > peakSessions.get()) { peakSessions.set(liveSessions.size()); } totalCreatedSessions.incrementAndGet(); } public void sessionDestroyed(String sessionId) { RequestCycle requestCycle = RequestCycle.get(); SessionData sessionData = liveSessions.remove(sessionId); if (requestCycle != null) requestCycle.setMetaData(SESSION_DATA, sessionData); } protected RequestData getCurrentRequest() { RequestCycle requestCycle = RequestCycle.get(); RequestData rd = requestCycle.getMetaData(REQUEST_DATA); if (rd == null) { rd = new RequestData(); requestCycle.setMetaData(REQUEST_DATA, rd); int activeCount = activeRequests.incrementAndGet(); if (activeCount > peakActiveRequests.get()) { peakActiveRequests.set(activeCount); } } return rd; } public void performLogging() { RequestData requestdata = RequestCycle.get().getMetaData(REQUEST_DATA); SessionData sessiondata = RequestCycle.get().getMetaData(SESSION_DATA); if (requestdata != null) { // log the request- and sessiondata (the latter can be null) log(requestdata, sessiondata); } } protected abstract void log(RequestData rd, SessionData sd); private Object getSessionInfo(Session session) { if (session instanceof ISessionLogInfo) { return ((ISessionLogInfo)session).getSessionInfo(); } return ""; } protected void addRequest(RequestData rd) { // ensure the buffer has the proper installed length resizeBuffer(); synchronized (requestWindow) { // if the requestWindow is a zero-length array, nothing gets stored if (requestWindow.length == 0) return; // use the oldest request data to recalculate the average request time RequestData old = requestWindow[indexInWindow]; // replace the oldest request with the nweset request requestWindow[indexInWindow] = rd; // move the cursor to the next writable position containing the oldest request or if the // buffer hasn't been filled completely the first empty slot indexInWindow = (indexInWindow + 1) % requestWindow.length; if (old != null) { startTimeOfOldestRequest = requestWindow[indexInWindow].getStartDate(); totalRequestTime -= old.getTimeTaken(); } else { if (startTimeOfOldestRequest == null) startTimeOfOldestRequest = rd.getStartDate(); } totalRequestTime += rd.getTimeTaken(); } } private int getWindowSize() { synchronized (requestWindow) { if (requestWindow[requestWindow.length - 1] == null) return indexInWindow; else return requestWindow.length; } } public long getAverageRequestTime() { synchronized (requestWindow) { int windowSize = getWindowSize(); if (windowSize == 0) return 0; return totalRequestTime / windowSize; } } public long getRequestsPerMinute() { synchronized (requestWindow) { int windowSize = getWindowSize(); if (windowSize == 0) return 0; long start = startTimeOfOldestRequest.getTime(); long end = System.currentTimeMillis(); double diff = end - start; return Math.round(windowSize / (diff / 60000.0)); } } public void logEventTarget(IRequestHandler requestHandler) { RequestData requestData = getCurrentRequest(); if (requestData != null) { requestData.setEventTarget(requestHandler); } } public void logRequestedUrl(String url) { getCurrentRequest().setRequestedUrl(url); } public void logResponseTarget(IRequestHandler requestHandler) { RequestData requestData = getCurrentRequest(); if (requestData != null) requestData.setResponseTarget(requestHandler); } /** * Resizes the request buffer to match the * {@link IRequestLoggerSettings#getRequestsWindowSize() configured window size} */ private void resizeBuffer() { int newCapacity = Application.get().getRequestLoggerSettings().getRequestsWindowSize(); // do nothing if the capacity requirement hasn't changed if (newCapacity == requestWindow.length) return; RequestData[] newRequestWindow = new RequestData[newCapacity]; synchronized (requestWindow) { int oldCapacity = requestWindow.length; int oldNumberOfElements = hasBufferRolledOver() ? oldCapacity : indexInWindow; if (newCapacity > oldCapacity) { // increase the capacity of the buffer when more requests need to be stored // and preserve the order of the requests while copying them into the new buffer. copyRequestsInOrder(newRequestWindow); // the next writable position is at the first non-copied element in the buffer indexInWindow = oldNumberOfElements; requestWindow = newRequestWindow; } else if (newCapacity < oldCapacity) { // sort the requests in the current buffer such that the oldest request is in slot 0 RequestData[] sortedRequestWindow = new RequestData[oldCapacity]; copyRequestsInOrder(sortedRequestWindow); // determine the number of elements that need to be copied into the smaller target int numberOfElementsToCopy = Math.min(newCapacity, oldNumberOfElements); // determine the position from where the copying must start int numberOfElementsToSkip = Math.max(0, oldNumberOfElements - numberOfElementsToCopy); // fill the new buffer with the leftovers of the old buffer, skipping the oldest // requests arraycopy(sortedRequestWindow, numberOfElementsToSkip, newRequestWindow, 0, numberOfElementsToCopy); // the next writable position is 0 when the buffer is filled to capacity, or the // number of copied elements when the buffer isn't filled to capacity. indexInWindow = numberOfElementsToCopy >= newCapacity ? 0 : numberOfElementsToCopy; requestWindow = newRequestWindow; } } } /** * Thread-safely formats the passed date in format 'yyyy-MM-dd hh:mm:ss,SSS' with GMT timezone * * @param date * the date to format * @return the formatted date */ protected static String formatDate(final Date date) { Args.notNull(date, "date"); final Calendar cal = Calendar.getInstance(Time.GMT); final StringBuilder buf = new StringBuilder(32); cal.setTimeInMillis(date.getTime()); int year = cal.get(Calendar.YEAR); int month = cal.get(Calendar.MONTH) + 1; int day = cal.get(Calendar.DAY_OF_MONTH); int hours = cal.get(Calendar.HOUR_OF_DAY); int minutes = cal.get(Calendar.MINUTE); int seconds = cal.get(Calendar.SECOND); int millis = cal.get(Calendar.MILLISECOND); buf.append(year); buf.append('-'); buf.append(String.format("%02d", month)); buf.append('-'); buf.append(String.format("%02d", day)); buf.append(' '); buf.append(String.format("%02d", hours)); buf.append(':'); buf.append(String.format("%02d", minutes)); buf.append(':'); buf.append(String.format("%02d", seconds)); buf.append(','); buf.append(String.format("%03d", millis)); return buf.toString(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy