From 98b2dbc7dce9eb10161fed0960ff6b0eaac59df5 Mon Sep 17 00:00:00 2001 From: Juri Leino Date: Thu, 31 Oct 2024 11:10:50 +0100 Subject: [PATCH] [bugfix] allow module imports in one-off xqueries fixes #5525 fixes #5530 - add functx to autodeploy for xquery tests - add tests for one-off queries with module imports - of a registered module without location hint - of a module with location hint - change XQueryContext to allow imports again - change SourceFactory to work with contextPath set to "." --- .../java/org/exist/source/SourceFactory.java | 2 +- .../java/org/exist/xquery/XQueryContext.java | 66 +- .../org/exist/xquery/ModuleImportTest.java | 174 ++++ .../org/exist/xquery/conf.xml | 976 ++++++++++++++++++ 4 files changed, 1186 insertions(+), 32 deletions(-) create mode 100644 exist-core/src/test/java/org/exist/xquery/ModuleImportTest.java create mode 100644 exist-core/src/test/resources-filtered/org/exist/xquery/conf.xml 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/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 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +