org.exist.xquery.modules.file.SerializeToFile Maven / Gradle / Ivy
/*
* 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.modules.file;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.exist.dom.QName;
import org.exist.storage.serializers.Serializer;
import org.exist.xquery.*;
import org.exist.xquery.util.SerializerUtils;
import org.exist.xquery.value.*;
import org.xml.sax.SAXException;
import javax.xml.transform.OutputKeys;
import java.io.*;
import java.nio.charset.Charset;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.StandardOpenOption;
import java.util.Properties;
public class SerializeToFile extends BasicFunction {
private final static Logger logger = LogManager.getLogger(SerializeToFile.class);
private final static String FN_SERIALIZE_LN = "serialize";
private final static String FN_SERIALIZE_BINARY_LN = "serialize-binary";
public final static FunctionSignature signatures[] = {
new FunctionSignature(
new QName( FN_SERIALIZE_LN, FileModule.NAMESPACE_URI, FileModule.PREFIX ),
"Writes the node set into a file on the file system. $parameters contains a " +
"sequence of zero or more serialization parameters specified as key=value pairs. The " +
"serialization options are the same as those recognized by \"declare option exist:serialize\". " +
"The function does NOT automatically inherit the serialization options of the XQuery it is " +
"called from. This method is only available to the DBA role.",
new SequenceType[] {
new FunctionParameterSequenceType( "node-set", Type.NODE,
Cardinality.ZERO_OR_MORE, "The contents to write to the file system." ),
new FunctionParameterSequenceType( "path", Type.ITEM,
Cardinality.EXACTLY_ONE, "The full path or URI to the file" ),
new FunctionParameterSequenceType( "parameters", Type.ITEM,
Cardinality.ZERO_OR_MORE, "The serialization parameters: either a sequence of key=value pairs or an output:serialization-parameters " +
"element as defined by the standard fn:serialize function." )
},
new FunctionReturnSequenceType( Type.BOOLEAN,
Cardinality.ZERO_OR_ONE, "true on success - false if the specified file can not be "
+ "created or is not writable. The empty sequence is returned if the argument sequence is empty." )
),
new FunctionSignature(
new QName( FN_SERIALIZE_LN, FileModule.NAMESPACE_URI, FileModule.PREFIX ),
"Writes the node set into a file on the file system, optionally appending to it. " +
"$parameters contains a sequence of zero or more serialization parameters specified as " +
"key=value pairs. The serialization options are the same as those recognized by " +
"\"declare option exist:serialize\". " +
"The function does NOT automatically inherit the serialization options of the XQuery it is " +
"called from. This method is only available to the DBA role.",
new SequenceType[] {
new FunctionParameterSequenceType( "node-set", Type.NODE,
Cardinality.ZERO_OR_MORE, "The contents to write to the file system." ),
new FunctionParameterSequenceType( "path", Type.ITEM,
Cardinality.EXACTLY_ONE, "The full path or URI to the file" ),
new FunctionParameterSequenceType( "parameters", Type.ITEM,
Cardinality.ZERO_OR_MORE, "The serialization parameters: either a sequence of key=value pairs or an output:serialization-parameters " +
"element as defined by the standard fn:serialize function." ),
new FunctionParameterSequenceType( "append", Type.BOOLEAN,
Cardinality.EXACTLY_ONE, "Should content be appended?")
},
new FunctionReturnSequenceType( Type.BOOLEAN,
Cardinality.ZERO_OR_ONE, "true on success - false if the specified file can "
+ "not be created or is not writable. The empty sequence is returned if the argument sequence is empty." )
),
new FunctionSignature(
new QName(FN_SERIALIZE_BINARY_LN, FileModule.NAMESPACE_URI, FileModule.PREFIX),
"Writes binary data into a file on the file system. This method is only available to the DBA role.",
new SequenceType[]{
new FunctionParameterSequenceType("binarydata", Type.BASE64_BINARY,
Cardinality.EXACTLY_ONE, "The contents to write to the file system."),
new FunctionParameterSequenceType("path", Type.ITEM,
Cardinality.EXACTLY_ONE, "The full path or URI to the file")
},
new FunctionReturnSequenceType(Type.BOOLEAN, Cardinality.EXACTLY_ONE, "true on success - false if the specified file can not be created or is not writable")
),
new FunctionSignature(
new QName(FN_SERIALIZE_BINARY_LN, FileModule.NAMESPACE_URI, FileModule.PREFIX),
"Writes binary data into a file on the file system, optionally appending the content. This method is only available to the DBA role.",
new SequenceType[]{
new FunctionParameterSequenceType("binarydata", Type.BASE64_BINARY,
Cardinality.EXACTLY_ONE, "The contents to write to the file system."),
new FunctionParameterSequenceType("path", Type.ITEM,
Cardinality.EXACTLY_ONE, "The full path or URI to the file"),
new FunctionParameterSequenceType("append", Type.BOOLEAN,
Cardinality.EXACTLY_ONE, "Should content be appended?")
},
new FunctionReturnSequenceType(Type.BOOLEAN, Cardinality.EXACTLY_ONE,
"true on success - false if the specified file can not be created or is not writable")
)
};
public SerializeToFile(final XQueryContext context, final FunctionSignature signature) {
super(context, signature);
}
@Override
public Sequence eval(final Sequence[] args, final Sequence contextSequence) throws XPathException {
if(args[0].isEmpty()) {
return Sequence.EMPTY_SEQUENCE;
}
if(!context.getSubject().hasDbaRole()) {
final XPathException xPathException = new XPathException(this, "Permission denied, calling user '" + context.getSubject().getName() + "' must be a DBA to call this function.");
logger.error("Invalid user", xPathException);
throw xPathException;
}
//check the file output path
final String inputPath = args[1].getStringValue();
final Path file = FileModuleHelper.getFile(inputPath, this);
if(Files.isDirectory(file)) {
logger.debug("Cannot serialize file. Output file is a directory: {}", file.toAbsolutePath().toString());
return BooleanValue.FALSE;
}
if(Files.exists(file) && !Files.isWritable(file)) {
logger.debug("Cannot serialize file. Cannot write to file {}", file.toAbsolutePath().toString());
return BooleanValue.FALSE;
}
if(isCalledAs(FN_SERIALIZE_LN)) {
//parse serialization options from third argument to function
final Properties outputProperties = parseXMLSerializationOptions( args[2] );
final boolean doAppend = (args.length > 3) && "true".equals(args[3].itemAt(0).getStringValue());
//do the serialization
serializeXML(args[0].iterate(), outputProperties, file, doAppend);
} else if(isCalledAs(FN_SERIALIZE_BINARY_LN)) {
final boolean doAppend = (args.length > 2) && "true".equals(args[2].itemAt(0).getStringValue());
serializeBinary((BinaryValue)args[0].itemAt(0), file, doAppend);
} else {
throw new XPathException(this, "Unknown function name");
}
return BooleanValue.TRUE;
}
private Properties parseXMLSerializationOptions(final Sequence sSerializeParams) throws XPathException {
//parse serialization options
final Properties outputProperties = new Properties();
// defaults
outputProperties.setProperty( OutputKeys.INDENT, "yes" );
outputProperties.setProperty( OutputKeys.OMIT_XML_DECLARATION, "yes" );
outputProperties.setProperty( OutputKeys.ENCODING, "UTF-8" );
if (sSerializeParams.hasOne() && Type.subTypeOf(sSerializeParams.getItemType(), Type.NODE)) {
SerializerUtils.getSerializationOptions(this, (NodeValue) sSerializeParams.itemAt(0), outputProperties);
} else {
SequenceIterator siSerializeParams = sSerializeParams.iterate();
while(siSerializeParams.hasNext()) {
final String serializeParam = siSerializeParams.nextItem().getStringValue();
final String opt[] = Option.parseKeyValuePair(serializeParam);
if(opt != null && opt.length == 2) {
outputProperties.setProperty( opt[0], opt[1] );
}
}
}
return outputProperties;
}
private void serializeXML(final SequenceIterator siNode, final Properties outputProperties, final Path file, final boolean doAppend) throws XPathException {
final Serializer serializer = context.getBroker().borrowSerializer();
StandardOpenOption ops[] = doAppend ? new StandardOpenOption[]{StandardOpenOption.WRITE, StandardOpenOption.CREATE, StandardOpenOption.APPEND}
: new StandardOpenOption[]{StandardOpenOption.WRITE, StandardOpenOption.CREATE, StandardOpenOption.TRUNCATE_EXISTING};
try (final OutputStream os = new BufferedOutputStream(Files.newOutputStream(file, ops));
final Writer writer = new OutputStreamWriter(os, Charset.forName(outputProperties.getProperty(OutputKeys.ENCODING)))) {
serializer.setProperties(outputProperties);
while (siNode.hasNext()) {
final NodeValue nv = (NodeValue) siNode.nextItem();
serializer.serialize(nv, writer);
}
} catch(UnsupportedOperationException | IllegalArgumentException e){
throw new XPathException(this, "Error wile writing to file: " + e.getMessage(), e);
} catch(final IOException | SAXException e) {
throw new XPathException(this, "Cannot serialize file. A problem occurred while serializing the node set: " + e.getMessage(), e);
} finally {
context.getBroker().returnSerializer(serializer);
}
}
private void serializeBinary(final BinaryValue binary, final Path file, final boolean doAppend) throws XPathException {
StandardOpenOption ops[] = doAppend ? new StandardOpenOption[]{StandardOpenOption.WRITE, StandardOpenOption.CREATE, StandardOpenOption.APPEND}
: new StandardOpenOption[]{StandardOpenOption.WRITE, StandardOpenOption.CREATE, StandardOpenOption.TRUNCATE_EXISTING};
try(final OutputStream os = new BufferedOutputStream(Files.newOutputStream(file, ops))) {
binary.streamBinaryTo(os);
} catch(final IOException ioe) {
throw new XPathException(this, "Cannot serialize file. A problem occurred while serializing the binary data: " + ioe.getMessage(), ioe);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy