All Downloads are FREE. Search and download functionalities are using the official Maven repository.

cucumber.deps.com.thoughtworks.xstream.io.binary.BinaryStreamReader Maven / Gradle / Ivy

There is a newer version: 1.0.5
Show newest version
/*
 * Copyright (C) 2006 Joe Walnes.
 * Copyright (C) 2006, 2007, 2011 XStream Committers.
 * All rights reserved.
 *
 * The software in this package is published under the terms of the BSD
 * style license a copy of which has been included with this distribution in
 * the LICENSE.txt file.
 * 
 * Created on 04. June 2006 by Joe Walnes
 */
package com.thoughtworks.xstream.io.binary;

import com.thoughtworks.xstream.converters.ErrorWriter;
import com.thoughtworks.xstream.io.ExtendedHierarchicalStreamReader;
import com.thoughtworks.xstream.io.HierarchicalStreamReader;
import com.thoughtworks.xstream.io.StreamException;

import java.io.DataInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;

/**
 * A HierarchicalStreamReader that reads from a binary stream created by
 * {@link BinaryStreamWriter}.
 * 

*

This produces * * @author Joe Walnes * @see BinaryStreamReader * @since 1.2 */ public class BinaryStreamReader implements ExtendedHierarchicalStreamReader { private final DataInputStream in; private final ReaderDepthState depthState = new ReaderDepthState(); private final IdRegistry idRegistry = new IdRegistry(); private Token pushback; private final Token.Formatter tokenFormatter = new Token.Formatter(); public BinaryStreamReader(InputStream inputStream) { in = new DataInputStream(inputStream); moveDown(); } public boolean hasMoreChildren() { return depthState.hasMoreChildren(); } public String getNodeName() { return depthState.getName(); } public String getValue() { return depthState.getValue(); } public String getAttribute(String name) { return depthState.getAttribute(name); } public String getAttribute(int index) { return depthState.getAttribute(index); } public int getAttributeCount() { return depthState.getAttributeCount(); } public String getAttributeName(int index) { return depthState.getAttributeName(index); } public Iterator getAttributeNames() { return depthState.getAttributeNames(); } public void moveDown() { depthState.push(); Token firstToken = readToken(); switch (firstToken.getType()) { case Token.TYPE_START_NODE: depthState.setName(idRegistry.get(firstToken.getId())); break; default: throw new StreamException("Expected StartNode"); } while (true) { Token nextToken = readToken(); switch (nextToken.getType()) { case Token.TYPE_ATTRIBUTE: depthState.addAttribute(idRegistry.get(nextToken.getId()), nextToken.getValue()); break; case Token.TYPE_VALUE: depthState.setValue(nextToken.getValue()); break; case Token.TYPE_END_NODE: depthState.setHasMoreChildren(false); pushBack(nextToken); return; case Token.TYPE_START_NODE: depthState.setHasMoreChildren(true); pushBack(nextToken); return; default: throw new StreamException("Unexpected token " + nextToken); } } } public void moveUp() { depthState.pop(); // We're done with this depth. Skip over all tokens until we get to the end. int depth = 0; slurp: while (true) { Token nextToken = readToken(); switch(nextToken.getType()) { case Token.TYPE_END_NODE: if (depth == 0) { break slurp; } else { depth--; } break; case Token.TYPE_START_NODE: depth++; break; default: // Ignore other tokens } } // Peek ahead to determine if there are any more kids at this level. Token nextToken = readToken(); switch(nextToken.getType()) { case Token.TYPE_END_NODE: depthState.setHasMoreChildren(false); break; case Token.TYPE_START_NODE: depthState.setHasMoreChildren(true); break; default: throw new StreamException("Unexpected token " + nextToken); } pushBack(nextToken); } private Token readToken() { if (pushback == null) { try { Token token = tokenFormatter.read(in); switch (token.getType()) { case Token.TYPE_MAP_ID_TO_VALUE: idRegistry.put(token.getId(), token.getValue()); return readToken(); // Next one please. default: return token; } } catch (IOException e) { throw new StreamException(e); } } else { Token result = pushback; pushback = null; return result; } } public void pushBack(Token token) { if (pushback == null) { pushback = token; } else { // If this happens, I've messed up :( -joe. throw new Error("Cannot push more than one token back"); } } public void close() { try { in.close(); } catch (IOException e) { throw new StreamException(e); } } public String peekNextChild() { if (depthState.hasMoreChildren()) { return idRegistry.get(pushback.getId()); } return null; } public HierarchicalStreamReader underlyingReader() { return this; } public void appendErrors(ErrorWriter errorWriter) { // TODO: When things go bad, it would be good to know where! } private static class IdRegistry { private Map map = new HashMap(); public void put(long id, String value) { map.put(new Long(id), value); } public String get(long id) { String result = (String) map.get(new Long(id)); if (result == null) { throw new StreamException("Unknown ID : " + id); } else { return result; } } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy