blob: fcd611382bd328936e5355597a27c2667d095db9 [file] [log] [blame]
/*
* 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.sling.engine.impl.request;
import javax.servlet.Servlet;
import javax.servlet.ServletConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.util.Collections;
import org.apache.sling.api.SlingHttpServletRequest;
import org.apache.sling.api.SlingHttpServletResponse;
import org.apache.sling.api.request.RequestProgressTracker;
import org.apache.sling.api.request.TooManyCallsException;
import org.apache.sling.engine.impl.SlingHttpServletRequestImpl;
import org.apache.sling.engine.impl.SlingHttpServletResponseImpl;
import org.apache.sling.engine.impl.SlingRequestProcessorImpl;
import org.jmock.Expectations;
import org.jmock.Mockery;
import org.jmock.imposters.ByteBuddyClassImposteriser;
import org.junit.Before;
import org.junit.Test;
import static org.junit.Assert.*;
public class RequestDataTest {
private Mockery context;
private RequestData requestData;
private HttpServletRequest req;
private HttpServletResponse resp;
private SlingHttpServletRequest slingRequest;
private SlingHttpServletResponse slingResponse;
@Before
public void setup() throws ServletException, IOException {
context = new Mockery() {
{
setImposteriser(ByteBuddyClassImposteriser.INSTANCE);
}
};
req = context.mock(HttpServletRequest.class);
resp = context.mock(HttpServletResponse.class);
final ContentData contentData = context.mock(ContentData.class);
final Servlet servlet = context.mock(Servlet.class);
final ServletConfig servletConfig = context.mock(ServletConfig.class);
final SlingRequestProcessorImpl processor = context.mock(SlingRequestProcessorImpl.class);
context.checking(new Expectations() {
{
allowing(req).getServletPath();
will(returnValue("/"));
allowing(req).getPathInfo();
will(returnValue(""));
allowing(req).getMethod();
will(returnValue("GET"));
allowing(req).setAttribute(with(any(String.class)), with(any(Object.class)));
allowing(req).setAttribute(with(any(String.class)), with(aNull(Object.class)));
allowing(contentData).getServlet();
will(returnValue(servlet));
allowing(servlet).getServletConfig();
will(returnValue(servletConfig));
allowing(contentData).getRequestPathInfo();
allowing(servlet).service(with(any(ServletRequest.class)), with(any(ServletResponse.class)));
allowing(servletConfig).getServletName();
will(returnValue("SERVLET_NAME"));
allowing(req).getAttribute(RequestProgressTracker.class.getName());
will(returnValue(null));
allowing(processor).getMaxCallCounter();
will(returnValue(2));
allowing(processor).getAdditionalResponseHeaders();
will(returnValue(Collections.emptyList()));
}
});
requestData = new RequestData(processor, req, resp, false, false, true) {
@Override
public ContentData getContentData() {
return contentData;
}
};
slingRequest = new SlingHttpServletRequestImpl(requestData, req);
slingResponse = new SlingHttpServletResponseImpl(requestData, resp);
}
private void assertTooManyCallsException(int failAtCall) throws Exception {
for (int i = 0; i < failAtCall - 1; i++) {
RequestData.service(slingRequest, slingResponse);
}
try {
RequestData.service(slingRequest, slingResponse);
fail("Expected RequestData.service to fail when called " + failAtCall + " times");
} catch (TooManyCallsException tme) {
// as expected
}
}
@Test
public void testTooManyCallsDefault() throws Exception {
context.checking(new Expectations() {
{
allowing(req).getAttribute(with(any(String.class)));
will(returnValue(null));
}
});
assertTooManyCallsException(3);
}
@Test
public void testTooManyCallsOverride() throws Exception {
context.checking(new Expectations() {
{
allowing(req).getAttribute(with(any(String.class)));
will(returnValue(1));
}
});
assertTooManyCallsException(2);
}
@Test
public void testConsecutiveDots() {
assertValidRequest(true, "/path/content../test");
assertValidRequest(false, "/../path/content../test");
assertValidRequest(false, "/../path/content/.../test");
assertValidRequest(false, "../path/content/.../test");
assertValidRequest(false, "/content/.../test");
}
@Test
public void testConsecutiveDotsAfterPathSeparator() {
assertValidRequest(false, "/path/....");
assertValidRequest(false, "/path/..");
assertValidRequest(true, "/path/foo..");
assertValidRequest(true, "/path/..foo..");
}
@Test
public void testDots() {
assertValidRequest(false, "/a/.../b");
assertValidRequest(false, "/a/............../b");
assertValidRequest(true, "/a/..........helloo......./b");
assertValidRequest(true, "/path/content./test");
assertValidRequest(true, "/./path/content./test");
assertValidRequest(true, "/./path/content/./test");
assertValidRequest(true, "./path/content/./test");
assertValidRequest(true, "/content/./test");
}
@Test
public void testNullResolutionPath() {
assertValidRequest(true, null);
}
@Test
public void testDotsAnd5B() {
assertValidRequest(false, "/a/..[[./b");
assertValidRequest(false, "/a/[............../b");
assertValidRequest(true, "/a/..........helloo......./b");
assertValidRequest(false, "/a/[..");
}
@Test
public void testDotsAnd7D() {
assertValidRequest(false, "/a/..}}./b");
assertValidRequest(false, "/a/}............../b");
assertValidRequest(false, "/a/..}..}./b");
assertValidRequest(false, "/a/}..");
assertValidRequest(false, "/a/..}");
assertValidRequest(true, "/a/}./b");
assertValidRequest(true, "/a//b/c");
}
@Test
public void testDotsAnd7D5B() {
assertValidRequest(false, "/a/..}[./b");
assertValidRequest(false, "/a/}....[........../b");
assertValidRequest(true, "/a/..........[helloo}......./b");
assertValidRequest(false, "/a/[}....");
assertValidRequest(false, "/a/....}[");
assertValidRequest(true, "/a/.}[");
}
@Test
public void testValidRequest() {
// HttpRequest with valid path
assertValidRequest(true, "/path");
}
private static void assertValidRequest(boolean expected, String path) {
assertEquals("Expected " + expected + " for " + path, expected, RequestData.isValidRequest(path));
}
}