All Downloads are FREE. Search and download functionalities are using the official Maven repository.
Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
com.xmlcalabash.io.WritableDocument Maven / Gradle / Ivy
/*
* WritableDocument.java
*
* Copyright 2008 Mark Logic Corporation.
* Portions Copyright 2007 Sun Microsystems, Inc.
* All rights reserved.
*
* The contents of this file are subject to the terms of either the GNU
* General Public License Version 2 only ("GPL") or the Common
* Development and Distribution License("CDDL") (collectively, the
* "License"). You may not use this file except in compliance with the
* License. You can obtain a copy of the License at
* https://xproc.dev.java.net/public/CDDL+GPL.html or
* docs/CDDL+GPL.txt in the distribution. See the License for the
* specific language governing permissions and limitations under the
* License. When distributing the software, include this License Header
* Notice in each file and include the License file at docs/CDDL+GPL.txt.
*/
package com.xmlcalabash.io;
import java.io.File;
import java.net.URI;
import java.net.URISyntaxException;
import java.net.URLConnection;
import com.xmlcalabash.util.S9apiUtils;
import net.sf.saxon.s9api.DocumentBuilder;
import net.sf.saxon.s9api.Processor;
import net.sf.saxon.s9api.SaxonApiException;
import net.sf.saxon.s9api.Serializer;
import net.sf.saxon.s9api.XQueryCompiler;
import net.sf.saxon.s9api.XQueryEvaluator;
import net.sf.saxon.s9api.XQueryExecutable;
import net.sf.saxon.s9api.XdmNode;
import com.xmlcalabash.core.XProcRuntime;
import com.xmlcalabash.core.XProcException;
import com.xmlcalabash.model.Serialization;
import com.xmlcalabash.model.Step;
import java.io.IOException;
import java.io.OutputStream;
import java.net.URL;
/**
*
* @author ndw
*/
public class WritableDocument implements WritablePipe {
private Serializer serializer = null;
private int writeCount = 0;
private String uri = null;
private URI journal = null;
private XProcRuntime runtime = null;
private Serialization serial = null;
private boolean writeSeqOk = false;
private Step writer = null;
private OutputStream ostream = null;
/** Creates a new instance of ReadableDocument */
public WritableDocument(XProcRuntime xproc, String uri, Serialization serial) {
this.runtime = xproc;
this.uri = uri;
if (serial == null) {
this.serial = new Serialization(xproc, null);
this.serial.setIndent(xproc.getDebug()); // indent stdio by default when debugging
} else {
this.serial = serial;
}
}
public WritableDocument(XProcRuntime xproc, String uri, Serialization serial, OutputStream out) {
this.runtime = xproc;
this.uri = uri;
this.ostream = out;
if (serial == null) {
this.serial = new Serialization(xproc, null);
this.serial.setIndent(xproc.getDebug()); // indent stdio by default when debugging
} else {
this.serial = serial;
}
}
public void canWriteSequence(boolean sequence) {
writeSeqOk = sequence;
}
public boolean writeSequence() {
return writeSeqOk;
}
public void resetWriter() {
throw new UnsupportedOperationException("You can't resetWriter a WritableDocument");
}
public void close()
{
if (ostream != null) {
try {
ostream.close();
} catch (IOException ex) {
throw new RuntimeException(ex.getMessage(),ex);
}
}
}
public void setWriter(Step step) {
writer = step;
}
public void write(XdmNode doc) {
try {
serializer = new Serializer();
serializer.setOutputProperty(Serializer.Property.BYTE_ORDER_MARK, serial.getByteOrderMark() ? "yes" : "no");
// FIXME: support CDATA_SECTION_ELEMENTS
if (serial.getDoctypePublic() != null) {
serializer.setOutputProperty(Serializer.Property.DOCTYPE_PUBLIC, serial.getDoctypePublic());
}
if (serial.getDoctypeSystem() != null) {
serializer.setOutputProperty(Serializer.Property.DOCTYPE_SYSTEM, serial.getDoctypeSystem());
}
if (serial.getEncoding() != null) {
serializer.setOutputProperty(Serializer.Property.ENCODING, serial.getEncoding());
}
serializer.setOutputProperty(Serializer.Property.ESCAPE_URI_ATTRIBUTES, serial.getEscapeURIAttributes() ? "yes" : "no");
serializer.setOutputProperty(Serializer.Property.INCLUDE_CONTENT_TYPE, serial.getIncludeContentType() ? "yes" : "no");
serializer.setOutputProperty(Serializer.Property.INDENT, serial.getIndent() ? "yes" : "no");
if (serial.getMediaType() != null) {
serializer.setOutputProperty(Serializer.Property.MEDIA_TYPE, serial.getMediaType());
}
if (serial.getMethod() != null) {
serializer.setOutputProperty(Serializer.Property.METHOD, serial.getMethod().getLocalName());
}
if (serial.getNormalizationForm() != null) {
serializer.setOutputProperty(Serializer.Property.NORMALIZATION_FORM, serial.getNormalizationForm());
}
serializer.setOutputProperty(Serializer.Property.OMIT_XML_DECLARATION, serial.getOmitXMLDeclaration() ? "yes" : "no");
if (serial.getStandalone() != null) {
String standalone = serial.getStandalone();
if ("true".equals(standalone)) {
serializer.setOutputProperty(Serializer.Property.STANDALONE, "yes");
} else if ("false".equals(standalone)) {
serializer.setOutputProperty(Serializer.Property.STANDALONE, "no");
}
// What about omit?
}
serializer.setOutputProperty(Serializer.Property.UNDECLARE_PREFIXES, serial.getUndeclarePrefixes() ? "yes" : "no");
if (serial.getVersion() != null) {
serializer.setOutputProperty(Serializer.Property.VERSION, serial.getVersion());
}
if (ostream != null) {
serializer.setOutputStream(ostream);
} else if (uri == null) {
serializer.setOutputStream(System.out);
} else {
try {
// Attempt to handle both the case where we're writing to a URI scheme that
// supports writing (like FTP?) and the case where we're writing to a file
// (which apparently *isn't* a scheme we can write to, WTF?)
URI ouri = new URI(uri);
if ("file".equals(ouri.getScheme())) {
runtime.finest(null, null, "Attempt to write file: " + uri);
File file = new File(decodeUTF8(ouri.toURL().getFile()));
serializer.setOutputFile(file);
} else {
runtime.finest(null, null, "Attempt to write: " + uri);
URL url = new URL(uri);
final URLConnection conn = url.openConnection();
conn.setDoOutput(true);
ostream = conn.getOutputStream();
serializer.setOutputStream(ostream);
}
} catch (IOException ex) {
runtime.error(ex);
} catch (URISyntaxException use) {
runtime.error(use);
}
}
S9apiUtils.serialize(runtime, doc, serializer);
if ((((ostream == null) && (uri == null)) || (System.out.equals(ostream))) && runtime.getDebug()) {
System.out.println("\n----------------------------------------------------------------------------");
}
} catch (SaxonApiException sae) {
if (runtime.getDebug()) {
sae.printStackTrace();
}
throw new XProcException(sae);
}
if (writer != null) {
runtime.finest(null, writer.getNode(), writer.getName() + " wrote '" + (doc == null ? "null" : doc.getBaseURI()));
}
}
public int documentsWritten() {
return writeCount;
}
public int documentsRead() {
return 1;
}
/**
* Decode UTF-8/URL encoded strings.
*
* @param s the string to be decoded
* @return the decoded string
*/
private String decodeUTF8(String s) {
if (s == null) {
return null;
}
if(s.indexOf('%') == -1) {
//Optimization, nothing to uncorrect here
return s;
}
StringBuilder sbuf = new StringBuilder();
int l = s.length();
int ch = -1;
int b = 0, sumb = 0;
boolean applyUTF8dec = false;
for (int i = 0, more = -1; i < l; i++) {
/* Get next byte b from URL segment s */
char current = s.charAt(i);
ch = current;
switch (ch) {
case '%' :
if (i + 2 < s.length()) {
ch = s.charAt(++i);
int hb =
(Character.isDigit((char) ch) ? ch - '0' : 10 + Character.toLowerCase((char) ch) - 'a')
& 0xF;
ch = s.charAt(++i);
int lb =
(Character.isDigit((char) ch) ? ch - '0' : 10 + Character.toLowerCase((char) ch) - 'a')
& 0xF;
b = (hb << 4) | lb;
applyUTF8dec = true;
}
break;
default :
b = ch;
applyUTF8dec = false;
}
/* Decode byte b as UTF-8, sumb collects incomplete chars */
if (applyUTF8dec) {
if ((b & 0xc0) == 0x80) { // 10xxxxxx (continuation byte)
sumb = (sumb << 6) | (b & 0x3f); // Add 6 bits to sumb
if (--more == 0) {
sbuf.append((char) sumb); // Add char to sbuf
}
} else if ((b & 0x80) == 0x00) { // 0xxxxxxx (yields 7 bits)
sbuf.append((char) b); // Store in sbuf
} else if ((b & 0xe0) == 0xc0) { // 110xxxxx (yields 5 bits)
sumb = b & 0x1f;
more = 1; // Expect 1 more byte
} else if ((b & 0xf0) == 0xe0) { // 1110xxxx (yields 4 bits)
sumb = b & 0x0f;
more = 2; // Expect 2 more bytes
} else if ((b & 0xf8) == 0xf0) { // 11110xxx (yields 3 bits)
sumb = b & 0x07;
more = 3; // Expect 3 more bytes
} else if ((b & 0xfc) == 0xf8) { // 111110xx (yields 2 bits)
sumb = b & 0x03;
more = 4; // Expect 4 more bytes
} else /*if ((b & 0xfe) == 0xfc)*/ { // 1111110x (yields 1 bit)
sumb = b & 0x01;
more = 5; // Expect 5 more bytes
}
} else {
sbuf.append(current);
// Do not expect other continuation.
more = -1;
}
/* We don't test if the UTF-8 encoding is well-formed */
}
return sbuf.toString();
}
}