diff --git a/exist-core/src/main/java/org/exist/source/SourceFactory.java b/exist-core/src/main/java/org/exist/source/SourceFactory.java
index ffe7fa39813..2c7dd48964a 100644
--- a/exist-core/src/main/java/org/exist/source/SourceFactory.java
+++ b/exist-core/src/main/java/org/exist/source/SourceFactory.java
@@ -111,7 +111,7 @@ public class SourceFactory {
&& ((location.startsWith("/db") && !Files.exists(Paths.get(firstPathSegment(location))))
|| (contextPath != null && contextPath.startsWith("/db") && !Files.exists(Paths.get(firstPathSegment(contextPath)))))) {
final XmldbURI pathUri;
- if (contextPath == null) {
+ if (contextPath == null || ".".equals(contextPath)) {
pathUri = XmldbURI.create(location);
} else {
pathUri = XmldbURI.create(contextPath).append(location);
diff --git a/exist-core/src/main/java/org/exist/test/XQueryCompilationTest.java b/exist-core/src/main/java/org/exist/test/XQueryCompilationTest.java
index 637670a2123..90ea364c64d 100644
--- a/exist-core/src/main/java/org/exist/test/XQueryCompilationTest.java
+++ b/exist-core/src/main/java/org/exist/test/XQueryCompilationTest.java
@@ -41,9 +41,9 @@
* @author Juri Leino
*/
public abstract class XQueryCompilationTest {
+
@ClassRule
public static final ExistEmbeddedServer server = new ExistEmbeddedServer(true, true);
-
protected static Either compileQuery(final String string) throws EXistException, PermissionDeniedException {
final BrokerPool pool = server.getBrokerPool();
final XQuery xqueryService = pool.getXQueryService();
diff --git a/exist-core/src/main/java/org/exist/xquery/XQueryContext.java b/exist-core/src/main/java/org/exist/xquery/XQueryContext.java
index d3999e3b62e..5fb53fbc9b0 100644
--- a/exist-core/src/main/java/org/exist/xquery/XQueryContext.java
+++ b/exist-core/src/main/java/org/exist/xquery/XQueryContext.java
@@ -574,46 +574,50 @@ public Optional getRepository() {
// the repo and its eXist handler
final Optional repo = getRepository();
+ if (!repo.isPresent()) {
+ return null;
+ }
// try an internal module
- if (repo.isPresent()) {
- final Module jMod = repo.get().resolveJavaModule(namespace, this);
- if (jMod != null) {
- return jMod;
- }
+ final Module jMod = repo.get().resolveJavaModule(namespace, this);
+ if (jMod != null) {
+ return jMod;
}
// try an eXist-specific module
- if (repo.isPresent()) {
- final Path resolved = repo.get().resolveXQueryModule(namespace);
-
- // use the resolved file or return null
- if (resolved != null) {
-
- String location = "";
-
- try {
-
- // see if the src exists in the database and if so, use that instead
- Source src = repo.get().resolveStoredXQueryModuleFromDb(getBroker(), resolved);
- if (src != null) {
- // NOTE(AR) set the location of the module to import relative to this module's load path - so that transient imports of the imported module will resolve correctly!
- location = Paths.get(XmldbURI.create(moduleLoadPath).getCollectionPath()).relativize(Paths.get(((DBSource)src).getDocumentPath().getCollectionPath())).toString();
- } else {
- // else, fallback to the one from the filesystem
- src = new FileSource(resolved, false);
- }
+ final Path resolved = repo.get().resolveXQueryModule(namespace);
- // build a module object from the source
- final ExternalModule module = compileOrBorrowModule(prefix, namespace, location, src);
- return module;
+ if (resolved == null) {
+ return null;
+ }
- } catch (final PermissionDeniedException e) {
- throw new XPathException(e.getMessage(), e);
+ // use the resolved file
+ try {
+ // see if the src exists in the database and if so, use that instead
+ Source src = repo.get().resolveStoredXQueryModuleFromDb(getBroker(), resolved);
+ String location = "";
+ if (src == null) {
+ // fallback to load the source from the filesystem
+ src = new FileSource(resolved, false);
+ } else {
+ final String sourceCollection = ((DBSource)src).getDocumentPath().getCollectionPath();
+ if (".".equals(moduleLoadPath)) {
+ // module is a string passed to the xquery context, has therefore no location of its own
+ location = sourceCollection;
+ } else {
+ // NOTE(AR) set the location of the module to import relative to this module's load path
+ // - so that transient imports of the imported module will resolve correctly!
+ final Path collectionPath = Paths.get(XmldbURI.create(moduleLoadPath).getCollectionPath());
+ final Path sourcePath = Paths.get(sourceCollection);
+ location = collectionPath.relativize(sourcePath).toString();
}
}
- }
- return null;
+ // build a module object from the source
+ return compileOrBorrowModule(prefix, namespace, location, src);
+
+ } catch (final PermissionDeniedException | IllegalArgumentException e) {
+ throw new XPathException(e.getMessage(), e);
+ }
}
/**
diff --git a/exist-core/src/test/java/org/exist/xquery/ModuleImportTest.java b/exist-core/src/test/java/org/exist/xquery/ModuleImportTest.java
new file mode 100644
index 00000000000..07dea7ddc8b
--- /dev/null
+++ b/exist-core/src/test/java/org/exist/xquery/ModuleImportTest.java
@@ -0,0 +1,174 @@
+/*
+ * eXist-db Open Source Native XML Database
+ * Copyright (C) 2001 The eXist-db Authors
+ *
+ * info@exist-db.org
+ * http://www.exist-db.org
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
+ */
+package org.exist.xquery;
+
+import com.evolvedbinary.j8fu.Either;
+
+import org.exist.EXistException;
+import org.exist.security.PermissionDeniedException;
+import org.exist.storage.BrokerPool;
+import org.exist.storage.DBBroker;
+import org.exist.test.ExistEmbeddedServer;
+import org.exist.xquery.value.Sequence;
+import org.exist.xquery.value.StringValue;
+
+import org.junit.ClassRule;
+import org.junit.Test;
+
+import java.net.URISyntaxException;
+import java.nio.file.Path;
+import java.nio.file.Paths;
+
+import static com.evolvedbinary.j8fu.Either.Left;
+import static com.evolvedbinary.j8fu.Either.Right;
+import static com.ibm.icu.impl.Assert.fail;
+import static org.exist.test.XQueryAssertions.assertThatXQResult;
+import static org.exist.test.XQueryAssertions.assertXQStaticError;
+import static org.hamcrest.Matchers.equalTo;
+
+/**
+ * Ensure library module imports work in one-off queries
+ * needs functx to be installed => conf.xml => triggers => autodeploy
+ *
+ * @author Juri Leino
+ */
+public class ModuleImportTest {
+ @ClassRule
+ public static final ExistEmbeddedServer server = new ExistEmbeddedServer(null, getConfigFile(), null, false, true);
+
+ protected static Either compileQuery(final String string) throws EXistException, PermissionDeniedException {
+ final BrokerPool pool = server.getBrokerPool();
+ final XQuery xqueryService = pool.getXQueryService();
+ try (final DBBroker broker = pool.getBroker()) {
+ try {
+ return Right(xqueryService.compile(new XQueryContext(broker.getDatabase()), string));
+ } catch (final XPathException e) {
+ return Left(e);
+ }
+ }
+ }
+
+ protected static Either executeQuery(final String string) throws EXistException, PermissionDeniedException {
+ final BrokerPool pool = server.getBrokerPool();
+ final XQuery xqueryService = pool.getXQueryService();
+ try (final DBBroker broker = pool.getBroker()) {
+ try {
+ return Right(xqueryService.execute(broker, string, null));
+ } catch (final XPathException e) {
+ return Left(e);
+ }
+ }
+ }
+
+ private static Path getConfigFile() {
+ final ClassLoader loader = ModuleImportTest.class.getClassLoader();
+ final char separator = System.getProperty("file.separator").charAt(0);
+ final String packagePath = ModuleImportTest.class.getPackage().getName().replace('.', separator);
+
+ try {
+ return Paths.get(loader.getResource(packagePath + separator + "conf.xml").toURI());
+ } catch (final URISyntaxException e) {
+ fail(e);
+ return null;
+ }
+ }
+
+ @Test
+ public void importLibraryWithoutLocation() throws EXistException, PermissionDeniedException {
+ final Sequence expected = new StringValue("xs:integer");
+
+ final String query = "import module namespace functx='http://www.functx.com';" +
+ "functx:atomic-type(4)";
+ final Either actual = executeQuery(query);
+
+ assertThatXQResult(actual, equalTo(expected));
+ }
+ @Test
+ public void importLibraryFromDbLocation() throws EXistException, PermissionDeniedException {
+ final Sequence expected = new StringValue("xs:integer");
+
+ final String query = "import module namespace functx='http://www.functx.com'" +
+ " at '/db/system/repo/functx-1.0.1/functx/functx.xq';" +
+ "functx:atomic-type(4)";
+ final Either actual = executeQuery(query);
+
+ assertThatXQResult(actual, equalTo(expected));
+ }
+
+ @Test
+ public void importLibraryFromXMLDBLocation() throws EXistException, PermissionDeniedException {
+ final Sequence expected = new StringValue("xs:integer");
+
+ final String query = "import module namespace functx='http://www.functx.com'" +
+ " at 'xmldb:/db/system/repo/functx-1.0.1/functx/functx.xq';" +
+ "functx:atomic-type(4)";
+ final Either actual = executeQuery(query);
+
+ assertThatXQResult(actual, equalTo(expected));
+ }
+
+ @Test
+ public void importLibraryFromXMLDBLocationDoubleSlash() throws EXistException, PermissionDeniedException {
+ final Sequence expected = new StringValue("xs:integer");
+
+ final String query = "import module namespace functx='http://www.functx.com'" +
+ " at 'xmldb:///db/system/repo/functx-1.0.1/functx/functx.xq';" +
+ "functx:atomic-type(4)";
+ final Either actual = executeQuery(query);
+
+ assertThatXQResult(actual, equalTo(expected));
+ }
+
+ @Test
+ public void importLibraryFromExistXMLDBLocation() throws EXistException, PermissionDeniedException {
+ final Sequence expected = new StringValue("xs:integer");
+
+ final String query = "import module namespace functx='http://www.functx.com'" +
+ " at 'xmldb:exist:///db/system/repo/functx-1.0.1/functx/functx.xq';" +
+ "functx:atomic-type(4)";
+ final Either actual = executeQuery(query);
+
+ assertThatXQResult(actual, equalTo(expected));
+ }
+
+ @Test
+ public void importLibraryFromUnknownLocation() throws EXistException, PermissionDeniedException {
+
+ final String query = "import module namespace functx='http://www.functx.com'" +
+ " at 'unknown:///db/system/repo/functx-1.0.1/functx/functx.xq';" +
+ "functx:atomic-type(4)";
+ final String expectedMessage = "error found while loading module functx: Source for module 'http://www.functx.com' not found module location hint URI 'unknown:///db/system/repo/functx-1.0.1/functx/functx.xq'.";
+
+ assertXQStaticError(ErrorCodes.XQST0059, -1,-1, expectedMessage, compileQuery(query));
+ }
+
+ @Test
+ public void importLibraryFromRelativeLocation() throws EXistException, PermissionDeniedException {
+ final String query = "import module namespace functx='http://www.functx.com'" +
+ " at './functx.xq';" +
+ "functx:atomic-type(4)";
+ final String expectedMessage = "error found while loading module functx: Source for module 'http://www.functx.com' not found module location hint URI './functx.xq'.";
+
+ assertXQStaticError(ErrorCodes.XQST0059, -1,-1, expectedMessage, compileQuery(query));
+ }
+
+}
diff --git a/exist-core/src/test/resources-filtered/org/exist/xquery/conf.xml b/exist-core/src/test/resources-filtered/org/exist/xquery/conf.xml
new file mode 100644
index 00000000000..b9bc14f5b53
--- /dev/null
+++ b/exist-core/src/test/resources-filtered/org/exist/xquery/conf.xml
@@ -0,0 +1,976 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+