001 /* 002 * (c) Copyright 2009 University of Bristol 003 * All rights reserved. 004 * [See end of file] 005 */ 006 package net.rootdev.javardfa.literal; 007 008 import java.util.Iterator; 009 import java.util.SortedMap; 010 import java.util.TreeMap; 011 import javax.xml.XMLConstants; 012 import javax.xml.namespace.NamespaceContext; 013 import javax.xml.namespace.QName; 014 import javax.xml.stream.XMLEventReader; 015 import javax.xml.stream.XMLEventWriter; 016 import javax.xml.stream.XMLStreamException; 017 import javax.xml.stream.XMLStreamWriter; 018 import javax.xml.stream.events.Attribute; 019 import javax.xml.stream.events.ProcessingInstruction; 020 import javax.xml.stream.events.StartElement; 021 import javax.xml.stream.events.XMLEvent; 022 023 /** 024 * All this class does currently is ensure the order of attributes, but 025 * (By using a stream writer) it is more controllable than the event writer. 026 * 027 * @author pldms 028 */ 029 public class CanonicalXMLEventWriter 030 implements XMLEventWriter { 031 032 private final XMLStreamWriter swriter; 033 private final Attribute contextLang; 034 private int level; 035 036 public CanonicalXMLEventWriter(XMLStreamWriter swriter, Attribute contextLang) { 037 this.swriter = swriter; 038 this.contextLang = contextLang; 039 this.level = 0; 040 } 041 042 public void flush() throws XMLStreamException { 043 swriter.flush(); 044 } 045 046 public void close() throws XMLStreamException { 047 swriter.close(); 048 } 049 050 public void add(XMLEvent event) throws XMLStreamException { 051 if (event.isEndElement()) { 052 level--; 053 swriter.writeEndElement(); 054 } else if (event.isCharacters()) { 055 swriter.writeCharacters(event.asCharacters().getData()); 056 } else if (event.isProcessingInstruction()) { 057 swriter.writeProcessingInstruction(((ProcessingInstruction) event).getData(), 058 ((ProcessingInstruction) event).getTarget()); 059 } else if (event.isStartElement()) { 060 level++; 061 StartElement se = event.asStartElement(); 062 if (se.getName().getNamespaceURI() == null || 063 se.getName().getNamespaceURI().length() == 0) 064 swriter.writeStartElement(se.getName().getLocalPart()); 065 else if (se.getName().getPrefix().length() == 0) { 066 swriter.setDefaultNamespace(se.getName().getNamespaceURI()); 067 swriter.writeStartElement( 068 se.getName().getNamespaceURI(), 069 se.getName().getLocalPart()); 070 } 071 else { 072 //swriter.setPrefix(se.getName().getPrefix(), se.getName().getNamespaceURI()); 073 swriter.writeStartElement(se.getName().getPrefix(), 074 se.getName().getLocalPart(), 075 se.getName().getNamespaceURI()); 076 } 077 writeAttributes(se); 078 swriter.writeCharacters(""); // Force close of start element 079 } else { 080 System.err.printf("Gah! Missed one <%s>, '%s'\n", event.getClass(), event); 081 } 082 } 083 084 public void add(XMLEventReader reader) throws XMLStreamException { 085 while (reader.hasNext()) { 086 this.add(reader.nextEvent()); 087 } 088 } 089 090 public String getPrefix(String uri) throws XMLStreamException { 091 return swriter.getPrefix(uri); 092 } 093 094 public void setPrefix(String prefix, String uri) throws XMLStreamException { 095 swriter.setPrefix(prefix, uri); 096 } 097 098 public void setDefaultNamespace(String uri) throws XMLStreamException { 099 swriter.setDefaultNamespace(uri); 100 } 101 102 public void setNamespaceContext(NamespaceContext context) throws XMLStreamException { 103 swriter.setNamespaceContext(context); 104 } 105 106 public NamespaceContext getNamespaceContext() { 107 return swriter.getNamespaceContext(); 108 } 109 110 private void writeAttributes(StartElement se) throws XMLStreamException { 111 SortedMap<String, Attribute> atts = new TreeMap<String, Attribute>(); 112 if (level == 2 && contextLang != null) atts.put("_xml:lang", contextLang); 113 for (Iterator i = se.getAttributes(); i.hasNext();) { 114 Attribute a = (Attribute) i.next(); 115 atts.put(getName(a), a); 116 } 117 for (Attribute a : atts.values()) { 118 if (a.getName().getNamespaceURI() == null || 119 a.getName().getNamespaceURI().length() == 0) 120 swriter.writeAttribute(a.getName().getLocalPart(), a.getValue()); 121 else if (a.getName().getPrefix().length() == 0) 122 swriter.writeAttribute(a.getName().getNamespaceURI(), 123 a.getName().getLocalPart(), a.getValue()); 124 else 125 swriter.writeAttribute( 126 a.getName().getPrefix(), 127 a.getName().getNamespaceURI(), 128 a.getName().getLocalPart(), 129 a.getValue()); 130 } 131 } 132 133 private String getName(Attribute a) { 134 QName name = a.getName(); 135 String toReturn = null; 136 // TODO I think something -- probably my code -- is wrong 137 // localName is sometimes xml:lang, so I got xml:xml:lang 138 if (name.getLocalPart().contains(":")) toReturn = name.getLocalPart(); 139 else toReturn = (name.getPrefix() == null) ? 140 name.getLocalPart() : 141 name.getPrefix() + ":" + name.getLocalPart(); 142 if (toReturn.startsWith("xml:")) { 143 return "_" + toReturn; 144 } else { 145 return toReturn; 146 } 147 } 148 } 149 150 /* 151 * (c) Copyright 2009 University of Bristol 152 * All rights reserved. 153 * 154 * Redistribution and use in source and binary forms, with or without 155 * modification, are permitted provided that the following conditions 156 * are met: 157 * 1. Redistributions of source code must retain the above copyright 158 * notice, this list of conditions and the following disclaimer. 159 * 2. Redistributions in binary form must reproduce the above copyright 160 * notice, this list of conditions and the following disclaimer in the 161 * documentation and/or other materials provided with the distribution. 162 * 3. The name of the author may not be used to endorse or promote products 163 * derived from this software without specific prior written permission. 164 * 165 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR 166 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES 167 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 168 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, 169 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT 170 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 171 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 172 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 173 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF 174 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 175 */