Issue #11260 - Allow `QuickStartConfiguration` to be used in mixed contexts environment where some do not have a WEB-INF/quickstart-web.xml (#11261)

* QuickStartConfiguration updated

Signed-off-by: Joakim Erdfelt <joakim.erdfelt@gmail.com>
This commit is contained in:
Joakim Erdfelt 2024-01-25 11:00:13 -06:00 committed by GitHub
parent d8c34c4ccd
commit 6fec10e51e
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
4 changed files with 110 additions and 33 deletions

View File

@ -48,6 +48,7 @@ public class QuickStartConfiguration extends AbstractConfiguration
public static final String ORIGIN_ATTRIBUTE = "org.eclipse.jetty.quickstart.origin"; public static final String ORIGIN_ATTRIBUTE = "org.eclipse.jetty.quickstart.origin";
public static final String QUICKSTART_WEB_XML = "org.eclipse.jetty.quickstart.xml"; public static final String QUICKSTART_WEB_XML = "org.eclipse.jetty.quickstart.xml";
public static final String MODE = "org.eclipse.jetty.quickstart.mode"; public static final String MODE = "org.eclipse.jetty.quickstart.mode";
private static final Mode DEFAULT_MODE = Mode.AUTO;
static static
{ {
@ -72,13 +73,10 @@ public class QuickStartConfiguration extends AbstractConfiguration
public enum Mode public enum Mode
{ {
GENERATE, // Generate quickstart-web.xml and then stop GENERATE, // Generate quickstart-web.xml and then stop
AUTO, // use or generate depending on the existance of quickstart-web.xml AUTO, // use quickstart depending on the existence of quickstart-web.xml
QUICKSTART // Use quickstart-web.xml QUICKSTART // Use quickstart-web.xml
} }
private Mode _mode = Mode.AUTO;
private boolean _quickStart;
public QuickStartConfiguration() public QuickStartConfiguration()
{ {
super(true); super(true);
@ -86,36 +84,46 @@ public class QuickStartConfiguration extends AbstractConfiguration
addDependents(WebXmlConfiguration.class); addDependents(WebXmlConfiguration.class);
} }
private Mode getModeForContext(WebAppContext context)
{
Mode mode = (Mode)context.getAttribute(MODE);
if (mode == null)
return DEFAULT_MODE;
return mode;
}
@Override @Override
public void preConfigure(WebAppContext context) throws Exception public void preConfigure(WebAppContext context) throws Exception
{ {
//check that webapp is suitable for quick start - it is not a packed war // check that webapp is suitable for quick start - it is not a packed war
String war = context.getWar(); String war = context.getWar();
if (war == null || war.length() <= 0 || !context.getBaseResource().isDirectory()) if (StringUtil.isBlank(war) || !context.getBaseResource().isDirectory())
throw new IllegalStateException("Bad Quickstart location"); throw new IllegalStateException("Invalid Quickstart location");
//look for quickstart-web.xml in WEB-INF of webapp // look for quickstart-web.xml in WEB-INF of webapp
Resource quickStartWebXml = getQuickStartWebXml(context); Resource quickStartWebXml = getQuickStartWebXml(context);
LOG.debug("quickStartWebXml={} exists={}", quickStartWebXml, quickStartWebXml.exists());
//Get the mode // Get the mode
Mode mode = (Mode)context.getAttribute(MODE); Mode mode = getModeForContext(context);
if (mode != null)
_mode = mode;
_quickStart = false; if (LOG.isDebugEnabled())
LOG.debug("mode={} quickStartWebXml={} exists={} for {}",
mode,
quickStartWebXml,
quickStartWebXml.exists(),
context);
switch (_mode) switch (mode)
{ {
case GENERATE: case GENERATE:
{ {
if (quickStartWebXml.exists()) if (quickStartWebXml.exists())
LOG.info("Regenerating {}", quickStartWebXml); LOG.info("Regenerating {} for {}", quickStartWebXml, context);
else else
LOG.info("Generating {}", quickStartWebXml); LOG.info("Generating {} for {}", quickStartWebXml, context);
super.preConfigure(context); super.preConfigure(context);
//generate the quickstart file then abort // generate the quickstart file then abort
QuickStartGeneratorConfiguration generator = new QuickStartGeneratorConfiguration(true); QuickStartGeneratorConfiguration generator = new QuickStartGeneratorConfiguration(true);
configure(generator, context); configure(generator, context);
context.addConfiguration(generator); context.addConfiguration(generator);
@ -130,20 +138,25 @@ public class QuickStartConfiguration extends AbstractConfiguration
else else
{ {
if (LOG.isDebugEnabled()) if (LOG.isDebugEnabled())
LOG.debug("No quickstart xml file, starting webapp {} normally", context); LOG.debug("No quickstart-web.xml found, starting webapp {} normally", context);
super.preConfigure(context); super.preConfigure(context);
} }
break; break;
} }
case QUICKSTART: case QUICKSTART:
{
if (quickStartWebXml.exists()) if (quickStartWebXml.exists())
{
quickStart(context); quickStart(context);
}
else else
throw new IllegalStateException("No " + quickStartWebXml); {
throw new IllegalStateException("No WEB-INF/quickstart-web.xml file for " + context);
}
break; break;
}
default: default:
throw new IllegalStateException(_mode.toString()); throw new IllegalStateException("Unhandled QuickStart.Mode: " + mode);
} }
} }
@ -160,7 +173,9 @@ public class QuickStartConfiguration extends AbstractConfiguration
@Override @Override
public void configure(WebAppContext context) throws Exception public void configure(WebAppContext context) throws Exception
{ {
if (!_quickStart) Resource quickStartWebXml = getQuickStartWebXml(context);
// Don't run configure() if quickstart-web.xml does not exist
if (quickStartWebXml == null || !quickStartWebXml.exists())
{ {
super.configure(context); super.configure(context);
} }
@ -203,7 +218,6 @@ public class QuickStartConfiguration extends AbstractConfiguration
throws Exception throws Exception
{ {
LOG.info("Quickstarting {}", context); LOG.info("Quickstarting {}", context);
_quickStart = true;
context.setConfigurations(context.getConfigurations().stream() context.setConfigurations(context.getConfigurations().stream()
.filter(c -> !__replacedConfigurations.contains(c.replaces()) && !__replacedConfigurations.contains(c.getClass())) .filter(c -> !__replacedConfigurations.contains(c.replaces()) && !__replacedConfigurations.contains(c.getClass()))
.collect(Collectors.toList()).toArray(new Configuration[]{})); .collect(Collectors.toList()).toArray(new Configuration[]{}));
@ -225,15 +239,34 @@ public class QuickStartConfiguration extends AbstractConfiguration
if (attr instanceof Resource) if (attr instanceof Resource)
return (Resource)attr; return (Resource)attr;
Resource qstart;
Resource webInf = context.getWebInf(); Resource webInf = context.getWebInf();
if (webInf == null || !webInf.exists()) if (webInf == null || !webInf.exists())
{ {
switch (getModeForContext(context))
{
case GENERATE:
// This mode allow for generation of quickstart-web.xml
// So we should be safe to attempt to create the missing WEB-INF directory.
File tmp = new File(context.getBaseResource().getFile(), "WEB-INF"); File tmp = new File(context.getBaseResource().getFile(), "WEB-INF");
tmp.mkdirs(); if (!tmp.mkdirs())
{
throw new IllegalStateException("Unable to create directory " + tmp);
}
webInf = context.getWebInf(); webInf = context.getWebInf();
break;
case AUTO:
case QUICKSTART:
// These modes do not allow for generation of quickstart-web.xml, so do not
// create the missing WEB-INF directory.
// Establish a Resource suitable for callers of this method (does not exist)
qstart = context.getBaseResource().addPath("WEB-INF/quickstart-web.xml");
context.setAttribute(QUICKSTART_WEB_XML, qstart);
return qstart;
}
} }
Resource qstart;
if (attr == null || StringUtil.isBlank(attr.toString())) if (attr == null || StringUtil.isBlank(attr.toString()))
{ {
qstart = webInf.addPath("quickstart-web.xml"); qstart = webInf.addPath("quickstart-web.xml");

View File

@ -13,7 +13,6 @@
package org.eclipse.jetty.quickstart; package org.eclipse.jetty.quickstart;
import java.io.InputStream;
import java.nio.file.Files; import java.nio.file.Files;
import java.nio.file.Path; import java.nio.file.Path;
@ -52,10 +51,12 @@ public class PreconfigureSpecWar
LOG.info("Preconfigured in {}ms", NanoTime.millisSince(__start)); LOG.info("Preconfigured in {}ms", NanoTime.millisSince(__start));
/*
Path quickStartXml = target.resolve("WEB-INF/quickstart-web.xml"); Path quickStartXml = target.resolve("WEB-INF/quickstart-web.xml");
try (InputStream in = Files.newInputStream(quickStartXml)) try (InputStream in = Files.newInputStream(quickStartXml))
{ {
IO.copy(in, System.out); IO.copy(in, System.out);
} }
*/
} }
} }

View File

@ -24,6 +24,8 @@ import org.eclipse.jetty.plus.webapp.EnvConfiguration;
import org.eclipse.jetty.plus.webapp.PlusConfiguration; import org.eclipse.jetty.plus.webapp.PlusConfiguration;
import org.eclipse.jetty.server.NetworkConnector; import org.eclipse.jetty.server.NetworkConnector;
import org.eclipse.jetty.server.Server; import org.eclipse.jetty.server.Server;
import org.eclipse.jetty.toolchain.test.FS;
import org.eclipse.jetty.toolchain.test.MavenPaths;
import org.eclipse.jetty.toolchain.test.MavenTestingUtils; import org.eclipse.jetty.toolchain.test.MavenTestingUtils;
import org.eclipse.jetty.util.IO; import org.eclipse.jetty.util.IO;
import org.eclipse.jetty.util.resource.PathResource; import org.eclipse.jetty.util.resource.PathResource;
@ -34,6 +36,8 @@ import org.eclipse.jetty.xml.XmlConfiguration;
import org.eclipse.jetty.xml.XmlParser.Node; import org.eclipse.jetty.xml.XmlParser.Node;
import org.hamcrest.Matchers; import org.hamcrest.Matchers;
import org.junit.jupiter.api.Test; import org.junit.jupiter.api.Test;
import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.ValueSource;
import static org.hamcrest.MatcherAssert.assertThat; import static org.hamcrest.MatcherAssert.assertThat;
import static org.junit.jupiter.api.Assertions.assertEquals; import static org.junit.jupiter.api.Assertions.assertEquals;
@ -41,6 +45,44 @@ import static org.junit.jupiter.api.Assertions.assertTrue;
public class QuickStartTest public class QuickStartTest
{ {
/**
* Test of an exploded webapp directory, no WEB-INF/quickstart-web.xml,
* with QuickStartConfiguration enabled.
*/
@ParameterizedTest
@ValueSource(booleans = {true, false})
public void testExplodedWebAppDirNoWebXml(boolean defaultMode) throws Exception
{
Path jettyHome = MavenPaths.targetDir();
Path webappDir = MavenPaths.targetTestDir("no-web-xml");
Path src = MavenPaths.projectBase().resolve("src/test/webapps/no-web-xml");
FS.ensureEmpty(webappDir);
org.eclipse.jetty.toolchain.test.IO.copyDir(src, webappDir);
System.setProperty("jetty.home", jettyHome.toString());
Server server = new Server(0);
WebAppContext webapp = new WebAppContext();
webapp.addConfiguration(new QuickStartConfiguration(),
new EnvConfiguration(),
new PlusConfiguration(),
new AnnotationConfiguration());
// Default mode should allow this style of exploded webapp dir.
if (!defaultMode)
webapp.setAttribute(QuickStartConfiguration.MODE, QuickStartConfiguration.Mode.AUTO);
webapp.setWarResource(new PathResource(webappDir));
webapp.setContextPath("/");
server.setHandler(webapp);
server.start();
URL url = new URL("http://127.0.0.1:" + server.getBean(NetworkConnector.class).getLocalPort() + "/index.html");
HttpURLConnection connection = (HttpURLConnection)url.openConnection();
assertEquals(200, connection.getResponseCode());
assertThat(IO.toString((InputStream)connection.getContent()), Matchers.containsString("<p>Contents of no-web-xml</p>"));
server.stop();
}
@Test @Test
public void testStandardTestWar() throws Exception public void testStandardTestWar() throws Exception

View File

@ -0,0 +1 @@
<p>Contents of no-web-xml</p>