Skip to content

Commit

Permalink
Merge pull request #5586 from line-o/backport/5529
Browse files Browse the repository at this point in the history
[6.x.x] allow module imports in one-off xqueries
  • Loading branch information
dizzzz authored Jan 9, 2025
2 parents 71a82ca + 98b2dbc commit 3011d1f
Show file tree
Hide file tree
Showing 4 changed files with 1,186 additions and 32 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -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);
Expand Down
66 changes: 35 additions & 31 deletions exist-core/src/main/java/org/exist/xquery/XQueryContext.java
Original file line number Diff line number Diff line change
Expand Up @@ -574,46 +574,50 @@ public Optional<ExistRepository> getRepository() {
// the repo and its eXist handler
final Optional<ExistRepository> 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);
}
}

/**
Expand Down
174 changes: 174 additions & 0 deletions exist-core/src/test/java/org/exist/xquery/ModuleImportTest.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,174 @@
/*
* eXist-db Open Source Native XML Database
* Copyright (C) 2001 The eXist-db Authors
*
* [email protected]
* 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 <a href="mailto:[email protected]">Juri Leino</a>
*/
public class ModuleImportTest {
@ClassRule
public static final ExistEmbeddedServer server = new ExistEmbeddedServer(null, getConfigFile(), null, false, true);

protected static Either<XPathException, CompiledXQuery> 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<XPathException, Sequence> 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<XPathException, Sequence> 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<XPathException, Sequence> 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<XPathException, Sequence> 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<XPathException, Sequence> 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<XPathException, Sequence> 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));
}

}
Loading

0 comments on commit 3011d1f

Please sign in to comment.