/* Copyright (c) 2002,2003, Stefan Haustein, Oberhausen, Rhld., Germany * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or * sell copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS * IN THE SOFTWARE. */ //Contributors: Jonathan Cox, Bogdan Onoiu, Jerry Tian package org.kxml2.wap; import java.io.*; import java.util.*; import org.xmlpull.v1.*; // TODO: make some of the "direct" WBXML token writing methods public?? /** * A class for writing WBXML. * */ public class WbxmlSerializer implements XmlSerializer { Hashtable stringTable = new Hashtable(); OutputStream out; ByteArrayOutputStream buf = new ByteArrayOutputStream(); ByteArrayOutputStream stringTableBuf = new ByteArrayOutputStream(); String pending; int depth; String name; String namespace; Vector attributes = new Vector(); Hashtable attrStartTable = new Hashtable(); Hashtable attrValueTable = new Hashtable(); Hashtable tagTable = new Hashtable(); private int attrPage; private int tagPage; private String encoding; public XmlSerializer attribute(String namespace, String name, String value) { attributes.addElement(name); attributes.addElement(value); return this; } public void cdsect (String cdsect) throws IOException{ text (cdsect); } /* silently ignore comment */ public void comment (String comment) { } public void docdecl (String docdecl) { throw new RuntimeException ("Cannot write docdecl for WBXML"); } public void entityRef (String er) { throw new RuntimeException ("EntityReference not supported for WBXML"); } public int getDepth() { return depth; } public boolean getFeature (String name) { return false; } public String getNamespace() { throw new RuntimeException("NYI"); } public String getName() { throw new RuntimeException("NYI"); } public String getPrefix(String nsp, boolean create) { throw new RuntimeException ("NYI"); } public Object getProperty (String name) { return null; } public void ignorableWhitespace (String sp) { } public void endDocument() throws IOException { writeInt(out, stringTableBuf.size()); // write StringTable out.write(stringTableBuf.toByteArray()); // write buf out.write(buf.toByteArray()); // ready! out.flush(); } /** ATTENTION: flush cannot work since Wbxml documents require buffering. Thus, this call does nothing. */ public void flush() { } public void checkPending(boolean degenerated) throws IOException { if (pending == null) return; int len = attributes.size(); int[] idx = (int[]) tagTable.get(pending); // if no entry in known table, then add as literal if (idx == null) { buf.write( len == 0 ? (degenerated ? Wbxml.LITERAL : Wbxml.LITERAL_C) : (degenerated ? Wbxml.LITERAL_A : Wbxml.LITERAL_AC)); writeStrT(pending, false); } else { if(idx[0] != tagPage){ tagPage=idx[0]; buf.write(Wbxml.SWITCH_PAGE); buf.write(tagPage); } buf.write( len == 0 ? (degenerated ? idx[1] : idx[1] | 64) : (degenerated ? idx[1] | 128 : idx[1] | 192)); } for (int i = 0; i < len;) { idx = (int[]) attrStartTable.get(attributes.elementAt(i)); if (idx == null) { buf.write(Wbxml.LITERAL); writeStrT((String) attributes.elementAt(i), false); } else { if(idx[0] != attrPage){ attrPage = idx[0]; buf.write(0); buf.write(attrPage); } buf.write(idx[1]); } idx = (int[]) attrValueTable.get(attributes.elementAt(++i)); if (idx == null) { writeStr((String) attributes.elementAt(i)); } else { if(idx[0] != attrPage){ attrPage = idx[0]; buf.write(0); buf.write(attrPage); } buf.write(idx[1]); } ++i; } if (len > 0) buf.write(Wbxml.END); pending = null; attributes.removeAllElements(); } public void processingInstruction(String pi) { throw new RuntimeException ("PI NYI"); } public void setFeature(String name, boolean value) { throw new IllegalArgumentException ("unknown feature "+name); } public void setOutput (Writer writer) { throw new RuntimeException ("Wbxml requires an OutputStream!"); } public void setOutput (OutputStream out, String encoding) throws IOException { this.encoding = encoding == null ? "UTF-8" : encoding; this.out = out; buf = new ByteArrayOutputStream(); stringTableBuf = new ByteArrayOutputStream(); // ok, write header } public void setPrefix(String prefix, String nsp) { throw new RuntimeException("NYI"); } public void setProperty(String property, Object value) { throw new IllegalArgumentException ("unknown property "+property); } public void startDocument(String s, Boolean b) throws IOException{ out.write(0x03); // version 1.3 // http://www.openmobilealliance.org/tech/omna/omna-wbxml-public-docid.htm out.write(0x01); // unknown or missing public identifier // default encoding is UTF-8 if(s != null){ encoding = s; } if (encoding.toUpperCase().equals("UTF-8")){ out.write(106); }else if (encoding.toUpperCase().equals("ISO-8859-1")){ out.write(0x04); }else{ throw new UnsupportedEncodingException(s); } } public XmlSerializer startTag(String namespace, String name) throws IOException { if (namespace != null && !"".equals(namespace)) throw new RuntimeException ("NSP NYI"); //current = new State(current, prefixMap, name); checkPending(false); pending = name; depth++; return this; } public XmlSerializer text(char[] chars, int start, int len) throws IOException { checkPending(false); writeStr(new String(chars, start, len)); return this; } public XmlSerializer text(String text) throws IOException { checkPending(false); writeStr(text); return this; } /** Used in text() and attribute() to write text */ private void writeStr(String text) throws IOException{ int p0 = 0; int lastCut = 0; int len = text.length(); while(p0 < len){ while(p0 < len && text.charAt(p0) < 'A' ){ // skip interpunctation p0++; } int p1 = p0; while(p1 < len && text.charAt(p1) >= 'A'){ p1++; } if(p1 - p0 > 10) { if(p0 > lastCut && text.charAt(p0-1) == ' ' && stringTable.get(text.substring(p0, p1)) == null){ buf.write(Wbxml.STR_T); writeStrT(text.substring(lastCut, p1), false); } else { if(p0 > lastCut && text.charAt(p0-1) == ' '){ p0--; } if(p0 > lastCut){ buf.write(Wbxml.STR_T); writeStrT(text.substring(lastCut, p0), false); } buf.write(Wbxml.STR_T); writeStrT(text.substring(p0, p1), true); } lastCut = p1; } p0 = p1; } if(lastCut < len){ buf.write(Wbxml.STR_T); writeStrT(text.substring(lastCut, len), false); } } public XmlSerializer endTag(String namespace, String name) throws IOException { // current = current.prev; if (pending != null) checkPending(true); else buf.write(Wbxml.END); depth--; return this; } /** * @throws IOException */ public void writeWapExtension(int type, Object data) throws IOException { checkPending(false); buf.write(type); switch(type){ case Wbxml.EXT_0: case Wbxml.EXT_1: case Wbxml.EXT_2: break; case Wbxml.OPAQUE: byte[] bytes = (byte[]) data; writeInt(buf, bytes.length); buf.write(bytes); break; case Wbxml.EXT_I_0: case Wbxml.EXT_I_1: case Wbxml.EXT_I_2: writeStrI(buf, (String) data); break; case Wbxml.EXT_T_0: case Wbxml.EXT_T_1: case Wbxml.EXT_T_2: writeStrT((String) data, false); break; default: throw new IllegalArgumentException(); } } // ------------- internal methods -------------------------- static void writeInt(OutputStream out, int i) throws IOException { byte[] buf = new byte[5]; int idx = 0; do { buf[idx++] = (byte) (i & 0x7f); i = i >> 7; } while (i != 0); while (idx > 1) { out.write(buf[--idx] | 0x80); } out.write(buf[0]); } void writeStrI(OutputStream out, String s) throws IOException { byte[] data = s.getBytes(encoding); out.write(data); out.write(0); } private final void writeStrT(String s, boolean mayPrependSpace) throws IOException { Integer idx = (Integer) stringTable.get(s); if (idx != null) { writeInt(buf, idx.intValue()); } else{ int i = stringTableBuf.size(); if(s.charAt(0) >= '0' && mayPrependSpace){ s = ' ' + s; writeInt(buf, i+1); } else{ writeInt(buf, i); } stringTable.put(s, new Integer(i)); if(s.charAt(0) == ' '){ stringTable.put(s.substring(1), new Integer(i+1)); } int j = s.lastIndexOf(' '); if(j > 1){ stringTable.put(s.substring(j), new Integer(i+j)); stringTable.put(s.substring(j+1), new Integer(i+j+1)); } writeStrI(stringTableBuf, s); stringTableBuf.flush(); } } /** * Sets the tag table for a given page. * The first string in the array defines tag 5, the second tag 6 etc. */ public void setTagTable(int page, String[] tagTable) { // TODO: clear entries in tagTable? for (int i = 0; i < tagTable.length; i++) { if (tagTable[i] != null) { Object idx = new int[]{page, i+5}; this.tagTable.put(tagTable[i], idx); } } } /** * Sets the attribute start Table for a given page. * The first string in the array defines attribute * 5, the second attribute 6 etc. * Please use the * character '=' (without quote!) as delimiter * between the attribute name and the (start of the) value */ public void setAttrStartTable(int page, String[] attrStartTable) { for (int i = 0; i < attrStartTable.length; i++) { if (attrStartTable[i] != null) { Object idx = new int[] {page, i + 5}; this.attrStartTable.put(attrStartTable[i], idx); } } } /** * Sets the attribute value Table for a given page. * The first string in the array defines attribute value 0x85, * the second attribute value 0x86 etc. */ public void setAttrValueTable(int page, String[] attrValueTable) { // clear entries in this.table! for (int i = 0; i < attrValueTable.length; i++) { if (attrValueTable[i] != null) { Object idx = new int[]{page, i + 0x085}; this.attrValueTable.put(attrValueTable[i], idx); } } } }