org.apache.wicket.request.cycle.RequestCycle Maven / Gradle / Ivy
/*
* 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.request.cycle;
import org.apache.wicket.Application;
import org.apache.wicket.MetaDataEntry;
import org.apache.wicket.MetaDataKey;
import org.apache.wicket.Page;
import org.apache.wicket.Session;
import org.apache.wicket.ThreadContext;
import org.apache.wicket.event.IEvent;
import org.apache.wicket.event.IEventSink;
import org.apache.wicket.protocol.http.IRequestLogger;
import org.apache.wicket.request.IExceptionMapper;
import org.apache.wicket.request.IRequestCycle;
import org.apache.wicket.request.IRequestHandler;
import org.apache.wicket.request.IRequestMapper;
import org.apache.wicket.request.Request;
import org.apache.wicket.request.RequestHandlerStack;
import org.apache.wicket.request.Response;
import org.apache.wicket.request.Url;
import org.apache.wicket.request.UrlRenderer;
import org.apache.wicket.request.component.IRequestablePage;
import org.apache.wicket.request.handler.BookmarkablePageRequestHandler;
import org.apache.wicket.request.handler.IPageProvider;
import org.apache.wicket.request.handler.PageProvider;
import org.apache.wicket.request.handler.RenderPageRequestHandler;
import org.apache.wicket.request.handler.resource.ResourceReferenceRequestHandler;
import org.apache.wicket.request.mapper.parameter.PageParameters;
import org.apache.wicket.request.resource.ResourceReference;
import org.apache.wicket.util.lang.Args;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* {@link RequestCycle} consists of two steps:
*
* - Resolve request handler
*
- Execute request handler
*
* During {@link IRequestHandler} execution the handler can execute other {@link IRequestHandler}s,
* schedule another {@link IRequestHandler} or replace all {@link IRequestHandler}s on stack with
* another {@link IRequestHandler}.
*
* @see #execute(IRequestHandler)
* @see #scheduleRequestHandlerAfterCurrent(IRequestHandler)
* @see #replaceAllRequestHandlers(IRequestHandler)
*
* @author Matej Knopp
* @author igor.vaynberg
*/
public class RequestCycle implements IRequestCycle, IEventSink
{
private static final Logger log = LoggerFactory.getLogger(RequestCycle.class);
private boolean cleanupFeedbackMessagesOnDetach = true;
private interface IExecutor
{
void execute(T object);
}
/**
* Returns request cycle associated with current thread.
*
* @return request cycle instance or null
if no request cycle is associated with
* current thread.
*/
public static RequestCycle get()
{
return ThreadContext.getRequestCycle();
}
/**
*
* @param requestCycle
*/
private static void set(RequestCycle requestCycle)
{
ThreadContext.setRequestCycle(requestCycle);
}
private Request request;
private final Response originalResponse;
private final IRequestMapper requestMapper;
private final IExceptionMapper exceptionMapper;
private final RequestCycleListenerCollection listeners = new RequestCycleListenerCollection();
private UrlRenderer urlRenderer;
/** MetaDataEntry array. */
private MetaDataEntry>[] metaData;
/** the time that this request cycle object was created. */
private final long startTime = System.currentTimeMillis();
private RequestHandlerStack requestHandlerExecutor;
private Response activeResponse;
/**
* Construct.
*
* @param context
*/
public RequestCycle(RequestCycleContext context)
{
Args.notNull(context, "context");
Args.notNull(context.getRequest(), "context.request");
Args.notNull(context.getResponse(), "context.response");
Args.notNull(context.getRequestMapper(), "context.requestMapper");
Args.notNull(context.getExceptionMapper(), "context.exceptionMapper");
requestHandlerExecutor = new HandlerExecutor();
activeResponse = context.getResponse();
request = context.getRequest();
originalResponse = context.getResponse();
requestMapper = context.getRequestMapper();
exceptionMapper = context.getExceptionMapper();
}
/**
*
* @return a new url renderer
*/
protected UrlRenderer newUrlRenderer()
{
// All URLs will be rendered relative to current request (can be overridden afterwards)
return new UrlRenderer(getRequest());
}
/**
* Get the original response the request was created with. Access to the original response may
* be necessary if the response has been temporarily replaced but the components require methods
* from original response (i.e. cookie methods of WebResponse, etc).
*
* @return The original response object.
*/
public Response getOriginalResponse()
{
return originalResponse;
}
/**
* Returns {@link UrlRenderer} for this {@link RequestCycle}.
*
* @return UrlRenderer instance.
*/
public final UrlRenderer getUrlRenderer()
{
if (urlRenderer == null)
{
urlRenderer = newUrlRenderer();
}
return urlRenderer;
}
/**
* Resolves current request to a {@link IRequestHandler}.
*
* @return RequestHandler instance
*/
protected IRequestHandler resolveRequestHandler()
{
return requestMapper.mapRequest(request);
}
/**
* @return How many times will Wicket attempt to render the exception request handler before
* giving up.
*/
protected int getExceptionRetryCount()
{
return 10;
}
/**
* Processes the request.
*
* @return true
if the request resolved to a Wicket request, false
* otherwise.
*/
public boolean processRequest()
{
try
{
set(this);
listeners.onBeginRequest(this);
onBeginRequest();
IRequestHandler handler = resolveRequestHandler();
if (handler != null)
{
listeners.onRequestHandlerResolved(this, handler);
requestHandlerExecutor.execute(handler);
return true;
}
// Did not find any suitable handler, thus not executing the request
log.debug(
"No suitable handler found for URL {}, falling back to container to process this request",
request.getUrl());
}
catch (Exception e)
{
IRequestHandler handler = handleException(e);
if (handler != null)
{
listeners.onExceptionRequestHandlerResolved(this, handler, e);
executeExceptionRequestHandler(handler, getExceptionRetryCount());
}
else
{
log.error("Error during request processing. URL=" + request.getUrl(), e);
}
return true;
}
finally
{
set(null);
}
return false;
}
/**
* Convenience method that processes the request and detaches the {@link RequestCycle}.
*
* @return true
if the request resolved to a Wicket request, false
* otherwise.
*/
public boolean processRequestAndDetach()
{
boolean result;
try
{
result = processRequest();
}
finally
{
detach();
}
return result;
}
/**
*
* @param handler
* @param retryCount
*/
private void executeExceptionRequestHandler(final IRequestHandler handler, final int retryCount)
{
scheduleRequestHandlerAfterCurrent(null);
try
{
requestHandlerExecutor.execute(handler);
}
catch (Exception e)
{
if (retryCount > 0)
{
IRequestHandler next = handleException(e);
if (handler != null)
{
executeExceptionRequestHandler(next, retryCount - 1);
return;
}
}
log.error("Error during processing error message", e);
}
}
/**
* Return {@link IRequestHandler} for the given exception.
*
* @param e
* @return RequestHandler instance
*/
protected IRequestHandler handleException(final Exception e)
{
IRequestHandler handler = listeners.onException(this, e);
if (handler != null)
{
return handler;
}
return exceptionMapper.map(e);
}
/**
* @return current request
*/
public Request getRequest()
{
return request;
}
/**
* INTERNAL This method is for internal Wicket use. Do not call it yourself unless you know what
* you are doing.
*
* @param request
*/
public void setRequest(Request request)
{
// It would be mighty nice if request was final. However during multipart it needs to be set
// to
// MultipartServletWebRequest by Form. It can't be done before creating the request cycle
// (in wicket filter)
// because only the form knows max upload size
this.request = request;
}
/**
* Sets the metadata for this request cycle using the given key. If the metadata object is not
* of the correct type for the metadata key, an IllegalArgumentException will be thrown. For
* information on creating MetaDataKeys, see {@link MetaDataKey}.
*
* @param key
* The singleton key for the metadata
* @param object
* The metadata object
* @param
* @throws IllegalArgumentException
* @see MetaDataKey
*/
public final void setMetaData(final MetaDataKey key, final T object)
{
metaData = key.set(metaData, object);
}
/**
* Gets metadata for this request cycle using the given key.
*
* @param
* The type of the metadata
*
* @param key
* The key for the data
* @return The metadata or null if no metadata was found for the given key
* @see MetaDataKey
*/
public final T getMetaData(final MetaDataKey key)
{
return key.get(metaData);
}
/**
* Returns URL for the request handler or null
if the handler couldn't have been
* encoded.
*
* @param handler
* @return Url instance or null
*/
public Url mapUrlFor(IRequestHandler handler)
{
return requestMapper.mapHandler(handler);
}
/**
* Returns a {@link Url} for the resource reference
*
* @param reference
* resource reference
* @param params
* parameters for the resource or {@code null} if none
* @return {@link Url} for the reference
*/
public Url mapUrlFor(ResourceReference reference, PageParameters params)
{
return mapUrlFor(new ResourceReferenceRequestHandler(reference, params));
}
/**
* Returns a bookmarkable URL that references a given page class using a given set of page
* parameters. Since the URL which is returned contains all information necessary to instantiate
* and render the page, it can be stored in a user's browser as a stable bookmark.
*
* @param
*
* @param pageClass
* Class of page
* @param parameters
* Parameters to page or {@code null} if none
* @return Bookmarkable URL to page
*/
public final Url mapUrlFor(final Class pageClass,
final PageParameters parameters)
{
IRequestHandler handler = new BookmarkablePageRequestHandler(new PageProvider(pageClass,
parameters));
return mapUrlFor(handler);
}
/**
* Returns a rendered {@link Url} for the resource reference
*
* @param reference
* resource reference
* @param params
* parameters for the resource or {@code null} if none
* @return {@link Url} for the reference
*/
public final CharSequence urlFor(ResourceReference reference, PageParameters params)
{
return renderUrl(mapUrlFor(reference, params));
}
/**
* Returns a rendered bookmarkable URL that references a given page class using a given set of
* page parameters. Since the URL which is returned contains all information necessary to
* instantiate and render the page, it can be stored in a user's browser as a stable bookmark.
*
* @param
*
* @param pageClass
* Class of page
* @param parameters
* Parameters to page or {@code null} if none
* @return Bookmarkable URL to page
*/
public final CharSequence urlFor(final Class pageClass,
final PageParameters parameters)
{
return renderUrl(mapUrlFor(pageClass, parameters));
}
/**
* Returns the rendered URL for the request handler or null
if the handler couldn't
* have been rendered.
*
* The resulting URL will be relative to current page.
*
* @param handler
* @return Url String or null
*/
public CharSequence urlFor(IRequestHandler handler)
{
return renderUrl(mapUrlFor(handler));
}
private String renderUrl(Url url)
{
if (url != null)
{
return getOriginalResponse().encodeURL(getUrlRenderer().renderUrl(url));
}
else
{
return null;
}
}
/**
* Detaches {@link RequestCycle} state. Called after request processing is complete
*/
public final void detach()
{
set(this);
try
{
onDetach();
}
finally
{
try
{
onInternalDetach();
}
finally
{
set(null);
}
}
}
private void onInternalDetach()
{
if (Session.exists())
{
Session.get().internalDetach();
}
if (Application.exists())
{
IRequestLogger requestLogger = Application.get().getRequestLogger();
if (requestLogger != null)
{
requestLogger.requestTime((System.currentTimeMillis() - startTime));
}
}
}
/**
* Called after request processing is complete, usually takes care of detaching state
*/
public void onDetach()
{
if (cleanupFeedbackMessagesOnDetach)
{
if (Session.exists())
{
Session.get().cleanupFeedbackMessages();
}
}
try
{
onEndRequest();
listeners.onEndRequest(this);
}
catch (RuntimeException e)
{
log.error("Exception occurred during onEndRequest", e);
}
try
{
requestHandlerExecutor.detach();
}
finally
{
listeners.onDetach(this);
}
if (Session.exists())
{
Session.get().detach();
}
}
/**
* Convenience method for setting next page to be rendered.
*
* @param page
*/
public void setResponsePage(IRequestablePage page)
{
scheduleRequestHandlerAfterCurrent(new RenderPageRequestHandler(new PageProvider(page),
RenderPageRequestHandler.RedirectPolicy.AUTO_REDIRECT));
}
/**
* Convenience method for setting next page to be rendered.
*
* @param pageClass
*/
public void setResponsePage(Class extends IRequestablePage> pageClass)
{
IPageProvider provider = new PageProvider(pageClass, null);
scheduleRequestHandlerAfterCurrent(new RenderPageRequestHandler(provider,
RenderPageRequestHandler.RedirectPolicy.AUTO_REDIRECT));
}
/**
* Convenience method for setting next page to be rendered.
*
* @param pageClass
* @param parameters
*/
public void setResponsePage(Class extends IRequestablePage> pageClass,
PageParameters parameters)
{
IPageProvider provider = new PageProvider(pageClass, parameters);
scheduleRequestHandlerAfterCurrent(new RenderPageRequestHandler(provider,
RenderPageRequestHandler.RedirectPolicy.AUTO_REDIRECT));
}
/**
* Gets whether or not feedback messages are to be cleaned up on detach.
*
* @return true if they are
*/
public boolean isCleanupFeedbackMessagesOnDetach()
{
return cleanupFeedbackMessagesOnDetach;
}
/**
* Sets whether or not feedback messages should be cleaned up on detach.
*
* @param cleanupFeedbackMessagesOnDetach
* true if you want them to be cleaned up
*/
public void setCleanupFeedbackMessagesOnDetach(boolean cleanupFeedbackMessagesOnDetach)
{
this.cleanupFeedbackMessagesOnDetach = cleanupFeedbackMessagesOnDetach;
}
/**
* @return The start time for this request
*/
public final long getStartTime()
{
return startTime;
}
/** {@inheritDoc} */
public void onEvent(IEvent> event)
{
}
/**
* Called when the request cycle object is beginning its response
*/
protected void onBeginRequest()
{
}
/**
* Called when the request cycle object has finished its response
*/
protected void onEndRequest()
{
}
/**
* @return listeners
*/
public RequestCycleListenerCollection getListeners()
{
return listeners;
}
/**
* {@inheritDoc}
*/
public Response getResponse()
{
return activeResponse;
}
/**
* {@inheritDoc}
*/
public Response setResponse(final Response response)
{
Response current = activeResponse;
activeResponse = response;
return current;
}
/**
* {@inheritDoc}
*/
public void scheduleRequestHandlerAfterCurrent(IRequestHandler handler)
{
// just delegating the call to {@link IRequestHandlerExecutor} and invoking listeners
requestHandlerExecutor.schedule(handler);
// only forward calls to the listeners when handler is null
if (handler != null)
listeners.onRequestHandlerScheduled(this, handler);
}
/**
* @see RequestHandlerStack#getActive()
* @return active handler on executor
*/
public IRequestHandler getActiveRequestHandler()
{
return requestHandlerExecutor.getActive();
}
/**
* @see RequestHandlerStack#next()
* @return the handler scheduled to be executed after current by the executor
*/
public IRequestHandler getRequestHandlerScheduledAfterCurrent()
{
return requestHandlerExecutor.next();
}
/**
* @see RequestHandlerStack#replaceAll(IRequestHandler)
* @param handler
*/
public void replaceAllRequestHandlers(final IRequestHandler handler)
{
requestHandlerExecutor.replaceAll(handler);
}
/**
* Adapts {@link RequestHandlerStack} to this {@link RequestCycle}
*
* @author Igor Vaynberg
*/
private class HandlerExecutor extends RequestHandlerStack
{
@Override
protected void respond(IRequestHandler handler)
{
Response originalResponse = getResponse();
try
{
handler.respond(RequestCycle.this);
}
finally
{
setResponse(originalResponse);
}
}
@Override
protected void detach(IRequestHandler handler)
{
handler.detach(RequestCycle.this);
}
}
}