org.openrdf.rio.turtle.TurtleWriter Maven / Gradle / Ivy
/*
* Licensed to Aduna under one or more contributor license agreements.
* See the NOTICE.txt file distributed with this work for additional
* information regarding copyright ownership.
*
* Aduna licenses this file to you under the terms of the Aduna BSD
* License (the "License"); you may not use this file except in compliance
* with the License. See the LICENSE.txt file distributed with this work
* for the full License.
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
* implied. See the License for the specific language governing permissions
* and limitations under the License.
*/
package org.openrdf.rio.turtle;
import java.io.IOException;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.io.Writer;
import java.nio.charset.Charset;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.StringTokenizer;
import info.aduna.io.IndentingWriter;
import org.openrdf.model.BNode;
import org.openrdf.model.Literal;
import org.openrdf.model.Resource;
import org.openrdf.model.Statement;
import org.openrdf.model.URI;
import org.openrdf.model.Value;
import org.openrdf.model.vocabulary.RDF;
import org.openrdf.rio.RDFFormat;
import org.openrdf.rio.RDFHandlerException;
import org.openrdf.rio.RDFWriter;
import org.openrdf.rio.helpers.RDFWriterBase;
/**
* An implementation of the RDFWriter interface that writes RDF documents in
* Turtle format. The Turtle format is defined in in this document.
*/
public class TurtleWriter extends RDFWriterBase implements RDFWriter {
/*-----------*
* Variables *
*-----------*/
protected IndentingWriter writer;
protected boolean writingStarted;
/**
* Flag indicating whether the last written statement has been closed.
*/
protected boolean statementClosed;
protected Resource lastWrittenSubject;
protected URI lastWrittenPredicate;
/*--------------*
* Constructors *
*--------------*/
/**
* Creates a new TurtleWriter that will write to the supplied OutputStream.
*
* @param out
* The OutputStream to write the Turtle document to.
*/
public TurtleWriter(OutputStream out) {
this(new OutputStreamWriter(out, Charset.forName("UTF-8")));
}
/**
* Creates a new TurtleWriter that will write to the supplied Writer.
*
* @param writer
* The Writer to write the Turtle document to.
*/
public TurtleWriter(Writer writer) {
this.writer = new IndentingWriter(writer);
namespaceTable = new LinkedHashMap();
writingStarted = false;
statementClosed = true;
lastWrittenSubject = null;
lastWrittenPredicate = null;
}
/*---------*
* Methods *
*---------*/
public RDFFormat getRDFFormat() {
return RDFFormat.TURTLE;
}
public void startRDF()
throws RDFHandlerException
{
if (writingStarted) {
throw new RuntimeException("Document writing has already started");
}
writingStarted = true;
try {
// Write namespace declarations
for (Map.Entry entry : namespaceTable.entrySet()) {
String name = entry.getKey();
String prefix = entry.getValue();
writeNamespace(prefix, name);
}
if (!namespaceTable.isEmpty()) {
writer.writeEOL();
}
}
catch (IOException e) {
throw new RDFHandlerException(e);
}
}
public void endRDF()
throws RDFHandlerException
{
if (!writingStarted) {
throw new RuntimeException("Document writing has not yet started");
}
try {
closePreviousStatement();
writer.flush();
}
catch (IOException e) {
throw new RDFHandlerException(e);
}
finally {
writingStarted = false;
}
}
public void handleNamespace(String prefix, String name)
throws RDFHandlerException
{
try {
if (!namespaceTable.containsKey(name)) {
// Namespace not yet mapped to a prefix, try to give it the
// specified prefix
boolean isLegalPrefix = prefix.length() == 0 || TurtleUtil.isLegalPrefix(prefix);
if (!isLegalPrefix || namespaceTable.containsValue(prefix)) {
// Specified prefix is not legal or the prefix is already in use,
// generate a legal unique prefix
if (prefix.length() == 0 || !isLegalPrefix) {
prefix = "ns";
}
int number = 1;
while (namespaceTable.containsValue(prefix + number)) {
number++;
}
prefix += number;
}
namespaceTable.put(name, prefix);
if (writingStarted) {
closePreviousStatement();
writeNamespace(prefix, name);
}
}
}
catch (IOException e) {
throw new RDFHandlerException(e);
}
}
public void handleStatement(Statement st)
throws RDFHandlerException
{
if (!writingStarted) {
throw new RuntimeException("Document writing has not yet been started");
}
Resource subj = st.getSubject();
URI pred = st.getPredicate();
Value obj = st.getObject();
try {
if (subj.equals(lastWrittenSubject)) {
if (pred.equals(lastWrittenPredicate)) {
// Identical subject and predicate
writer.write(" , ");
}
else {
// Identical subject, new predicate
writer.write(" ;");
writer.writeEOL();
// Write new predicate
writePredicate(pred);
writer.write(" ");
lastWrittenPredicate = pred;
}
}
else {
// New subject
closePreviousStatement();
// Write new subject:
writer.writeEOL();
writeResource(subj);
writer.write(" ");
lastWrittenSubject = subj;
// Write new predicate
writePredicate(pred);
writer.write(" ");
lastWrittenPredicate = pred;
statementClosed = false;
writer.increaseIndentation();
}
writeValue(obj);
// Don't close the line just yet. Maybe the next
// statement has the same subject and/or predicate.
}
catch (IOException e) {
throw new RDFHandlerException(e);
}
}
public void handleComment(String comment)
throws RDFHandlerException
{
try {
closePreviousStatement();
if (comment.indexOf('\r') != -1 || comment.indexOf('\n') != -1) {
// Comment is not allowed to contain newlines or line feeds.
// Split comment in individual lines and write comment lines
// for each of them.
StringTokenizer st = new StringTokenizer(comment, "\r\n");
while (st.hasMoreTokens()) {
writeCommentLine(st.nextToken());
}
}
else {
writeCommentLine(comment);
}
}
catch (IOException e) {
throw new RDFHandlerException(e);
}
}
protected void writeCommentLine(String line)
throws IOException
{
writer.write("# ");
writer.write(line);
writer.writeEOL();
}
protected void writeNamespace(String prefix, String name)
throws IOException
{
writer.write("@prefix ");
writer.write(prefix);
writer.write(": <");
writer.write(TurtleUtil.encodeURIString(name));
writer.write("> .");
writer.writeEOL();
}
protected void writePredicate(URI predicate)
throws IOException
{
if (predicate.equals(RDF.TYPE)) {
// Write short-cut for rdf:type
writer.write("a");
}
else {
writeURI(predicate);
}
}
protected void writeValue(Value val)
throws IOException
{
if (val instanceof Resource) {
writeResource((Resource)val);
}
else {
writeLiteral((Literal)val);
}
}
protected void writeResource(Resource res)
throws IOException
{
if (res instanceof URI) {
writeURI((URI)res);
}
else {
writeBNode((BNode)res);
}
}
protected void writeURI(URI uri)
throws IOException
{
String uriString = uri.toString();
// Try to find a prefix for the URI's namespace
String prefix = null;
int splitIdx = TurtleUtil.findURISplitIndex(uriString);
if (splitIdx > 0) {
String namespace = uriString.substring(0, splitIdx);
prefix = namespaceTable.get(namespace);
}
if (prefix != null) {
// Namespace is mapped to a prefix; write abbreviated URI
writer.write(prefix);
writer.write(":");
writer.write(uriString.substring(splitIdx));
}
else {
// Write full URI
writer.write("<");
writer.write(TurtleUtil.encodeURIString(uriString));
writer.write(">");
}
}
protected void writeBNode(BNode bNode)
throws IOException
{
writer.write("_:");
writer.write(bNode.getID());
}
protected void writeLiteral(Literal lit)
throws IOException
{
String label = lit.getLabel();
if (label.indexOf('\n') != -1 || label.indexOf('\r') != -1 || label.indexOf('\t') != -1) {
// Write label as long string
writer.write("\"\"\"");
writer.write(TurtleUtil.encodeLongString(label));
writer.write("\"\"\"");
}
else {
// Write label as normal string
writer.write("\"");
writer.write(TurtleUtil.encodeString(label));
writer.write("\"");
}
if (lit.getLanguage() != null) {
// Append the literal's language
writer.write("@");
writer.write(lit.getLanguage());
}
else if (lit.getDatatype() != null) {
// Append the literal's datatype (possibly written as an abbreviated
// URI)
writer.write("^^");
writeURI(lit.getDatatype());
}
}
protected void closePreviousStatement()
throws IOException
{
if (!statementClosed) {
// The previous statement still needs to be closed:
writer.write(" .");
writer.writeEOL();
writer.decreaseIndentation();
statementClosed = true;
lastWrittenSubject = null;
lastWrittenPredicate = null;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy