All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.saxonica.xqj.SaxonXQDataSource Maven / Gradle / Ivy

////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// Copyright (c) 2015 Saxonica Limited.
// This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0.
// If a copy of the MPL was not distributed with this file, You can obtain one at http://mozilla.org/MPL/2.0/.
// This Source Code Form is "Incompatible With Secondary Licenses", as defined by the Mozilla Public License, v. 2.0.
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

package com.saxonica.xqj;

import net.sf.saxon.Configuration;
import net.sf.saxon.lib.ExtensionFunctionDefinition;
import net.sf.saxon.lib.FeatureKeys;
import net.sf.saxon.lib.Validation;
import net.sf.saxon.value.BooleanValue;
import net.sf.saxon.value.Whitespace;

import javax.xml.xquery.XQConnection;
import javax.xml.xquery.XQDataSource;
import javax.xml.xquery.XQException;
import java.io.PrintWriter;
import java.sql.Connection;
import java.util.Enumeration;
import java.util.Properties;

/**
 * Saxon implementation of the XQJ XQDataSource interface. The first action of a client application
 * is to instantiate a SaxonXQDataSource. This is done directly: there is no factory class as with JAXP.
 * An application that does not want compile-time references to the Saxon XQJ implementation can instantiate
 * this class dynamically using the reflection API (class.newInstance()).
 * 

* For full Javadoc descriptions of the public methods, see the XQJ specification. */ public class SaxonXQDataSource implements XQDataSource { private Configuration config; private PrintWriter logger; /** * Create a SaxonXQDataSource using a default configuration. * The Configuration will be an EE, PE, or HE configuration depending on the * what is found on the classpath */ public SaxonXQDataSource() { config = Configuration.newConfiguration(); config.setProcessor(this); } /** * Create a Saxon XQDataSource with a specific configuration * * @param config The Saxon configuration to be used */ public SaxonXQDataSource(Configuration config) { this.config = config; } /** * Get the Saxon Configuration in use. Changes made to this Configuration will affect this * data source and XQJ connections created from it (either before or afterwards). Equally, * changes made to this SaxonXQDataSource are reflected in the Configuration object (which means * they also impact any other SaxonXQDataSource objects that share the same Configuration). * * @return the configuration in use. */ public Configuration getConfiguration() { return config; } /*@NotNull*/ public XQConnection getConnection() throws XQException { return new SaxonXQConnection(this); } /** * Get a connection based on an underlying JDBC connection * * @param con the JDBC connection * @return a connection based on an underlying JDBC connection * @throws XQException The Saxon implementation of this method always throws * an XQException, indicating that Saxon does not support connection to a JDBC data source. */ /*@NotNull*/ public XQConnection getConnection(Connection con) throws XQException { throw new XQException("Saxon cannot connect to a SQL data source"); } /** * Get a connection, by supplying a username and password. The Saxon implementation of this is equivalent * to the default constructor: the username and password are ignored. * * @param username the user name * @param password the password * @return a connection * @throws XQException */ /*@NotNull*/ public XQConnection getConnection(String username, String password) throws XQException { return getConnection(); } public int getLoginTimeout() { return 0; } public PrintWriter getLogWriter() { return logger; } /** * Get a configuration property setting. The properties that are supported, and their meanings, are the * same as the properties that can be obtained using "get" methods; for example * getProperty("dtdValidation") returns the same result as getDtdValidation(). *

*

Further Saxon configuration properties are available using the URIs defined as constants in * {@link net.sf.saxon.lib.FeatureKeys}

* * @param name the name of the configuration property * @return the value of the configuration property. Note that in the case of on/off properties this * will be returned as the string true/false * @throws XQException */ /*@Nullable*/ public String getProperty(String name) throws XQException { checkNotNull(name, "name"); if ("allowExternalFunctions".equals(name)) { return getAllowExternalFunctions(); } else if ("dtdValidation".equals(name)) { return getDtdValidation(); } else if ("expandAttributeDefaults".equals(name)) { return getExpandAttributeDefaults(); } else if ("expandXInclude".equals(name)) { return getExpandXInclude(); } else if ("retainLineNumbers".equals(name)) { return getRetainLineNumbers(); } else if ("schemaValidationMode".equals(name)) { return getSchemaValidationMode(); } else if ("stripWhitespace".equals(name)) { return getStripWhitespace(); } else if ("useXsiSchemaLocation".equals(name)) { return getUseXsiSchemaLocation(); } else if ("xmlVersion".equals(name)) { return getXmlVersion(); } else if ("xsdVersion".equals(name)) { return getXsdVersion(); } else { try { return config.getConfigurationProperty(name).toString(); } catch (IllegalArgumentException e) { throw new XQException("Property " + name + " is not recognized"); } catch (NullPointerException err) { throw new XQException("Null property name or value"); } } } /*@NotNull*/ private static String[] supportedPropertyNames = { "allowExternalFunctions", "dtdValidation", "expandAttributeDefaults", "expandXInclude", "retainLineNumbers", "schemaValidationMode", "stripWhitespace", "useXsiSchemaLocation", "xmlVersion", "xsdVersion" }; /*@NotNull*/ public String[] getSupportedPropertyNames() { return supportedPropertyNames; } public void setLoginTimeout(int seconds) throws XQException { // no-op } public void setLogWriter(PrintWriter out) throws XQException { logger = out; } public void setProperties(Properties props) throws XQException { checkNotNull(props, "props"); Enumeration iter = props.keys(); while (iter.hasMoreElements()) { String name = (String) iter.nextElement(); String value = props.getProperty(name); setProperty(name, value); } } /** * Set a configuration property. The properties that are supported, and their meanings, are the * same as the properties that can be obtained using "set" methods; for example * setProperty("dtdValidation", "true") has the same effect as * setDtdValidation("true"). *

*

Further Saxon configuration properties are available using the URIs defined as constants in * {@link net.sf.saxon.lib.FeatureKeys}

* * @param name the name of the configuration property * @param value the value of the configuration property * @throws XQException */ public void setProperty(String name, String value) throws XQException { SaxonXQDataSource.checkNotNull(name, "name"); SaxonXQDataSource.checkNotNull(value, "value"); if ("allowExternalFunctions".equals(name)) { setAllowExternalFunctions(value); } else if ("dtdValidation".equals(name)) { setDtdValidation(value); } else if ("expandAttributeDefaults".equals(name)) { setExpandAttributeDefaults(value); } else if ("expandXInclude".equals(name)) { setExpandXInclude(value); } else if ("retainLineNumbers".equals(name)) { setRetainLineNumbers(value); } else if ("schemaValidationMode".equals(name)) { setSchemaValidationMode(value); } else if ("stripWhitespace".equals(name)) { setStripWhitespace(value); } else if ("useXsiSchemaLocation".equals(name)) { setUseXsiSchemaLocation(value); } else if ("xmlVersion".equals(name)) { setXmlVersion(value); } else if ("xsdVersion".equals(name)) { setXsdVersion(value); } else { try { config.setConfigurationProperty(name, value); } catch (IllegalArgumentException err) { throw new XQException("Unrecognized property or invalid value for " + name + ": " + err.getMessage()); } catch (NullPointerException err) { throw new XQException("Null property name or value"); } } } static void checkNotNull(/*@Nullable*/ Object arg, String name) throws XQException { if (arg == null) { throw new XQException("Argument " + name + " is null"); } } /** * Say whether queries are allowed to call external functions. * * @param value set to "true" if external function calls are allowed (default) or "false" otherwise */ public void setAllowExternalFunctions(String value) { if ("true".equals(value)) { config.setBooleanProperty(FeatureKeys.ALLOW_EXTERNAL_FUNCTIONS, true); } else if ("false".equals(value)) { config.setBooleanProperty(FeatureKeys.ALLOW_EXTERNAL_FUNCTIONS, false); } else { throw new IllegalArgumentException("allowExternalFunctions"); } } /** * Ask whether queries are allowed to call external functions. * * @return "true" if external function calls are allowed, "false" otherwise */ /*@NotNull*/ public String getAllowExternalFunctions() { return (config.getBooleanProperty(FeatureKeys.ALLOW_EXTERNAL_FUNCTIONS) ? "true" : "false"); } /** * Say whether source documents are to be parsed with DTD validation enabled * * @param value "true" if DTD validation is to be enabled, otherwise "false". Default is "false". */ public void setDtdValidation(String value) { if ("true".equals(value)) { config.setValidation(true); } else if ("false".equals(value)) { config.setValidation(false); } else { throw new IllegalArgumentException("dtdValidation"); } } /** * Ask whether source documents are to be parsed with DTD validation enabled * * @return "true" if DTD validation is to be enabled, otherwise "false". Default is "false". */ /*@NotNull*/ public String getDtdValidation() { return (config.isValidation() ? "true" : "false"); } /** * Say whether whether fixed and default values defined * in a schema or DTD will be expanded. By default, or if the value is "true" (and for conformance with the * specification) validation against a DTD or schema will cause default values defined in the schema * or DTD to be inserted into the document. Setting this feature to "false" suppresses this behaviour. In * the case of DTD-defined defaults this only works if the XML parser reports whether each attribute was * specified in the source or generated by expanding a default value. Not all XML parsers report this * information. * * @param value "true" if default values are to be expanded, otherwise "false". Default is "true". */ public void setExpandAttributeDefaults(String value) { if ("true".equals(value)) { config.setExpandAttributeDefaults(true); } else if ("false".equals(value)) { config.setExpandAttributeDefaults(false); } else { throw new IllegalArgumentException("expandAttributeDefaults"); } } /** * Ask whether fixed or default values defined in a schema or DTD will be expanded * * @return "true" if such values will be expanded, otherwise "false" */ /*@NotNull*/ public String getExpandAttributeDefaults() { return (config.isExpandAttributeDefaults() ? "true" : "false"); } /** * Say whether XInclude processing is to be applied to source documents * * @param value "true" if XInclude directives are to expanded, otherwise "false". Default is "false". */ public void setExpandXInclude(String value) { if ("true".equals(value)) { config.setXIncludeAware(true); } else if ("false".equals(value)) { config.setXIncludeAware(false); } else { throw new IllegalArgumentException("expandXInclude"); } } /** * Ask whether XInclude processing is to be applied to source documents * * @return "true" if XInclude directives are to expanded, otherwise "false". Default is "false". */ /*@NotNull*/ public String getExpandXInclude() { return (config.isXIncludeAware() ? "true" : "false"); } /** * Say whether source documents should have line and column information retained. This * information is available via extension functions saxon:line-number() and * saxon:column-number() * * @param value "true" if line and column information is to be retained, otherwise "false". Default is "false". */ public void setRetainLineNumbers(String value) { if ("true".equals(value)) { config.setLineNumbering(true); } else if ("false".equals(value)) { config.setLineNumbering(false); } else { throw new IllegalArgumentException("retainLineNumbers"); } } /** * Ask whether line and column information will be retained for source documents * * @return "true" if line and column information is retained, otherwise "false" */ /*@NotNull*/ public String getRetainLineNumbers() { return (config.isLineNumbering() ? "true" : "false"); } /** * Say whether source documents should be validated against a schema * * @param value set to "strict" if source documents are to be subjected to strict validation, * "lax" if source documents are to be subjected to lax validation, "skip" if source documents * are not to be subjected to schema validation */ public void setSchemaValidationMode(String value) { if ("strict".equals(value)) { config.setSchemaValidationMode(Validation.STRICT); } else if ("lax".equals(value)) { config.setSchemaValidationMode(Validation.LAX); } else if ("skip".equals(value)) { config.setSchemaValidationMode(Validation.SKIP); } else { throw new IllegalArgumentException("schemaValidationMode"); } } /** * Ask whether source documents will be validated against a schema * * @return "strict" if source documents are to be subjected to strict validation, * "lax" if source documents are to be subjected to lax validation, "skip" if source documents * are not to be subjected to schema validation */ public String getSchemaValidationMode() { return Validation.toString(config.getSchemaValidationMode()); } /** * Say whether whitespace should be stripped when loading source documents * * @param value "all" if all whitespace text nodes are to be stripped, "ignorable" if * only whitespace text nodes in elements defined in a schema or DTD as having element-only * content are to be stripped, "none" if no whitespace text nodes are to be stripped */ public void setStripWhitespace(String value) { if ("all".equals(value)) { config.setStripsWhiteSpace(Whitespace.ALL); } else if ("ignorable".equals(value)) { config.setStripsWhiteSpace(Whitespace.IGNORABLE); } else if ("none".equals(value)) { config.setStripsWhiteSpace(Whitespace.NONE); } else { throw new IllegalArgumentException("stripWhitespace"); } } /** * Ask whether whitespace will be stripped when loading source documents * * @return "all" if all whitespace text nodes are to be stripped, "ignorable" if * only whitespace text nodes in elements defined in a schema or DTD as having element-only * content are to be stripped, "none" if no whitespace text nodes are to be stripped */ /*@NotNull*/ public String getStripWhitespace() { switch (config.getStripsWhiteSpace()) { case Whitespace.ALL: return "all"; case Whitespace.IGNORABLE: return "ignorable"; default: return "none"; } } /** * Say whether the schema processor is to take account of xsi:schemaLocation and * xsi:noNamespaceSchemaLocation attributes encountered in an instance document being validated * * @param value set to "true" if these attributes are to be recognized (default) or "false" otherwise */ public void setUseXsiSchemaLocation(String value) { if ("true".equals(value)) { config.setConfigurationProperty(FeatureKeys.USE_XSI_SCHEMA_LOCATION, BooleanValue.TRUE); } else if ("false".equals(value)) { config.setConfigurationProperty(FeatureKeys.USE_XSI_SCHEMA_LOCATION, BooleanValue.FALSE); } else { throw new IllegalArgumentException("useXsiSchemaLocation"); } } /** * Ask whether the schema processor is to take account of xsi:schemaLocation and * xsi:noNamespaceSchemaLocation attributes encountered in an instance document being validated * * @return "true" if these attributes are to be recognized (default) or "false" otherwise */ /*@NotNull*/ public String getUseXsiSchemaLocation() { Boolean b = (Boolean) config.getConfigurationProperty(FeatureKeys.USE_XSI_SCHEMA_LOCATION); return (b.booleanValue() ? "true" : "false"); } /** * Say whether XML 1.0 or XML 1.1 rules for XML names are to be followed * * @param value "1.0" (default) if XML 1.0 rules are to be used, "1.1" if XML 1.1 * rules apply */ public void setXmlVersion(String value) { if ("1.0".equals(value)) { config.setXMLVersion(Configuration.XML10); } else if ("1.1".equals(value)) { config.setXMLVersion(Configuration.XML11); } else { throw new IllegalArgumentException("xmlVersion"); } } /** * Ask whether XML 1.0 or XML 1.1 rules for XML names are to be followed * * @return "1.0" if XML 1.0 rules are to be used, "1.1" if XML 1.1 * rules apply */ /*@NotNull*/ public String getXmlVersion() { return (config.getXMLVersion() == Configuration.XML10 ? "1.0" : "1.1"); } /** * Say whether XML Schema 1.0 syntax must be used or whether XML Schema 1.1 features are allowed * * @param value "1.0" (default) if XML Schema 1.0 rules are to be followed, "1.1" if XML Schema 1.1 * features may be used */ public void setXsdVersion(String value) { if ("1.0".equals(value)) { config.setConfigurationProperty(FeatureKeys.XSD_VERSION, "1.0"); } else if ("1.1".equals(value)) { config.setConfigurationProperty(FeatureKeys.XSD_VERSION, "1.1"); } else { throw new IllegalArgumentException("xsdVersion"); } } /** * Ask whether XML Schema 1.0 syntax must be used or whether XML Schema 1.1 features are allowed * * @return "1.0" (default) if XML Schema 1.0 rules are to be followed, "1.1" if XML Schema 1.1 * features may be used */ /*@Nullable*/ public String getXsdVersion() { return (String) config.getConfigurationProperty(FeatureKeys.XSD_VERSION); } /** * Register an extension function that is to be made available within any query. This method * allows deep integration of an extension function implemented as an instance of * {@link net.sf.saxon.lib.ExtensionFunctionDefinition}, using an arbitrary name and namespace. * This supplements the ability to call arbitrary Java methods using a namespace and local name * that are related to the Java class and method name. *

This method is a Saxon extension to the XQJ interface

* * @param function the class that implements the extension function. This must be a class that extends * {@link net.sf.saxon.lib.ExtensionFunctionCall}, and it must have a public zero-argument * constructor * @throws IllegalArgumentException if the class cannot be instantiated or does not extend * {@link net.sf.saxon.lib.ExtensionFunctionCall} * @since 9.2 */ public void registerExtensionFunction(ExtensionFunctionDefinition function) { try { config.registerExtensionFunction(function); } catch (Exception err) { throw new IllegalArgumentException(err); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy