diff --git a/.classpath b/.classpath
index c777dbbde3a8bbd90fb2b852389bae263e970b1e..2ed9bb6ee03d54c507ff19daa62b1c761d8cb246 100644
--- a/.classpath
+++ b/.classpath
@@ -15,6 +15,5 @@
 	<classpathentry kind="lib" path="lib/test/hamcrest-core-1.1.jar"/>
 	<classpathentry kind="lib" path="lib/test/hamcrest-library-1.1.jar"/>
 	<classpathentry kind="lib" path="lib/test/jmock-2.5.1.jar"/>
-	<classpathentry kind="lib" path="lib/protobuf.jar"/>
 	<classpathentry kind="output" path="bin"/>
 </classpath>
diff --git a/api/net/sf/briar/api/protocol/Transport.java b/api/net/sf/briar/api/protocol/Transport.java
deleted file mode 100644
index 2a27836cd28b018025bebc9d399d8ff56c2ddbf0..0000000000000000000000000000000000000000
--- a/api/net/sf/briar/api/protocol/Transport.java
+++ /dev/null
@@ -1,1108 +0,0 @@
-// Generated by the protocol buffer compiler.  DO NOT EDIT!
-// source: net/sf/briar/api/protocol/Transport.proto
-
-package net.sf.briar.api.protocol;
-
-public final class Transport {
-  private Transport() {}
-  public static void registerAllExtensions(
-      com.google.protobuf.ExtensionRegistry registry) {
-  }
-  public interface TransportDetailsOrBuilder
-      extends com.google.protobuf.MessageOrBuilder {
-    
-    // repeated .protocol.TransportDetails.TransportDetail details = 1;
-    java.util.List<net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail> 
-        getDetailsList();
-    net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail getDetails(int index);
-    int getDetailsCount();
-    java.util.List<? extends net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetailOrBuilder> 
-        getDetailsOrBuilderList();
-    net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetailOrBuilder getDetailsOrBuilder(
-        int index);
-  }
-  public static final class TransportDetails extends
-      com.google.protobuf.GeneratedMessage
-      implements TransportDetailsOrBuilder {
-    // Use TransportDetails.newBuilder() to construct.
-    private TransportDetails(Builder builder) {
-      super(builder);
-    }
-    private TransportDetails(boolean noInit) {}
-    
-    private static final TransportDetails defaultInstance;
-    public static TransportDetails getDefaultInstance() {
-      return defaultInstance;
-    }
-    
-    public TransportDetails getDefaultInstanceForType() {
-      return defaultInstance;
-    }
-    
-    public static final com.google.protobuf.Descriptors.Descriptor
-        getDescriptor() {
-      return net.sf.briar.api.protocol.Transport.internal_static_protocol_TransportDetails_descriptor;
-    }
-    
-    protected com.google.protobuf.GeneratedMessage.FieldAccessorTable
-        internalGetFieldAccessorTable() {
-      return net.sf.briar.api.protocol.Transport.internal_static_protocol_TransportDetails_fieldAccessorTable;
-    }
-    
-    public interface TransportDetailOrBuilder
-        extends com.google.protobuf.MessageOrBuilder {
-      
-      // required string key = 1;
-      boolean hasKey();
-      String getKey();
-      
-      // optional string value = 2;
-      boolean hasValue();
-      String getValue();
-    }
-    public static final class TransportDetail extends
-        com.google.protobuf.GeneratedMessage
-        implements TransportDetailOrBuilder {
-      // Use TransportDetail.newBuilder() to construct.
-      private TransportDetail(Builder builder) {
-        super(builder);
-      }
-      private TransportDetail(boolean noInit) {}
-      
-      private static final TransportDetail defaultInstance;
-      public static TransportDetail getDefaultInstance() {
-        return defaultInstance;
-      }
-      
-      public TransportDetail getDefaultInstanceForType() {
-        return defaultInstance;
-      }
-      
-      public static final com.google.protobuf.Descriptors.Descriptor
-          getDescriptor() {
-        return net.sf.briar.api.protocol.Transport.internal_static_protocol_TransportDetails_TransportDetail_descriptor;
-      }
-      
-      protected com.google.protobuf.GeneratedMessage.FieldAccessorTable
-          internalGetFieldAccessorTable() {
-        return net.sf.briar.api.protocol.Transport.internal_static_protocol_TransportDetails_TransportDetail_fieldAccessorTable;
-      }
-      
-      private int bitField0_;
-      // required string key = 1;
-      public static final int KEY_FIELD_NUMBER = 1;
-      private java.lang.Object key_;
-      public boolean hasKey() {
-        return ((bitField0_ & 0x00000001) == 0x00000001);
-      }
-      public String getKey() {
-        java.lang.Object ref = key_;
-        if (ref instanceof String) {
-          return (String) ref;
-        } else {
-          com.google.protobuf.ByteString bs = 
-              (com.google.protobuf.ByteString) ref;
-          String s = bs.toStringUtf8();
-          if (com.google.protobuf.Internal.isValidUtf8(bs)) {
-            key_ = s;
-          }
-          return s;
-        }
-      }
-      private com.google.protobuf.ByteString getKeyBytes() {
-        java.lang.Object ref = key_;
-        if (ref instanceof String) {
-          com.google.protobuf.ByteString b = 
-              com.google.protobuf.ByteString.copyFromUtf8((String) ref);
-          key_ = b;
-          return b;
-        } else {
-          return (com.google.protobuf.ByteString) ref;
-        }
-      }
-      
-      // optional string value = 2;
-      public static final int VALUE_FIELD_NUMBER = 2;
-      private java.lang.Object value_;
-      public boolean hasValue() {
-        return ((bitField0_ & 0x00000002) == 0x00000002);
-      }
-      public String getValue() {
-        java.lang.Object ref = value_;
-        if (ref instanceof String) {
-          return (String) ref;
-        } else {
-          com.google.protobuf.ByteString bs = 
-              (com.google.protobuf.ByteString) ref;
-          String s = bs.toStringUtf8();
-          if (com.google.protobuf.Internal.isValidUtf8(bs)) {
-            value_ = s;
-          }
-          return s;
-        }
-      }
-      private com.google.protobuf.ByteString getValueBytes() {
-        java.lang.Object ref = value_;
-        if (ref instanceof String) {
-          com.google.protobuf.ByteString b = 
-              com.google.protobuf.ByteString.copyFromUtf8((String) ref);
-          value_ = b;
-          return b;
-        } else {
-          return (com.google.protobuf.ByteString) ref;
-        }
-      }
-      
-      private void initFields() {
-        key_ = "";
-        value_ = "";
-      }
-      private byte memoizedIsInitialized = -1;
-      public final boolean isInitialized() {
-        byte isInitialized = memoizedIsInitialized;
-        if (isInitialized != -1) return isInitialized == 1;
-        
-        if (!hasKey()) {
-          memoizedIsInitialized = 0;
-          return false;
-        }
-        memoizedIsInitialized = 1;
-        return true;
-      }
-      
-      public void writeTo(com.google.protobuf.CodedOutputStream output)
-                          throws java.io.IOException {
-        getSerializedSize();
-        if (((bitField0_ & 0x00000001) == 0x00000001)) {
-          output.writeBytes(1, getKeyBytes());
-        }
-        if (((bitField0_ & 0x00000002) == 0x00000002)) {
-          output.writeBytes(2, getValueBytes());
-        }
-        getUnknownFields().writeTo(output);
-      }
-      
-      private int memoizedSerializedSize = -1;
-      public int getSerializedSize() {
-        int size = memoizedSerializedSize;
-        if (size != -1) return size;
-      
-        size = 0;
-        if (((bitField0_ & 0x00000001) == 0x00000001)) {
-          size += com.google.protobuf.CodedOutputStream
-            .computeBytesSize(1, getKeyBytes());
-        }
-        if (((bitField0_ & 0x00000002) == 0x00000002)) {
-          size += com.google.protobuf.CodedOutputStream
-            .computeBytesSize(2, getValueBytes());
-        }
-        size += getUnknownFields().getSerializedSize();
-        memoizedSerializedSize = size;
-        return size;
-      }
-      
-      private static final long serialVersionUID = 0L;
-      @java.lang.Override
-      protected java.lang.Object writeReplace()
-          throws java.io.ObjectStreamException {
-        return super.writeReplace();
-      }
-      
-      public static net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail parseFrom(
-          com.google.protobuf.ByteString data)
-          throws com.google.protobuf.InvalidProtocolBufferException {
-        return newBuilder().mergeFrom(data).buildParsed();
-      }
-      public static net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail parseFrom(
-          com.google.protobuf.ByteString data,
-          com.google.protobuf.ExtensionRegistryLite extensionRegistry)
-          throws com.google.protobuf.InvalidProtocolBufferException {
-        return newBuilder().mergeFrom(data, extensionRegistry)
-                 .buildParsed();
-      }
-      public static net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail parseFrom(byte[] data)
-          throws com.google.protobuf.InvalidProtocolBufferException {
-        return newBuilder().mergeFrom(data).buildParsed();
-      }
-      public static net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail parseFrom(
-          byte[] data,
-          com.google.protobuf.ExtensionRegistryLite extensionRegistry)
-          throws com.google.protobuf.InvalidProtocolBufferException {
-        return newBuilder().mergeFrom(data, extensionRegistry)
-                 .buildParsed();
-      }
-      public static net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail parseFrom(java.io.InputStream input)
-          throws java.io.IOException {
-        return newBuilder().mergeFrom(input).buildParsed();
-      }
-      public static net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail parseFrom(
-          java.io.InputStream input,
-          com.google.protobuf.ExtensionRegistryLite extensionRegistry)
-          throws java.io.IOException {
-        return newBuilder().mergeFrom(input, extensionRegistry)
-                 .buildParsed();
-      }
-      public static net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail parseDelimitedFrom(java.io.InputStream input)
-          throws java.io.IOException {
-        Builder builder = newBuilder();
-        if (builder.mergeDelimitedFrom(input)) {
-          return builder.buildParsed();
-        } else {
-          return null;
-        }
-      }
-      public static net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail parseDelimitedFrom(
-          java.io.InputStream input,
-          com.google.protobuf.ExtensionRegistryLite extensionRegistry)
-          throws java.io.IOException {
-        Builder builder = newBuilder();
-        if (builder.mergeDelimitedFrom(input, extensionRegistry)) {
-          return builder.buildParsed();
-        } else {
-          return null;
-        }
-      }
-      public static net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail parseFrom(
-          com.google.protobuf.CodedInputStream input)
-          throws java.io.IOException {
-        return newBuilder().mergeFrom(input).buildParsed();
-      }
-      public static net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail parseFrom(
-          com.google.protobuf.CodedInputStream input,
-          com.google.protobuf.ExtensionRegistryLite extensionRegistry)
-          throws java.io.IOException {
-        return newBuilder().mergeFrom(input, extensionRegistry)
-                 .buildParsed();
-      }
-      
-      public static Builder newBuilder() { return Builder.create(); }
-      public Builder newBuilderForType() { return newBuilder(); }
-      public static Builder newBuilder(net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail prototype) {
-        return newBuilder().mergeFrom(prototype);
-      }
-      public Builder toBuilder() { return newBuilder(this); }
-      
-      @java.lang.Override
-      protected Builder newBuilderForType(
-          com.google.protobuf.GeneratedMessage.BuilderParent parent) {
-        Builder builder = new Builder(parent);
-        return builder;
-      }
-      public static final class Builder extends
-          com.google.protobuf.GeneratedMessage.Builder<Builder>
-         implements net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetailOrBuilder {
-        public static final com.google.protobuf.Descriptors.Descriptor
-            getDescriptor() {
-          return net.sf.briar.api.protocol.Transport.internal_static_protocol_TransportDetails_TransportDetail_descriptor;
-        }
-        
-        protected com.google.protobuf.GeneratedMessage.FieldAccessorTable
-            internalGetFieldAccessorTable() {
-          return net.sf.briar.api.protocol.Transport.internal_static_protocol_TransportDetails_TransportDetail_fieldAccessorTable;
-        }
-        
-        // Construct using net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail.newBuilder()
-        private Builder() {
-          maybeForceBuilderInitialization();
-        }
-        
-        private Builder(BuilderParent parent) {
-          super(parent);
-          maybeForceBuilderInitialization();
-        }
-        private void maybeForceBuilderInitialization() {
-          if (com.google.protobuf.GeneratedMessage.alwaysUseFieldBuilders) {
-          }
-        }
-        private static Builder create() {
-          return new Builder();
-        }
-        
-        public Builder clear() {
-          super.clear();
-          key_ = "";
-          bitField0_ = (bitField0_ & ~0x00000001);
-          value_ = "";
-          bitField0_ = (bitField0_ & ~0x00000002);
-          return this;
-        }
-        
-        public Builder clone() {
-          return create().mergeFrom(buildPartial());
-        }
-        
-        public com.google.protobuf.Descriptors.Descriptor
-            getDescriptorForType() {
-          return net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail.getDescriptor();
-        }
-        
-        public net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail getDefaultInstanceForType() {
-          return net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail.getDefaultInstance();
-        }
-        
-        public net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail build() {
-          net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail result = buildPartial();
-          if (!result.isInitialized()) {
-            throw newUninitializedMessageException(result);
-          }
-          return result;
-        }
-        
-        private net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail buildParsed()
-            throws com.google.protobuf.InvalidProtocolBufferException {
-          net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail result = buildPartial();
-          if (!result.isInitialized()) {
-            throw newUninitializedMessageException(
-              result).asInvalidProtocolBufferException();
-          }
-          return result;
-        }
-        
-        public net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail buildPartial() {
-          net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail result = new net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail(this);
-          int from_bitField0_ = bitField0_;
-          int to_bitField0_ = 0;
-          if (((from_bitField0_ & 0x00000001) == 0x00000001)) {
-            to_bitField0_ |= 0x00000001;
-          }
-          result.key_ = key_;
-          if (((from_bitField0_ & 0x00000002) == 0x00000002)) {
-            to_bitField0_ |= 0x00000002;
-          }
-          result.value_ = value_;
-          result.bitField0_ = to_bitField0_;
-          onBuilt();
-          return result;
-        }
-        
-        public Builder mergeFrom(com.google.protobuf.Message other) {
-          if (other instanceof net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail) {
-            return mergeFrom((net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail)other);
-          } else {
-            super.mergeFrom(other);
-            return this;
-          }
-        }
-        
-        public Builder mergeFrom(net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail other) {
-          if (other == net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail.getDefaultInstance()) return this;
-          if (other.hasKey()) {
-            setKey(other.getKey());
-          }
-          if (other.hasValue()) {
-            setValue(other.getValue());
-          }
-          this.mergeUnknownFields(other.getUnknownFields());
-          return this;
-        }
-        
-        public final boolean isInitialized() {
-          if (!hasKey()) {
-            
-            return false;
-          }
-          return true;
-        }
-        
-        public Builder mergeFrom(
-            com.google.protobuf.CodedInputStream input,
-            com.google.protobuf.ExtensionRegistryLite extensionRegistry)
-            throws java.io.IOException {
-          com.google.protobuf.UnknownFieldSet.Builder unknownFields =
-            com.google.protobuf.UnknownFieldSet.newBuilder(
-              this.getUnknownFields());
-          while (true) {
-            int tag = input.readTag();
-            switch (tag) {
-              case 0:
-                this.setUnknownFields(unknownFields.build());
-                onChanged();
-                return this;
-              default: {
-                if (!parseUnknownField(input, unknownFields,
-                                       extensionRegistry, tag)) {
-                  this.setUnknownFields(unknownFields.build());
-                  onChanged();
-                  return this;
-                }
-                break;
-              }
-              case 10: {
-                bitField0_ |= 0x00000001;
-                key_ = input.readBytes();
-                break;
-              }
-              case 18: {
-                bitField0_ |= 0x00000002;
-                value_ = input.readBytes();
-                break;
-              }
-            }
-          }
-        }
-        
-        private int bitField0_;
-        
-        // required string key = 1;
-        private java.lang.Object key_ = "";
-        public boolean hasKey() {
-          return ((bitField0_ & 0x00000001) == 0x00000001);
-        }
-        public String getKey() {
-          java.lang.Object ref = key_;
-          if (!(ref instanceof String)) {
-            String s = ((com.google.protobuf.ByteString) ref).toStringUtf8();
-            key_ = s;
-            return s;
-          } else {
-            return (String) ref;
-          }
-        }
-        public Builder setKey(String value) {
-          if (value == null) {
-    throw new NullPointerException();
-  }
-  bitField0_ |= 0x00000001;
-          key_ = value;
-          onChanged();
-          return this;
-        }
-        public Builder clearKey() {
-          bitField0_ = (bitField0_ & ~0x00000001);
-          key_ = getDefaultInstance().getKey();
-          onChanged();
-          return this;
-        }
-        void setKey(com.google.protobuf.ByteString value) {
-          bitField0_ |= 0x00000001;
-          key_ = value;
-          onChanged();
-        }
-        
-        // optional string value = 2;
-        private java.lang.Object value_ = "";
-        public boolean hasValue() {
-          return ((bitField0_ & 0x00000002) == 0x00000002);
-        }
-        public String getValue() {
-          java.lang.Object ref = value_;
-          if (!(ref instanceof String)) {
-            String s = ((com.google.protobuf.ByteString) ref).toStringUtf8();
-            value_ = s;
-            return s;
-          } else {
-            return (String) ref;
-          }
-        }
-        public Builder setValue(String value) {
-          if (value == null) {
-    throw new NullPointerException();
-  }
-  bitField0_ |= 0x00000002;
-          value_ = value;
-          onChanged();
-          return this;
-        }
-        public Builder clearValue() {
-          bitField0_ = (bitField0_ & ~0x00000002);
-          value_ = getDefaultInstance().getValue();
-          onChanged();
-          return this;
-        }
-        void setValue(com.google.protobuf.ByteString value) {
-          bitField0_ |= 0x00000002;
-          value_ = value;
-          onChanged();
-        }
-        
-        // @@protoc_insertion_point(builder_scope:protocol.TransportDetails.TransportDetail)
-      }
-      
-      static {
-        defaultInstance = new TransportDetail(true);
-        defaultInstance.initFields();
-      }
-      
-      // @@protoc_insertion_point(class_scope:protocol.TransportDetails.TransportDetail)
-    }
-    
-    // repeated .protocol.TransportDetails.TransportDetail details = 1;
-    public static final int DETAILS_FIELD_NUMBER = 1;
-    private java.util.List<net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail> details_;
-    public java.util.List<net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail> getDetailsList() {
-      return details_;
-    }
-    public java.util.List<? extends net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetailOrBuilder> 
-        getDetailsOrBuilderList() {
-      return details_;
-    }
-    public int getDetailsCount() {
-      return details_.size();
-    }
-    public net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail getDetails(int index) {
-      return details_.get(index);
-    }
-    public net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetailOrBuilder getDetailsOrBuilder(
-        int index) {
-      return details_.get(index);
-    }
-    
-    private void initFields() {
-      details_ = java.util.Collections.emptyList();
-    }
-    private byte memoizedIsInitialized = -1;
-    public final boolean isInitialized() {
-      byte isInitialized = memoizedIsInitialized;
-      if (isInitialized != -1) return isInitialized == 1;
-      
-      for (int i = 0; i < getDetailsCount(); i++) {
-        if (!getDetails(i).isInitialized()) {
-          memoizedIsInitialized = 0;
-          return false;
-        }
-      }
-      memoizedIsInitialized = 1;
-      return true;
-    }
-    
-    public void writeTo(com.google.protobuf.CodedOutputStream output)
-                        throws java.io.IOException {
-      getSerializedSize();
-      for (int i = 0; i < details_.size(); i++) {
-        output.writeMessage(1, details_.get(i));
-      }
-      getUnknownFields().writeTo(output);
-    }
-    
-    private int memoizedSerializedSize = -1;
-    public int getSerializedSize() {
-      int size = memoizedSerializedSize;
-      if (size != -1) return size;
-    
-      size = 0;
-      for (int i = 0; i < details_.size(); i++) {
-        size += com.google.protobuf.CodedOutputStream
-          .computeMessageSize(1, details_.get(i));
-      }
-      size += getUnknownFields().getSerializedSize();
-      memoizedSerializedSize = size;
-      return size;
-    }
-    
-    private static final long serialVersionUID = 0L;
-    @java.lang.Override
-    protected java.lang.Object writeReplace()
-        throws java.io.ObjectStreamException {
-      return super.writeReplace();
-    }
-    
-    public static net.sf.briar.api.protocol.Transport.TransportDetails parseFrom(
-        com.google.protobuf.ByteString data)
-        throws com.google.protobuf.InvalidProtocolBufferException {
-      return newBuilder().mergeFrom(data).buildParsed();
-    }
-    public static net.sf.briar.api.protocol.Transport.TransportDetails parseFrom(
-        com.google.protobuf.ByteString data,
-        com.google.protobuf.ExtensionRegistryLite extensionRegistry)
-        throws com.google.protobuf.InvalidProtocolBufferException {
-      return newBuilder().mergeFrom(data, extensionRegistry)
-               .buildParsed();
-    }
-    public static net.sf.briar.api.protocol.Transport.TransportDetails parseFrom(byte[] data)
-        throws com.google.protobuf.InvalidProtocolBufferException {
-      return newBuilder().mergeFrom(data).buildParsed();
-    }
-    public static net.sf.briar.api.protocol.Transport.TransportDetails parseFrom(
-        byte[] data,
-        com.google.protobuf.ExtensionRegistryLite extensionRegistry)
-        throws com.google.protobuf.InvalidProtocolBufferException {
-      return newBuilder().mergeFrom(data, extensionRegistry)
-               .buildParsed();
-    }
-    public static net.sf.briar.api.protocol.Transport.TransportDetails parseFrom(java.io.InputStream input)
-        throws java.io.IOException {
-      return newBuilder().mergeFrom(input).buildParsed();
-    }
-    public static net.sf.briar.api.protocol.Transport.TransportDetails parseFrom(
-        java.io.InputStream input,
-        com.google.protobuf.ExtensionRegistryLite extensionRegistry)
-        throws java.io.IOException {
-      return newBuilder().mergeFrom(input, extensionRegistry)
-               .buildParsed();
-    }
-    public static net.sf.briar.api.protocol.Transport.TransportDetails parseDelimitedFrom(java.io.InputStream input)
-        throws java.io.IOException {
-      Builder builder = newBuilder();
-      if (builder.mergeDelimitedFrom(input)) {
-        return builder.buildParsed();
-      } else {
-        return null;
-      }
-    }
-    public static net.sf.briar.api.protocol.Transport.TransportDetails parseDelimitedFrom(
-        java.io.InputStream input,
-        com.google.protobuf.ExtensionRegistryLite extensionRegistry)
-        throws java.io.IOException {
-      Builder builder = newBuilder();
-      if (builder.mergeDelimitedFrom(input, extensionRegistry)) {
-        return builder.buildParsed();
-      } else {
-        return null;
-      }
-    }
-    public static net.sf.briar.api.protocol.Transport.TransportDetails parseFrom(
-        com.google.protobuf.CodedInputStream input)
-        throws java.io.IOException {
-      return newBuilder().mergeFrom(input).buildParsed();
-    }
-    public static net.sf.briar.api.protocol.Transport.TransportDetails parseFrom(
-        com.google.protobuf.CodedInputStream input,
-        com.google.protobuf.ExtensionRegistryLite extensionRegistry)
-        throws java.io.IOException {
-      return newBuilder().mergeFrom(input, extensionRegistry)
-               .buildParsed();
-    }
-    
-    public static Builder newBuilder() { return Builder.create(); }
-    public Builder newBuilderForType() { return newBuilder(); }
-    public static Builder newBuilder(net.sf.briar.api.protocol.Transport.TransportDetails prototype) {
-      return newBuilder().mergeFrom(prototype);
-    }
-    public Builder toBuilder() { return newBuilder(this); }
-    
-    @java.lang.Override
-    protected Builder newBuilderForType(
-        com.google.protobuf.GeneratedMessage.BuilderParent parent) {
-      Builder builder = new Builder(parent);
-      return builder;
-    }
-    public static final class Builder extends
-        com.google.protobuf.GeneratedMessage.Builder<Builder>
-       implements net.sf.briar.api.protocol.Transport.TransportDetailsOrBuilder {
-      public static final com.google.protobuf.Descriptors.Descriptor
-          getDescriptor() {
-        return net.sf.briar.api.protocol.Transport.internal_static_protocol_TransportDetails_descriptor;
-      }
-      
-      protected com.google.protobuf.GeneratedMessage.FieldAccessorTable
-          internalGetFieldAccessorTable() {
-        return net.sf.briar.api.protocol.Transport.internal_static_protocol_TransportDetails_fieldAccessorTable;
-      }
-      
-      // Construct using net.sf.briar.api.protocol.Transport.TransportDetails.newBuilder()
-      private Builder() {
-        maybeForceBuilderInitialization();
-      }
-      
-      private Builder(BuilderParent parent) {
-        super(parent);
-        maybeForceBuilderInitialization();
-      }
-      private void maybeForceBuilderInitialization() {
-        if (com.google.protobuf.GeneratedMessage.alwaysUseFieldBuilders) {
-          getDetailsFieldBuilder();
-        }
-      }
-      private static Builder create() {
-        return new Builder();
-      }
-      
-      public Builder clear() {
-        super.clear();
-        if (detailsBuilder_ == null) {
-          details_ = java.util.Collections.emptyList();
-          bitField0_ = (bitField0_ & ~0x00000001);
-        } else {
-          detailsBuilder_.clear();
-        }
-        return this;
-      }
-      
-      public Builder clone() {
-        return create().mergeFrom(buildPartial());
-      }
-      
-      public com.google.protobuf.Descriptors.Descriptor
-          getDescriptorForType() {
-        return net.sf.briar.api.protocol.Transport.TransportDetails.getDescriptor();
-      }
-      
-      public net.sf.briar.api.protocol.Transport.TransportDetails getDefaultInstanceForType() {
-        return net.sf.briar.api.protocol.Transport.TransportDetails.getDefaultInstance();
-      }
-      
-      public net.sf.briar.api.protocol.Transport.TransportDetails build() {
-        net.sf.briar.api.protocol.Transport.TransportDetails result = buildPartial();
-        if (!result.isInitialized()) {
-          throw newUninitializedMessageException(result);
-        }
-        return result;
-      }
-      
-      private net.sf.briar.api.protocol.Transport.TransportDetails buildParsed()
-          throws com.google.protobuf.InvalidProtocolBufferException {
-        net.sf.briar.api.protocol.Transport.TransportDetails result = buildPartial();
-        if (!result.isInitialized()) {
-          throw newUninitializedMessageException(
-            result).asInvalidProtocolBufferException();
-        }
-        return result;
-      }
-      
-      public net.sf.briar.api.protocol.Transport.TransportDetails buildPartial() {
-        net.sf.briar.api.protocol.Transport.TransportDetails result = new net.sf.briar.api.protocol.Transport.TransportDetails(this);
-        int from_bitField0_ = bitField0_;
-        if (detailsBuilder_ == null) {
-          if (((bitField0_ & 0x00000001) == 0x00000001)) {
-            details_ = java.util.Collections.unmodifiableList(details_);
-            bitField0_ = (bitField0_ & ~0x00000001);
-          }
-          result.details_ = details_;
-        } else {
-          result.details_ = detailsBuilder_.build();
-        }
-        onBuilt();
-        return result;
-      }
-      
-      public Builder mergeFrom(com.google.protobuf.Message other) {
-        if (other instanceof net.sf.briar.api.protocol.Transport.TransportDetails) {
-          return mergeFrom((net.sf.briar.api.protocol.Transport.TransportDetails)other);
-        } else {
-          super.mergeFrom(other);
-          return this;
-        }
-      }
-      
-      public Builder mergeFrom(net.sf.briar.api.protocol.Transport.TransportDetails other) {
-        if (other == net.sf.briar.api.protocol.Transport.TransportDetails.getDefaultInstance()) return this;
-        if (detailsBuilder_ == null) {
-          if (!other.details_.isEmpty()) {
-            if (details_.isEmpty()) {
-              details_ = other.details_;
-              bitField0_ = (bitField0_ & ~0x00000001);
-            } else {
-              ensureDetailsIsMutable();
-              details_.addAll(other.details_);
-            }
-            onChanged();
-          }
-        } else {
-          if (!other.details_.isEmpty()) {
-            if (detailsBuilder_.isEmpty()) {
-              detailsBuilder_.dispose();
-              detailsBuilder_ = null;
-              details_ = other.details_;
-              bitField0_ = (bitField0_ & ~0x00000001);
-              detailsBuilder_ = 
-                com.google.protobuf.GeneratedMessage.alwaysUseFieldBuilders ?
-                   getDetailsFieldBuilder() : null;
-            } else {
-              detailsBuilder_.addAllMessages(other.details_);
-            }
-          }
-        }
-        this.mergeUnknownFields(other.getUnknownFields());
-        return this;
-      }
-      
-      public final boolean isInitialized() {
-        for (int i = 0; i < getDetailsCount(); i++) {
-          if (!getDetails(i).isInitialized()) {
-            
-            return false;
-          }
-        }
-        return true;
-      }
-      
-      public Builder mergeFrom(
-          com.google.protobuf.CodedInputStream input,
-          com.google.protobuf.ExtensionRegistryLite extensionRegistry)
-          throws java.io.IOException {
-        com.google.protobuf.UnknownFieldSet.Builder unknownFields =
-          com.google.protobuf.UnknownFieldSet.newBuilder(
-            this.getUnknownFields());
-        while (true) {
-          int tag = input.readTag();
-          switch (tag) {
-            case 0:
-              this.setUnknownFields(unknownFields.build());
-              onChanged();
-              return this;
-            default: {
-              if (!parseUnknownField(input, unknownFields,
-                                     extensionRegistry, tag)) {
-                this.setUnknownFields(unknownFields.build());
-                onChanged();
-                return this;
-              }
-              break;
-            }
-            case 10: {
-              net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail.Builder subBuilder = net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail.newBuilder();
-              input.readMessage(subBuilder, extensionRegistry);
-              addDetails(subBuilder.buildPartial());
-              break;
-            }
-          }
-        }
-      }
-      
-      private int bitField0_;
-      
-      // repeated .protocol.TransportDetails.TransportDetail details = 1;
-      private java.util.List<net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail> details_ =
-        java.util.Collections.emptyList();
-      private void ensureDetailsIsMutable() {
-        if (!((bitField0_ & 0x00000001) == 0x00000001)) {
-          details_ = new java.util.ArrayList<net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail>(details_);
-          bitField0_ |= 0x00000001;
-         }
-      }
-      
-      private com.google.protobuf.RepeatedFieldBuilder<
-          net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail, net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail.Builder, net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetailOrBuilder> detailsBuilder_;
-      
-      public java.util.List<net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail> getDetailsList() {
-        if (detailsBuilder_ == null) {
-          return java.util.Collections.unmodifiableList(details_);
-        } else {
-          return detailsBuilder_.getMessageList();
-        }
-      }
-      public int getDetailsCount() {
-        if (detailsBuilder_ == null) {
-          return details_.size();
-        } else {
-          return detailsBuilder_.getCount();
-        }
-      }
-      public net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail getDetails(int index) {
-        if (detailsBuilder_ == null) {
-          return details_.get(index);
-        } else {
-          return detailsBuilder_.getMessage(index);
-        }
-      }
-      public Builder setDetails(
-          int index, net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail value) {
-        if (detailsBuilder_ == null) {
-          if (value == null) {
-            throw new NullPointerException();
-          }
-          ensureDetailsIsMutable();
-          details_.set(index, value);
-          onChanged();
-        } else {
-          detailsBuilder_.setMessage(index, value);
-        }
-        return this;
-      }
-      public Builder setDetails(
-          int index, net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail.Builder builderForValue) {
-        if (detailsBuilder_ == null) {
-          ensureDetailsIsMutable();
-          details_.set(index, builderForValue.build());
-          onChanged();
-        } else {
-          detailsBuilder_.setMessage(index, builderForValue.build());
-        }
-        return this;
-      }
-      public Builder addDetails(net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail value) {
-        if (detailsBuilder_ == null) {
-          if (value == null) {
-            throw new NullPointerException();
-          }
-          ensureDetailsIsMutable();
-          details_.add(value);
-          onChanged();
-        } else {
-          detailsBuilder_.addMessage(value);
-        }
-        return this;
-      }
-      public Builder addDetails(
-          int index, net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail value) {
-        if (detailsBuilder_ == null) {
-          if (value == null) {
-            throw new NullPointerException();
-          }
-          ensureDetailsIsMutable();
-          details_.add(index, value);
-          onChanged();
-        } else {
-          detailsBuilder_.addMessage(index, value);
-        }
-        return this;
-      }
-      public Builder addDetails(
-          net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail.Builder builderForValue) {
-        if (detailsBuilder_ == null) {
-          ensureDetailsIsMutable();
-          details_.add(builderForValue.build());
-          onChanged();
-        } else {
-          detailsBuilder_.addMessage(builderForValue.build());
-        }
-        return this;
-      }
-      public Builder addDetails(
-          int index, net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail.Builder builderForValue) {
-        if (detailsBuilder_ == null) {
-          ensureDetailsIsMutable();
-          details_.add(index, builderForValue.build());
-          onChanged();
-        } else {
-          detailsBuilder_.addMessage(index, builderForValue.build());
-        }
-        return this;
-      }
-      public Builder addAllDetails(
-          java.lang.Iterable<? extends net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail> values) {
-        if (detailsBuilder_ == null) {
-          ensureDetailsIsMutable();
-          super.addAll(values, details_);
-          onChanged();
-        } else {
-          detailsBuilder_.addAllMessages(values);
-        }
-        return this;
-      }
-      public Builder clearDetails() {
-        if (detailsBuilder_ == null) {
-          details_ = java.util.Collections.emptyList();
-          bitField0_ = (bitField0_ & ~0x00000001);
-          onChanged();
-        } else {
-          detailsBuilder_.clear();
-        }
-        return this;
-      }
-      public Builder removeDetails(int index) {
-        if (detailsBuilder_ == null) {
-          ensureDetailsIsMutable();
-          details_.remove(index);
-          onChanged();
-        } else {
-          detailsBuilder_.remove(index);
-        }
-        return this;
-      }
-      public net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail.Builder getDetailsBuilder(
-          int index) {
-        return getDetailsFieldBuilder().getBuilder(index);
-      }
-      public net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetailOrBuilder getDetailsOrBuilder(
-          int index) {
-        if (detailsBuilder_ == null) {
-          return details_.get(index);  } else {
-          return detailsBuilder_.getMessageOrBuilder(index);
-        }
-      }
-      public java.util.List<? extends net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetailOrBuilder> 
-           getDetailsOrBuilderList() {
-        if (detailsBuilder_ != null) {
-          return detailsBuilder_.getMessageOrBuilderList();
-        } else {
-          return java.util.Collections.unmodifiableList(details_);
-        }
-      }
-      public net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail.Builder addDetailsBuilder() {
-        return getDetailsFieldBuilder().addBuilder(
-            net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail.getDefaultInstance());
-      }
-      public net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail.Builder addDetailsBuilder(
-          int index) {
-        return getDetailsFieldBuilder().addBuilder(
-            index, net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail.getDefaultInstance());
-      }
-      public java.util.List<net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail.Builder> 
-           getDetailsBuilderList() {
-        return getDetailsFieldBuilder().getBuilderList();
-      }
-      private com.google.protobuf.RepeatedFieldBuilder<
-          net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail, net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail.Builder, net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetailOrBuilder> 
-          getDetailsFieldBuilder() {
-        if (detailsBuilder_ == null) {
-          detailsBuilder_ = new com.google.protobuf.RepeatedFieldBuilder<
-              net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail, net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail.Builder, net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetailOrBuilder>(
-                  details_,
-                  ((bitField0_ & 0x00000001) == 0x00000001),
-                  getParentForChildren(),
-                  isClean());
-          details_ = null;
-        }
-        return detailsBuilder_;
-      }
-      
-      // @@protoc_insertion_point(builder_scope:protocol.TransportDetails)
-    }
-    
-    static {
-      defaultInstance = new TransportDetails(true);
-      defaultInstance.initFields();
-    }
-    
-    // @@protoc_insertion_point(class_scope:protocol.TransportDetails)
-  }
-  
-  private static com.google.protobuf.Descriptors.Descriptor
-    internal_static_protocol_TransportDetails_descriptor;
-  private static
-    com.google.protobuf.GeneratedMessage.FieldAccessorTable
-      internal_static_protocol_TransportDetails_fieldAccessorTable;
-  private static com.google.protobuf.Descriptors.Descriptor
-    internal_static_protocol_TransportDetails_TransportDetail_descriptor;
-  private static
-    com.google.protobuf.GeneratedMessage.FieldAccessorTable
-      internal_static_protocol_TransportDetails_TransportDetail_fieldAccessorTable;
-  
-  public static com.google.protobuf.Descriptors.FileDescriptor
-      getDescriptor() {
-    return descriptor;
-  }
-  private static com.google.protobuf.Descriptors.FileDescriptor
-      descriptor;
-  static {
-    java.lang.String[] descriptorData = {
-      "\n)net/sf/briar/api/protocol/Transport.pr" +
-      "oto\022\010protocol\"~\n\020TransportDetails\022;\n\007det" +
-      "ails\030\001 \003(\0132*.protocol.TransportDetails.T" +
-      "ransportDetail\032-\n\017TransportDetail\022\013\n\003key" +
-      "\030\001 \002(\t\022\r\n\005value\030\002 \001(\tB\033\n\031net.sf.briar.ap" +
-      "i.protocol"
-    };
-    com.google.protobuf.Descriptors.FileDescriptor.InternalDescriptorAssigner assigner =
-      new com.google.protobuf.Descriptors.FileDescriptor.InternalDescriptorAssigner() {
-        public com.google.protobuf.ExtensionRegistry assignDescriptors(
-            com.google.protobuf.Descriptors.FileDescriptor root) {
-          descriptor = root;
-          internal_static_protocol_TransportDetails_descriptor =
-            getDescriptor().getMessageTypes().get(0);
-          internal_static_protocol_TransportDetails_fieldAccessorTable = new
-            com.google.protobuf.GeneratedMessage.FieldAccessorTable(
-              internal_static_protocol_TransportDetails_descriptor,
-              new java.lang.String[] { "Details", },
-              net.sf.briar.api.protocol.Transport.TransportDetails.class,
-              net.sf.briar.api.protocol.Transport.TransportDetails.Builder.class);
-          internal_static_protocol_TransportDetails_TransportDetail_descriptor =
-            internal_static_protocol_TransportDetails_descriptor.getNestedTypes().get(0);
-          internal_static_protocol_TransportDetails_TransportDetail_fieldAccessorTable = new
-            com.google.protobuf.GeneratedMessage.FieldAccessorTable(
-              internal_static_protocol_TransportDetails_TransportDetail_descriptor,
-              new java.lang.String[] { "Key", "Value", },
-              net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail.class,
-              net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail.Builder.class);
-          return null;
-        }
-      };
-    com.google.protobuf.Descriptors.FileDescriptor
-      .internalBuildGeneratedFileFrom(descriptorData,
-        new com.google.protobuf.Descriptors.FileDescriptor[] {
-        }, assigner);
-  }
-  
-  // @@protoc_insertion_point(outer_class_scope)
-}
diff --git a/api/net/sf/briar/api/protocol/Transport.proto b/api/net/sf/briar/api/protocol/Transport.proto
deleted file mode 100644
index 749c2e9df449df6143fc6b0d9f2863637ba240f9..0000000000000000000000000000000000000000
--- a/api/net/sf/briar/api/protocol/Transport.proto
+++ /dev/null
@@ -1,13 +0,0 @@
-package protocol;
-
-option java_package = "net.sf.briar.api.protocol";
-
-message TransportDetails {
-
-	message TransportDetail {
-		required string key = 1;
-		optional string value = 2;
-	}
-
-	repeated TransportDetail details = 1;
-}
\ No newline at end of file
diff --git a/api/net/sf/briar/api/serial/FormatException.java b/api/net/sf/briar/api/serial/FormatException.java
new file mode 100644
index 0000000000000000000000000000000000000000..784f44e30685166f4a7202e17fb4be39419c9fc9
--- /dev/null
+++ b/api/net/sf/briar/api/serial/FormatException.java
@@ -0,0 +1,9 @@
+package net.sf.briar.api.serial;
+
+import java.io.IOException;
+
+public class FormatException extends IOException {
+
+	private static final long serialVersionUID = 2274966775687766337L;
+
+}
diff --git a/api/net/sf/briar/api/serial/Raw.java b/api/net/sf/briar/api/serial/Raw.java
new file mode 100644
index 0000000000000000000000000000000000000000..3196dcc46417bb23417de5a3dbce84ee0f638b54
--- /dev/null
+++ b/api/net/sf/briar/api/serial/Raw.java
@@ -0,0 +1,6 @@
+package net.sf.briar.api.serial;
+
+public interface Raw {
+
+	byte[] getBytes();
+}
diff --git a/api/net/sf/briar/api/serial/Reader.java b/api/net/sf/briar/api/serial/Reader.java
new file mode 100644
index 0000000000000000000000000000000000000000..ec90dd65f110b65f3feaa18e22bd24efdf8d5435
--- /dev/null
+++ b/api/net/sf/briar/api/serial/Reader.java
@@ -0,0 +1,54 @@
+package net.sf.briar.api.serial;
+
+import java.io.IOException;
+import java.util.List;
+import java.util.Map;
+
+public interface Reader {
+
+	boolean eof() throws IOException;
+
+	boolean hasBoolean() throws IOException;
+	boolean readBoolean() throws IOException;
+
+	boolean hasUint7() throws IOException;
+	byte readUint7() throws IOException;
+	boolean hasInt8() throws IOException;
+	byte readInt8() throws IOException;
+	boolean hasInt16() throws IOException;
+	short readInt16() throws IOException;
+	boolean hasInt32() throws IOException;
+	int readInt32() throws IOException;
+	boolean hasInt64() throws IOException;
+	long readInt64() throws IOException;
+	boolean hasIntAny() throws IOException;
+	long readIntAny() throws IOException;
+
+	boolean hasFloat32() throws IOException;
+	float readFloat32() throws IOException;
+	boolean hasFloat64() throws IOException;
+	double readFloat64() throws IOException;
+
+	boolean hasUtf8() throws IOException;
+	String readUtf8() throws IOException;
+	String readUtf8(int maxLength) throws IOException;
+
+	boolean hasRaw() throws IOException;
+	byte[] readRaw() throws IOException;
+	byte[] readRaw(int maxLength) throws IOException;
+
+	// FIXME: Add type-safe readers and iterator readers
+
+	boolean hasList(boolean definite) throws IOException;
+	List<?> readList(boolean definite) throws IOException;
+	boolean hasList() throws IOException;
+	List<?> readList() throws IOException;
+
+	boolean hasMap(boolean definite) throws IOException;
+	Map<?, ?> readMap(boolean definite) throws IOException;
+	boolean hasMap() throws IOException;
+	Map<?, ?> readMap() throws IOException;
+
+	boolean hasNull() throws IOException;
+	void readNull() throws IOException;
+}
diff --git a/api/net/sf/briar/api/serial/ReaderFactory.java b/api/net/sf/briar/api/serial/ReaderFactory.java
new file mode 100644
index 0000000000000000000000000000000000000000..64576308979ac19711e36c5bdc49344ed155ecd0
--- /dev/null
+++ b/api/net/sf/briar/api/serial/ReaderFactory.java
@@ -0,0 +1,8 @@
+package net.sf.briar.api.serial;
+
+import java.io.InputStream;
+
+public interface ReaderFactory {
+
+	Reader createReader(InputStream in);
+}
diff --git a/api/net/sf/briar/api/serial/Tag.java b/api/net/sf/briar/api/serial/Tag.java
new file mode 100644
index 0000000000000000000000000000000000000000..f133aaa161e0ac0dd0206fe38f45e5a920187f53
--- /dev/null
+++ b/api/net/sf/briar/api/serial/Tag.java
@@ -0,0 +1,13 @@
+package net.sf.briar.api.serial;
+
+public interface Tag {
+
+	// FIXME: Definite lists and maps
+	public static final byte FALSE = -1, TRUE = -2;
+	public static final byte INT8 = -3, INT16 = -4, INT32 = -5, INT64 = -6;
+	public static final byte FLOAT32 = -7, FLOAT64 = -8;
+	public static final byte UTF8 = -9, RAW = -10;
+	public static final byte LIST_DEF = -11, MAP_DEF = -12;
+	public static final byte LIST_INDEF = -13, MAP_INDEF = -14, END = -15;
+	public static final byte NULL = -16;
+}
diff --git a/api/net/sf/briar/api/serial/Writer.java b/api/net/sf/briar/api/serial/Writer.java
new file mode 100644
index 0000000000000000000000000000000000000000..48661a82fba593b5c4bc39d62047b36fdba27f11
--- /dev/null
+++ b/api/net/sf/briar/api/serial/Writer.java
@@ -0,0 +1,32 @@
+package net.sf.briar.api.serial;
+
+import java.io.IOException;
+import java.util.List;
+import java.util.Map;
+
+public interface Writer {
+
+	void writeBoolean(boolean b) throws IOException;
+
+	void writeUint7(byte b) throws IOException;
+	void writeInt8(byte b) throws IOException;
+	void writeInt16(short s) throws IOException;
+	void writeInt32(int i) throws IOException;
+	void writeInt64(long l) throws IOException;
+	void writeIntAny(long l) throws IOException;
+
+	void writeFloat32(float f) throws IOException;
+	void writeFloat64(double d) throws IOException;
+
+	void writeUtf8(String s) throws IOException;
+	void writeRaw(byte[] b) throws IOException;
+	void writeRaw(Raw r) throws IOException;
+
+	void writeList(List<?> l, boolean definite) throws IOException;
+	void writeList(List<?> l) throws IOException;
+
+	void writeMap(Map<?, ?> m, boolean definite) throws IOException;
+	void writeMap(Map<?, ?> m) throws IOException;
+
+	void writeNull() throws IOException;
+}
diff --git a/api/net/sf/briar/api/serial/WriterFactory.java b/api/net/sf/briar/api/serial/WriterFactory.java
new file mode 100644
index 0000000000000000000000000000000000000000..39fd31fe09f328ae291c592d318e7bd2a35e426f
--- /dev/null
+++ b/api/net/sf/briar/api/serial/WriterFactory.java
@@ -0,0 +1,8 @@
+package net.sf.briar.api.serial;
+
+import java.io.OutputStream;
+
+public interface WriterFactory {
+
+	Writer createWriter(OutputStream out);
+}
diff --git a/components/net/sf/briar/invitation/InvitationWorker.java b/components/net/sf/briar/invitation/InvitationWorker.java
index 9f62c59bbef0260b4d53b4865485a0a22a031b9f..cfa03930aa9893d7fd91aefe0252d233a6a722f6 100644
--- a/components/net/sf/briar/invitation/InvitationWorker.java
+++ b/components/net/sf/briar/invitation/InvitationWorker.java
@@ -7,14 +7,13 @@ import java.util.ArrayList;
 import java.util.Arrays;
 import java.util.List;
 import java.util.Map;
-import java.util.Map.Entry;
 
 import net.sf.briar.api.db.DatabaseComponent;
 import net.sf.briar.api.db.DbException;
 import net.sf.briar.api.invitation.InvitationCallback;
 import net.sf.briar.api.invitation.InvitationParameters;
-import net.sf.briar.api.protocol.Transport.TransportDetails;
-import net.sf.briar.api.protocol.Transport.TransportDetails.TransportDetail;
+import net.sf.briar.api.serial.Writer;
+import net.sf.briar.api.serial.WriterFactory;
 import net.sf.briar.util.FileUtils;
 
 class InvitationWorker implements Runnable {
@@ -22,13 +21,16 @@ class InvitationWorker implements Runnable {
 	private final InvitationCallback callback;
 	private final InvitationParameters parameters;
 	private final DatabaseComponent databaseComponent;
+	private final WriterFactory writerFactory;
 
 	InvitationWorker(final InvitationCallback callback,
 			InvitationParameters parameters,
-			DatabaseComponent databaseComponent) {
+			DatabaseComponent databaseComponent,
+			WriterFactory writerFactory) {
 		this.callback = callback;
 		this.parameters = parameters;
 		this.databaseComponent = databaseComponent;
+		this.writerFactory = writerFactory;
 	}
 
 	public void run() {
@@ -65,6 +67,7 @@ class InvitationWorker implements Runnable {
 	private File createInvitationDat(File dir) throws IOException {
 		char[] password = parameters.getPassword();
 		assert password != null;
+		Arrays.fill(password, (char) 0);
 		File invitationDat = new File(dir, "invitation.dat");
 		callback.encryptingFile(invitationDat);
 		// FIXME: Create a real invitation
@@ -74,19 +77,11 @@ class InvitationWorker implements Runnable {
 		} catch(DbException e) {
 			throw new IOException(e);
 		}
-		TransportDetails.Builder b = TransportDetails.newBuilder();
-		for(Entry<String, String> e : transports.entrySet()) {
-			TransportDetail.Builder b1 = TransportDetail.newBuilder();
-			b1.setKey(e.getKey());
-			b1.setValue(e.getValue());
-			b.addDetails(b1.build());
-		}
-		TransportDetails t = b.build();
 		FileOutputStream out = new FileOutputStream(invitationDat);
-		t.writeTo(out);
+		Writer w = writerFactory.createWriter(out);
+		w.writeMap(transports);
 		out.flush();
 		out.close();
-		Arrays.fill(password, (char) 0);
 		return invitationDat;
 	}
 
diff --git a/components/net/sf/briar/invitation/InvitationWorkerFactoryImpl.java b/components/net/sf/briar/invitation/InvitationWorkerFactoryImpl.java
index 62b9897262f02a2f10b336d75c161d7e8f4e4621..b31bab0ba013df0acdd72868035235bac8ab2a43 100644
--- a/components/net/sf/briar/invitation/InvitationWorkerFactoryImpl.java
+++ b/components/net/sf/briar/invitation/InvitationWorkerFactoryImpl.java
@@ -4,20 +4,25 @@ import net.sf.briar.api.db.DatabaseComponent;
 import net.sf.briar.api.invitation.InvitationCallback;
 import net.sf.briar.api.invitation.InvitationParameters;
 import net.sf.briar.api.invitation.InvitationWorkerFactory;
+import net.sf.briar.api.serial.WriterFactory;
 
 import com.google.inject.Inject;
 
 class InvitationWorkerFactoryImpl implements InvitationWorkerFactory {
 
 	private final DatabaseComponent databaseComponent;
+	private final WriterFactory writerFactory;
 
 	@Inject
-	InvitationWorkerFactoryImpl(DatabaseComponent databaseComponent) {
+	InvitationWorkerFactoryImpl(DatabaseComponent databaseComponent,
+			WriterFactory writerFactory) {
 		this.databaseComponent = databaseComponent;
+		this.writerFactory = writerFactory;
 	}
 
 	public Runnable createWorker(InvitationCallback callback,
 			InvitationParameters parameters) {
-		return new InvitationWorker(callback, parameters, databaseComponent);
+		return new InvitationWorker(callback, parameters, databaseComponent,
+				writerFactory);
 	}
 }
diff --git a/components/net/sf/briar/serial/RawImpl.java b/components/net/sf/briar/serial/RawImpl.java
new file mode 100644
index 0000000000000000000000000000000000000000..56bd80f2fec7f08d1a3be8f65f461b63a8a36a79
--- /dev/null
+++ b/components/net/sf/briar/serial/RawImpl.java
@@ -0,0 +1,29 @@
+package net.sf.briar.serial;
+
+import java.util.Arrays;
+
+import net.sf.briar.api.serial.Raw;
+
+class RawImpl implements Raw {
+
+	private final byte[] bytes;
+
+	RawImpl(byte[] bytes) {
+		this.bytes = bytes;
+	}
+
+	public byte[] getBytes() {
+		return bytes;
+	}
+
+	@Override
+	public int hashCode() {
+		return Arrays.hashCode(bytes);
+	}
+
+	@Override
+	public boolean equals(Object o) {
+		if(o instanceof Raw) return Arrays.equals(bytes, ((Raw) o).getBytes());
+		return false;
+	}
+}
diff --git a/components/net/sf/briar/serial/ReaderFactoryImpl.java b/components/net/sf/briar/serial/ReaderFactoryImpl.java
new file mode 100644
index 0000000000000000000000000000000000000000..2f52c03e2e1161b96a8746141b83317b56e86c11
--- /dev/null
+++ b/components/net/sf/briar/serial/ReaderFactoryImpl.java
@@ -0,0 +1,13 @@
+package net.sf.briar.serial;
+
+import java.io.InputStream;
+
+import net.sf.briar.api.serial.Reader;
+import net.sf.briar.api.serial.ReaderFactory;
+
+public class ReaderFactoryImpl implements ReaderFactory {
+
+	public Reader createReader(InputStream in) {
+		return new ReaderImpl(in);
+	}
+}
diff --git a/components/net/sf/briar/serial/ReaderImpl.java b/components/net/sf/briar/serial/ReaderImpl.java
new file mode 100644
index 0000000000000000000000000000000000000000..a57390f33ae10e9e0605963d468d29ad87a88864
--- /dev/null
+++ b/components/net/sf/briar/serial/ReaderImpl.java
@@ -0,0 +1,360 @@
+package net.sf.briar.serial;
+
+import java.io.IOException;
+import java.io.InputStream;
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+
+import net.sf.briar.api.serial.FormatException;
+import net.sf.briar.api.serial.Reader;
+import net.sf.briar.api.serial.Tag;
+
+public class ReaderImpl implements Reader {
+
+	private static final int TOO_LARGE_TO_KEEP = 4096;
+
+	private final InputStream in;
+	private boolean started = false, eof = false;
+	private byte next;
+	private byte[] stringBuffer = null;
+
+	public ReaderImpl(InputStream in) {
+		this.in = in;
+	}
+
+	private byte readNext(boolean eofAcceptable) throws IOException {
+		started = true;
+		int i = in.read();
+		if(i == -1) {
+			eof = true;
+			if(!eofAcceptable) throw new FormatException();
+		}
+		if(i > 127) i -= 256;
+		next = (byte) i;
+		return next;
+	}
+
+	public boolean eof() throws IOException {
+		if(!started) readNext(true);
+		return eof;
+	}
+
+	public boolean hasBoolean() throws IOException {
+		if(!started) readNext(true);
+		if(eof) return false;
+		return next == Tag.FALSE || next == Tag.TRUE;
+	}
+
+	public boolean readBoolean() throws IOException {
+		if(!hasBoolean()) throw new FormatException();
+		int i = next;
+		readNext(true);
+		return i == Tag.TRUE;
+	}
+
+	public boolean hasUint7() throws IOException {
+		if(!started) readNext(true);
+		if(eof) return false;
+		return next >= 0;
+	}
+
+	public byte readUint7() throws IOException {
+		if(!hasUint7()) throw new FormatException();
+		byte b = next;
+		readNext(true);
+		return b;
+	}
+
+	public boolean hasInt8() throws IOException {
+		if(!started) readNext(true);
+		if(eof) return false;
+		return next == Tag.INT8;
+	}
+
+	public byte readInt8() throws IOException {
+		if(!hasInt8()) throw new FormatException();
+		byte b = readNext(false);
+		readNext(true);
+		return b;
+	}
+
+	public boolean hasInt16() throws IOException {
+		if(!started) readNext(true);
+		if(eof) return false;
+		return next == Tag.INT16;
+	}
+
+	public short readInt16() throws IOException {
+		if(!hasInt16()) throw new FormatException();
+		byte b1 = readNext(false);
+		byte b2 = readNext(false);
+		readNext(true);
+		int i = ((b1 & 0xFF) << 8) | (b2 & 0xFF);
+		return (short) i;
+	}
+
+	public boolean hasInt32() throws IOException {
+		if(!started) readNext(true);
+		if(eof) return false;
+		return next == Tag.INT32;
+	}
+
+	public int readInt32() throws IOException {
+		if(!hasInt32()) throw new FormatException();
+		return readInt32Bits();
+	}
+
+	private int readInt32Bits() throws IOException {
+		byte b1 = readNext(false);
+		byte b2 = readNext(false);
+		byte b3 = readNext(false);
+		byte b4 = readNext(false);
+		readNext(true);
+		return ((b1 & 0xFF) << 24) | ((b2 & 0xFF) << 16) |
+		((b3 & 0xFF) << 8) | (b4 & 0xFF);
+	}
+
+	public boolean hasInt64() throws IOException {
+		if(!started) readNext(true);
+		if(eof) return false;
+		return next == Tag.INT64;
+	}
+
+	public long readInt64() throws IOException {
+		if(!hasInt64()) throw new FormatException();
+		return readInt64Bits();
+	}
+
+	private long readInt64Bits() throws IOException {
+		byte b1 = readNext(false);
+		byte b2 = readNext(false);
+		byte b3 = readNext(false);
+		byte b4 = readNext(false);
+		byte b5 = readNext(false);
+		byte b6 = readNext(false);
+		byte b7 = readNext(false);
+		byte b8 = readNext(false);
+		readNext(true);
+		return ((b1 & 0xFFL) << 56) | ((b2 & 0xFFL) << 48) |
+		((b3 & 0xFFL) << 40) | ((b4 & 0xFFL) << 32) |
+		((b5 & 0xFFL) << 24) | ((b6 & 0xFFL) << 16) |
+		((b7 & 0xFFL) << 8) | (b8 & 0xFFL);
+	}
+
+	public boolean hasIntAny() throws IOException {
+		if(!started) readNext(true);
+		if(eof) return false;
+		return next >= 0 || next == Tag.INT8 || next == Tag.INT16
+		|| next == Tag.INT32 || next == Tag.INT64;
+	}
+
+	public long readIntAny() throws IOException {
+		if(!hasIntAny()) throw new FormatException();
+		if(next >= 0) return readUint7();
+		if(next == Tag.INT8) return readInt8();
+		if(next == Tag.INT16) return readInt16();
+		if(next == Tag.INT32) return readInt32();
+		if(next == Tag.INT64) return readInt64();
+		throw new IllegalStateException();
+	}
+
+	public boolean hasFloat32() throws IOException {
+		if(!started) readNext(true);
+		if(eof) return false;
+		return next == Tag.FLOAT32;
+	}
+
+	public float readFloat32() throws IOException {
+		if(!hasFloat32()) throw new FormatException();
+		return Float.intBitsToFloat(readInt32Bits());
+	}
+
+	public boolean hasFloat64() throws IOException {
+		if(!started) readNext(true);
+		if(eof) return false;
+		return next == Tag.FLOAT64;
+	}
+
+	public double readFloat64() throws IOException {
+		if(!hasFloat64()) throw new FormatException();
+		return Double.longBitsToDouble(readInt64Bits());
+	}
+
+	public boolean hasUtf8() throws IOException {
+		if(!started) readNext(true);
+		if(eof) return false;
+		return next == Tag.UTF8;
+	}
+
+	public String readUtf8() throws IOException {
+		return readUtf8(Integer.MAX_VALUE);
+	}
+
+	public String readUtf8(int maxLength) throws IOException {
+		if(!hasUtf8()) throw new FormatException();
+		readNext(false);
+		long l = readIntAny();
+		if(l < 0 || l > maxLength) throw new FormatException();
+		int length = (int) l;
+		if(length == 0) return "";
+		if(stringBuffer == null || stringBuffer.length < length)
+			stringBuffer = new byte[length];
+		stringBuffer[0] = next;
+		int offset = 1, read = 0;
+		while(offset < length && read != -1) {
+			read = in.read(stringBuffer, offset, length - offset);
+			if(read != -1) offset += read;
+		}
+		if(offset < length) throw new FormatException();
+		String s = new String(stringBuffer, 0, length, "UTF-8");
+		if(length >= TOO_LARGE_TO_KEEP) stringBuffer = null;
+		readNext(true);
+		return s;
+	}
+
+	public boolean hasRaw() throws IOException {
+		if(!started) readNext(true);
+		if(eof) return false;
+		return next == Tag.RAW;
+	}
+
+	public byte[] readRaw() throws IOException {
+		return readRaw(Integer.MAX_VALUE);
+	}
+
+	public byte[] readRaw(int maxLength) throws IOException {
+		if(!hasRaw()) throw new FormatException();
+		readNext(false);
+		long l = readIntAny();
+		if(l < 0 || l > maxLength) throw new FormatException();
+		int length = (int) l;
+		if(length == 0) return new byte[] {};
+		byte[] b = new byte[length];
+		b[0] = next;
+		int offset = 1, read = 0;
+		while(offset < length && read != -1) {
+			read = in.read(b, offset, length - offset);
+			if(read != -1) offset += read;
+		}
+		if(offset < length) throw new FormatException();
+		readNext(true);
+		return b;
+	}
+
+	public boolean hasList(boolean definite) throws IOException {
+		if(!started) readNext(true);
+		if(eof) return false;
+		if(definite) return next == Tag.LIST_DEF;
+		else return next == Tag.LIST_INDEF;
+	}
+
+	@SuppressWarnings({ "unchecked", "rawtypes" })
+	public List<?> readList(boolean definite) throws IOException {
+		if(!hasList(definite)) throw new FormatException();
+		readNext(false);
+		List list = new ArrayList();
+		if(definite) {
+			long l = readIntAny();
+			if(l < 0 || l > Integer.MAX_VALUE) throw new FormatException();
+			int length = (int) l;
+			for(int i = 0; i < length; i++) list.add(readObject());
+		} else {
+			while(!hasEnd()) list.add(readObject());
+			readEnd();
+		}
+		return list;
+	}
+
+	private boolean hasEnd() throws IOException {
+		if(!started) readNext(true);
+		if(eof) return false;
+		return next == Tag.END;
+	}
+
+	private void readEnd() throws IOException {
+		if(!hasEnd()) throw new FormatException();
+		readNext(true);
+	}
+
+	private Object readObject() throws IOException {
+		if(!started) throw new IllegalStateException();
+		if(hasBoolean()) return Boolean.valueOf(readBoolean());
+		if(hasUint7()) return Byte.valueOf(readUint7());
+		if(hasInt8()) return Byte.valueOf(readInt8());
+		if(hasInt16()) return Short.valueOf(readInt16());
+		if(hasInt32()) return Integer.valueOf(readInt32());
+		if(hasInt64()) return Long.valueOf(readInt64());
+		if(hasFloat32()) return Float.valueOf(readFloat32());
+		if(hasFloat64()) return Double.valueOf(readFloat64());
+		if(hasUtf8()) return readUtf8();
+		if(hasRaw()) return new RawImpl(readRaw());
+		if(hasList()) return readList();
+		if(hasMap()) return readMap();
+		if(hasNull()) {
+			readNull();
+			return null;
+		}
+		throw new IllegalStateException();
+	}
+
+	public boolean hasList() throws IOException {
+		if(!started) readNext(true);
+		if(eof) return false;
+		return next == Tag.LIST_DEF || next == Tag.LIST_INDEF;
+	}
+
+	public List<?> readList() throws IOException {
+		if(hasList(true)) return readList(true);
+		if(hasList(false)) return readList(false);
+		throw new FormatException();
+	}
+
+	public boolean hasMap(boolean definite) throws IOException {
+		if(!started) readNext(true);
+		if(eof) return false;
+		if(definite) return next == Tag.MAP_DEF;
+		else return next == Tag.MAP_INDEF;
+	}
+
+	@SuppressWarnings({ "unchecked", "rawtypes" })
+	public Map<?, ?> readMap(boolean definite) throws IOException {
+		if(!hasMap(definite)) throw new FormatException();
+		readNext(false);
+		Map m = new HashMap();
+		if(definite) {
+			long l = readIntAny();
+			if(l < 0 || l > Integer.MAX_VALUE) throw new FormatException();
+			int length = (int) l;
+			for(int i = 0; i < length; i++) m.put(readObject(), readObject());
+		} else {
+			while(!hasEnd()) m.put(readObject(), readObject());
+			readEnd();
+		}
+		return m;
+	}
+
+	public boolean hasMap() throws IOException {
+		if(!started) readNext(true);
+		if(eof) return false;
+		return next == Tag.MAP_DEF || next == Tag.MAP_INDEF;
+	}
+
+	public Map<?, ?> readMap() throws IOException {
+		if(hasMap(true)) return readMap(true);
+		if(hasMap(false)) return readMap(false);
+		throw new FormatException();
+	}
+
+	public boolean hasNull() throws IOException {
+		if(!started) readNext(true);
+		if(eof) return false;
+		return next == Tag.NULL;
+	}
+
+	public void readNull() throws IOException {
+		if(!hasNull()) throw new FormatException();
+		readNext(true);
+	}
+}
diff --git a/components/net/sf/briar/serial/WriterFactoryImpl.java b/components/net/sf/briar/serial/WriterFactoryImpl.java
new file mode 100644
index 0000000000000000000000000000000000000000..de01ca18f4e24c05be122356b7362b25304fda18
--- /dev/null
+++ b/components/net/sf/briar/serial/WriterFactoryImpl.java
@@ -0,0 +1,13 @@
+package net.sf.briar.serial;
+
+import java.io.OutputStream;
+
+import net.sf.briar.api.serial.Writer;
+import net.sf.briar.api.serial.WriterFactory;
+
+public class WriterFactoryImpl implements WriterFactory {
+
+	public Writer createWriter(OutputStream out) {
+		return new WriterImpl(out);
+	}
+}
diff --git a/components/net/sf/briar/serial/WriterImpl.java b/components/net/sf/briar/serial/WriterImpl.java
new file mode 100644
index 0000000000000000000000000000000000000000..cb692846b0e2801eb63cdd99e89316344383728e
--- /dev/null
+++ b/components/net/sf/briar/serial/WriterImpl.java
@@ -0,0 +1,166 @@
+package net.sf.briar.serial;
+
+import java.io.IOException;
+import java.io.OutputStream;
+import java.util.List;
+import java.util.Map;
+import java.util.Map.Entry;
+
+import net.sf.briar.api.serial.Raw;
+import net.sf.briar.api.serial.Tag;
+import net.sf.briar.api.serial.Writer;
+
+public class WriterImpl implements Writer {
+
+	private final OutputStream out;
+
+	public WriterImpl(OutputStream out) {
+		this.out = out;
+	}
+
+	public void writeBoolean(boolean b) throws IOException {
+		if(b) out.write(Tag.TRUE);
+		else out.write(Tag.FALSE);
+	}
+
+	public void writeUint7(byte b) throws IOException {
+		if(b < 0) throw new IllegalArgumentException();
+		out.write(b);
+	}
+
+	public void writeInt8(byte b) throws IOException {
+		out.write(Tag.INT8);
+		out.write(b);
+	}
+
+	public void writeInt16(short s) throws IOException {
+		out.write(Tag.INT16);
+		out.write((byte) (s >> 8));
+		out.write((byte) ((s << 8) >> 8));
+	}
+
+	public void writeInt32(int i) throws IOException {
+		out.write(Tag.INT32);
+		writeInt32Bits(i);
+	}
+
+	private void writeInt32Bits(int i) throws IOException {
+		out.write((byte) (i >> 24));
+		out.write((byte) ((i << 8) >> 24));
+		out.write((byte) ((i << 16) >> 24));
+		out.write((byte) ((i << 24) >> 24));
+	}
+
+	public void writeInt64(long l) throws IOException {
+		out.write(Tag.INT64);
+		writeInt64Bits(l);
+	}
+
+	private void writeInt64Bits(long l) throws IOException {
+		out.write((byte) (l >> 56));
+		out.write((byte) ((l << 8) >> 56));
+		out.write((byte) ((l << 16) >> 56));
+		out.write((byte) ((l << 24) >> 56));
+		out.write((byte) ((l << 32) >> 56));
+		out.write((byte) ((l << 40) >> 56));
+		out.write((byte) ((l << 48) >> 56));
+		out.write((byte) ((l << 56) >> 56));
+	}
+
+	public void writeIntAny(long l) throws IOException {
+		if(l >= 0 && l <= Byte.MAX_VALUE)
+			writeUint7((byte) l);
+		else if(l >= Byte.MIN_VALUE && l <= Byte.MAX_VALUE)
+			writeInt8((byte) l);
+		else if(l >= Short.MIN_VALUE && l <= Short.MAX_VALUE)
+			writeInt16((short) l);
+		else if(l >= Integer.MIN_VALUE && l <= Integer.MAX_VALUE)
+			writeInt32((int) l);
+		else writeInt64(l);
+	}
+
+	public void writeFloat32(float f) throws IOException {
+		out.write(Tag.FLOAT32);
+		writeInt32Bits(Float.floatToRawIntBits(f));
+	}
+
+	public void writeFloat64(double d) throws IOException {
+		out.write(Tag.FLOAT64);
+		writeInt64Bits(Double.doubleToRawLongBits(d));
+	}
+
+	public void writeUtf8(String s) throws IOException {
+		out.write(Tag.UTF8);
+		byte[] b = s.getBytes("UTF-8");
+		writeIntAny(b.length);
+		out.write(b);
+	}
+
+	public void writeRaw(byte[] b) throws IOException {
+		out.write(Tag.RAW);
+		writeIntAny(b.length);
+		out.write(b);
+	}
+
+	public void writeRaw(Raw r) throws IOException {
+		writeRaw(r.getBytes());
+	}
+
+	public void writeList(List<?> l, boolean definite) throws IOException {
+		if(definite) {
+			out.write(Tag.LIST_DEF);
+			writeIntAny(l.size());
+			for(Object o : l) writeObject(o);
+		} else {
+			out.write(Tag.LIST_INDEF);
+			for(Object o : l) writeObject(o);
+			out.write(Tag.END);
+		}
+	}
+
+	private void writeObject(Object o) throws IOException {
+		if(o instanceof Boolean) writeBoolean((Boolean) o);
+		else if(o instanceof Byte) writeIntAny((Byte) o);
+		else if(o instanceof Short) writeIntAny((Short) o);
+		else if(o instanceof Integer) writeIntAny((Integer) o);
+		else if(o instanceof Long) writeIntAny((Long) o);
+		else if(o instanceof Float) writeFloat32((Float) o);
+		else if(o instanceof Double) writeFloat64((Double) o);
+		else if(o instanceof String) writeUtf8((String) o);
+		else if(o instanceof Raw) writeRaw((Raw) o);
+		else if(o instanceof List) writeList((List<?>) o);
+		else if(o instanceof Map) writeMap((Map<?, ?>) o);
+		else if(o == null) writeNull();
+		else throw new IllegalStateException();
+	}
+
+	public void writeList(List<?> l) throws IOException {
+		writeList(l, true);
+	}
+
+	public void writeMap(Map<?, ?> m, boolean definite) throws IOException {
+		if(definite) {
+			out.write(Tag.MAP_DEF);
+			writeIntAny(m.size());
+			for(Entry<?, ?> e : m.entrySet()) {
+				writeObject(e.getKey());
+				writeObject(e.getValue());
+			}
+		} else {
+			out.write(Tag.MAP_INDEF);
+			for(Entry<?, ?> e : m.entrySet()) {
+				writeObject(e.getKey());
+				writeObject(e.getValue());
+			}
+			out.write(Tag.END);
+		}
+	}
+
+	public void writeMap(Map<?, ?> m) throws IOException {
+		writeMap(m, true);
+	}
+
+	public void writeNull() throws IOException {
+		out.write(Tag.NULL);
+	}
+}
diff --git a/lib/protobuf.jar b/lib/protobuf.jar
deleted file mode 100644
index b5f71525296cd43eefae9b3313c14be6fda696a4..0000000000000000000000000000000000000000
Binary files a/lib/protobuf.jar and /dev/null differ
diff --git a/test/net/sf/briar/invitation/InvitationWorkerTest.java b/test/net/sf/briar/invitation/InvitationWorkerTest.java
index 60c24597b48c9c67bcf9a28eda40bd92e1955948..6d2b0790cdfd4963ce783bdf3cd3addcfca8f288 100644
--- a/test/net/sf/briar/invitation/InvitationWorkerTest.java
+++ b/test/net/sf/briar/invitation/InvitationWorkerTest.java
@@ -2,9 +2,11 @@ package net.sf.briar.invitation;
 
 import java.io.File;
 import java.io.IOException;
+import java.io.OutputStream;
 import java.util.ArrayList;
 import java.util.Collections;
 import java.util.List;
+import java.util.Map;
 
 import junit.framework.TestCase;
 import net.sf.briar.TestUtils;
@@ -12,6 +14,8 @@ import net.sf.briar.api.db.DatabaseComponent;
 import net.sf.briar.api.db.DbException;
 import net.sf.briar.api.invitation.InvitationCallback;
 import net.sf.briar.api.invitation.InvitationParameters;
+import net.sf.briar.api.serial.Writer;
+import net.sf.briar.api.serial.WriterFactory;
 
 import org.jmock.Expectations;
 import org.jmock.Mockery;
@@ -38,13 +42,14 @@ public class InvitationWorkerTest extends TestCase {
 			context.mock(InvitationParameters.class);
 		final DatabaseComponent database =
 			context.mock(DatabaseComponent.class);
+		final WriterFactory writerFactory = context.mock(WriterFactory.class);
 		context.checking(new Expectations() {{
 			oneOf(params).getChosenLocation();
 			will(returnValue(nonExistent));
 			oneOf(callback).notFound(nonExistent);
 		}});
 
-		new InvitationWorker(callback, params, database).run();
+		new InvitationWorker(callback, params, database, writerFactory).run();
 
 		context.assertIsSatisfied();
 		File[] children = testDir.listFiles();
@@ -64,13 +69,14 @@ public class InvitationWorkerTest extends TestCase {
 			context.mock(InvitationParameters.class);
 		final DatabaseComponent database =
 			context.mock(DatabaseComponent.class);
+		final WriterFactory writerFactory = context.mock(WriterFactory.class);
 		context.checking(new Expectations() {{
 			oneOf(params).getChosenLocation();
 			will(returnValue(exists));
 			oneOf(callback).notDirectory(exists);
 		}});
 
-		new InvitationWorker(callback, params, database).run();
+		new InvitationWorker(callback, params, database, writerFactory).run();
 
 		context.assertIsSatisfied();
 		File[] children = testDir.listFiles();
@@ -101,6 +107,8 @@ public class InvitationWorkerTest extends TestCase {
 
 	private void testInstallerCreation(final boolean createExe,
 			final boolean createJar) throws IOException, DbException {
+		final Map<String, String> transports =
+			Collections.singletonMap("foo", "bar");
 		final File setup = new File(testDir, "setup.dat");
 		TestUtils.createFile(setup, "foo bar baz");
 		final File invitation = new File(testDir, "invitation.dat");
@@ -121,6 +129,8 @@ public class InvitationWorkerTest extends TestCase {
 			context.mock(InvitationParameters.class);
 		final DatabaseComponent database =
 			context.mock(DatabaseComponent.class);
+		final WriterFactory writerFactory = context.mock(WriterFactory.class);
+		final Writer writer = context.mock(Writer.class);
 		context.checking(new Expectations() {{
 			oneOf(params).getChosenLocation();
 			will(returnValue(testDir));
@@ -130,7 +140,10 @@ public class InvitationWorkerTest extends TestCase {
 			will(returnValue(new char[] {'x', 'y', 'z', 'z', 'y'}));
 			oneOf(callback).encryptingFile(invitation);
 			oneOf(database).getTransports();
-			will(returnValue(Collections.singletonMap("foo", "bar")));
+			will(returnValue(transports));
+			oneOf(writerFactory).createWriter(with(any(OutputStream.class)));
+			will(returnValue(writer));
+			oneOf(writer).writeMap(transports);
 			oneOf(params).shouldCreateExe();
 			will(returnValue(createExe));
 			oneOf(params).shouldCreateJar();
@@ -148,7 +161,7 @@ public class InvitationWorkerTest extends TestCase {
 			oneOf(callback).created(expectedFiles);
 		}});
 
-		new InvitationWorker(callback, params, database).run();
+		new InvitationWorker(callback, params, database, writerFactory).run();
 
 		assertTrue(invitation.exists());
 		assertEquals(createExe, exe.exists());
diff --git a/test/net/sf/briar/serial/ReaderImplTest.java b/test/net/sf/briar/serial/ReaderImplTest.java
new file mode 100644
index 0000000000000000000000000000000000000000..ea9fb88b52c0570b58bc1740150991d0e45a0616
--- /dev/null
+++ b/test/net/sf/briar/serial/ReaderImplTest.java
@@ -0,0 +1,249 @@
+package net.sf.briar.serial;
+
+import java.io.ByteArrayInputStream;
+import java.io.IOException;
+import java.util.Arrays;
+import java.util.List;
+import java.util.Map;
+import java.util.Map.Entry;
+
+import junit.framework.TestCase;
+import net.sf.briar.api.serial.FormatException;
+import net.sf.briar.api.serial.Raw;
+import net.sf.briar.util.StringUtils;
+
+import org.junit.Test;
+
+public class ReaderImplTest extends TestCase {
+
+	private ByteArrayInputStream in = null;
+	private ReaderImpl r = null;
+
+	@Test
+	public void testReadBoolean() throws IOException {
+		setContents("FFFE");
+		assertFalse(r.readBoolean());
+		assertTrue(r.readBoolean());
+		assertTrue(r.eof());
+	}
+
+	@Test
+	public void testReadInt8() throws IOException {
+		setContents("FD00" + "FDFF" + "FD7F" + "FD80");
+		assertEquals((byte) 0, r.readInt8());
+		assertEquals((byte) -1, r.readInt8());
+		assertEquals(Byte.MAX_VALUE, r.readInt8());
+		assertEquals(Byte.MIN_VALUE, r.readInt8());
+		assertTrue(r.eof());
+	}
+
+	@Test
+	public void testReadInt16() throws IOException {
+		setContents("FC0000" + "FCFFFF" + "FC7FFF" + "FC8000");
+		assertEquals((short) 0, r.readInt16());
+		assertEquals((short) -1, r.readInt16());
+		assertEquals(Short.MAX_VALUE, r.readInt16());
+		assertEquals(Short.MIN_VALUE, r.readInt16());
+		assertTrue(r.eof());
+	}
+
+	@Test
+	public void testReadInt32() throws IOException {
+		setContents("FB00000000" + "FBFFFFFFFF" + "FB7FFFFFFF" + "FB80000000");
+		assertEquals(0, r.readInt32());
+		assertEquals(-1, r.readInt32());
+		assertEquals(Integer.MAX_VALUE, r.readInt32());
+		assertEquals(Integer.MIN_VALUE, r.readInt32());
+		assertTrue(r.eof());
+	}
+
+	@Test
+	public void testReadInt64() throws IOException {
+		setContents("FA0000000000000000" + "FAFFFFFFFFFFFFFFFF" +
+				"FA7FFFFFFFFFFFFFFF" + "FA8000000000000000");
+		assertEquals(0L, r.readInt64());
+		assertEquals(-1L, r.readInt64());
+		assertEquals(Long.MAX_VALUE, r.readInt64());
+		assertEquals(Long.MIN_VALUE, r.readInt64());
+		assertTrue(r.eof());
+	}
+
+	@Test
+	public void testReadIntAny() throws IOException {
+		setContents("00" + "7F" + "FD80" + "FDFF" + "FC0080" + "FC7FFF" +
+				"FB00008000" + "FB7FFFFFFF" + "FA0000000080000000");
+		assertEquals(0L, r.readIntAny());
+		assertEquals(127L, r.readIntAny());
+		assertEquals(-128L, r.readIntAny());
+		assertEquals(-1L, r.readIntAny());
+		assertEquals(128L, r.readIntAny());
+		assertEquals(32767L, r.readIntAny());
+		assertEquals(32768L, r.readIntAny());
+		assertEquals(2147483647L, r.readIntAny());
+		assertEquals(2147483648L, r.readIntAny());
+		assertTrue(r.eof());
+	}
+
+	@Test
+	public void testReadFloat32() throws IOException {
+		// http://babbage.cs.qc.edu/IEEE-754/Decimal.html
+		// http://steve.hollasch.net/cgindex/coding/ieeefloat.html
+		setContents("F900000000" + "F93F800000" + "F940000000" + "F9BF800000" +
+				"F980000000" + "F9FF800000" + "F97F800000" + "F97FC00000");
+		assertEquals(0F, r.readFloat32());
+		assertEquals(1F, r.readFloat32());
+		assertEquals(2F, r.readFloat32());
+		assertEquals(-1F, r.readFloat32());
+		assertEquals(-0F, r.readFloat32());
+		assertEquals(Float.NEGATIVE_INFINITY, r.readFloat32());
+		assertEquals(Float.POSITIVE_INFINITY, r.readFloat32());
+		assertTrue(Float.isNaN(r.readFloat32()));
+		assertTrue(r.eof());
+	}
+
+	@Test
+	public void testReadFloat64() throws IOException {
+		setContents("F80000000000000000" + "F83FF0000000000000" +
+				"F84000000000000000" + "F8BFF0000000000000" +
+				"F88000000000000000" + "F8FFF0000000000000" +
+				"F87FF0000000000000" + "F87FF8000000000000");
+		assertEquals(0.0, r.readFloat64());
+		assertEquals(1.0, r.readFloat64());
+		assertEquals(2.0, r.readFloat64());
+		assertEquals(-1.0, r.readFloat64());
+		assertEquals(-0.0, r.readFloat64());
+		assertEquals(Double.NEGATIVE_INFINITY, r.readFloat64());
+		assertEquals(Double.POSITIVE_INFINITY, r.readFloat64());
+		assertTrue(Double.isNaN(r.readFloat64()));
+		assertTrue(r.eof());
+	}
+
+	@Test
+	public void testReadUtf8() throws IOException {
+		setContents("F703666F6F" + "F703666F6F" + "F700");
+		assertEquals("foo", r.readUtf8());
+		assertEquals("foo", r.readUtf8());
+		assertEquals("", r.readUtf8());
+		assertTrue(r.eof());
+	}
+
+	@Test
+	public void testReadUtf8MaxLengthNotExceeded() throws IOException {
+		setContents("F703666F6F");
+		assertEquals("foo", r.readUtf8(3));
+		assertTrue(r.eof());
+	}
+
+	@Test
+	public void testReadUtf8MaxLengthExceeded() throws IOException {
+		setContents("F703666F6F");
+		try {
+			r.readUtf8(2);
+			assertTrue(false);
+		} catch(FormatException expected) {}
+	}
+
+	@Test
+	public void testReadRaw() throws IOException {
+		setContents("F603010203" + "F603010203" + "F600");
+		assertTrue(Arrays.equals(new byte[] {1, 2, 3}, r.readRaw()));
+		assertTrue(Arrays.equals(new byte[] {1, 2, 3}, r.readRaw()));
+		assertTrue(Arrays.equals(new byte[] {}, r.readRaw()));
+		assertTrue(r.eof());
+	}
+
+	@Test
+	public void testReadRawMaxLengthNotExceeded() throws IOException {
+		setContents("F603010203");
+		assertTrue(Arrays.equals(new byte[] {1, 2, 3}, r.readRaw(3)));
+		assertTrue(r.eof());
+	}
+
+	@Test
+	public void testReadRawMaxLengthExceeded() throws IOException {
+		setContents("F603010203");
+		try {
+			r.readRaw(2);
+			assertTrue(false);
+		} catch(FormatException expected) {}
+	}
+
+	@Test
+	@SuppressWarnings("rawtypes")
+	public void testReadDefiniteList() throws IOException {
+		setContents("F5" + "03" + "01" + "F703666F6F" + "FC0080");
+		List l = r.readList(true);
+		assertNotNull(l);
+		assertEquals(3, l.size());
+		assertEquals((byte) 1, l.get(0));
+		assertEquals("foo", l.get(1));
+		assertEquals((short) 128, l.get(2));
+		assertTrue(r.eof());
+	}
+
+	@Test
+	@SuppressWarnings("rawtypes")
+	public void testReadDefiniteMap() throws IOException {
+		setContents("F4" + "02" + "F703666F6F" + "7B" + "F600" + "F0");
+		Map m = r.readMap(true);
+		assertNotNull(m);
+		assertEquals(2, m.size());
+		assertEquals((byte) 123, m.get("foo"));
+		Raw raw = new RawImpl(new byte[] {});
+		assertTrue(m.containsKey(raw));
+		assertNull(m.get(raw));
+		assertTrue(r.eof());
+	}
+
+	@Test
+	@SuppressWarnings("rawtypes")
+	public void testReadIndefiniteList() throws IOException {
+		setContents("F3" + "01" + "F703666F6F" + "FC0080" + "F1");
+		List l = r.readList(false);
+		assertNotNull(l);
+		assertEquals(3, l.size());
+		assertEquals((byte) 1, l.get(0));
+		assertEquals("foo", l.get(1));
+		assertEquals((short) 128, l.get(2));
+		assertTrue(r.eof());
+	}
+
+	@Test
+	@SuppressWarnings("rawtypes")
+	public void testReadIndefiniteMap() throws IOException {
+		setContents("F2" + "F703666F6F" + "7B" + "F600" + "F0" + "F1");
+		Map m = r.readMap(false);
+		assertNotNull(m);
+		assertEquals(2, m.size());
+		assertEquals((byte) 123, m.get("foo"));
+		Raw raw = new RawImpl(new byte[] {});
+		assertTrue(m.containsKey(raw));
+		assertNull(m.get(raw));
+		assertTrue(r.eof());
+	}
+
+	@Test
+	@SuppressWarnings("rawtypes")
+	public void testReadNestedMapsAndLists() throws IOException {
+		setContents("F4" + "01" + "F4" + "01" + "F703666F6F" + "7B" +
+				"F5" + "01" + "01");
+		Map m = r.readMap();
+		assertNotNull(m);
+		assertEquals(1, m.size());
+		Entry e = (Entry) m.entrySet().iterator().next();
+		Map m1 = (Map) e.getKey();
+		assertNotNull(m1);
+		assertEquals(1, m1.size());
+		assertEquals((byte) 123, m1.get("foo"));
+		List l = (List) e.getValue();
+		assertNotNull(l);
+		assertEquals(1, l.size());
+		assertEquals((byte) 1, l.get(0));
+		assertTrue(r.eof());
+	}
+
+	private void setContents(String hex) {
+		in = new ByteArrayInputStream(StringUtils.fromHexString(hex));
+		r = new ReaderImpl(in);
+	}
+}
diff --git a/test/net/sf/briar/serial/WriterImplTest.java b/test/net/sf/briar/serial/WriterImplTest.java
new file mode 100644
index 0000000000000000000000000000000000000000..c50e3a552fa68dc7bb359bef6a0978cebb38efa3
--- /dev/null
+++ b/test/net/sf/briar/serial/WriterImplTest.java
@@ -0,0 +1,219 @@
+package net.sf.briar.serial;
+
+import java.io.ByteArrayOutputStream;
+import java.io.IOException;
+import java.util.ArrayList;
+import java.util.Arrays;
+import java.util.LinkedHashMap;
+import java.util.List;
+import java.util.Map;
+
+import junit.framework.TestCase;
+import net.sf.briar.util.StringUtils;
+
+import org.junit.Before;
+import org.junit.Test;
+
+public class WriterImplTest extends TestCase {
+
+	private ByteArrayOutputStream out = null;
+	private WriterImpl w = null;
+
+	@Before
+	public void setUp() {
+		out = new ByteArrayOutputStream();
+		w = new WriterImpl(out);
+	}
+
+	@Test
+	public void testWriteBoolean() throws IOException {
+		w.writeBoolean(true);
+		w.writeBoolean(false);
+		checkContents("FEFF");
+	}
+
+	@Test
+	public void testWriteUint7() throws IOException {
+		w.writeUint7((byte) 0);
+		w.writeUint7((byte) 127);
+		checkContents("00" + "7F");
+	}
+
+	@Test
+	public void testWriteInt8() throws IOException {
+		w.writeInt8((byte) 0);
+		w.writeInt8((byte) -1);
+		w.writeInt8((byte) -128);
+		w.writeInt8((byte) 127);
+		checkContents("FD00" + "FDFF" + "FD80" + "FD7F");
+	}
+
+	@Test
+	public void testWriteInt16() throws IOException {
+		w.writeInt16((short) 0);
+		w.writeInt16((short) -1);
+		w.writeInt16((short) -32768);
+		w.writeInt16((short) 32767);
+		checkContents("FC0000" + "FCFFFF" + "FC8000" + "FC7FFF");
+	}
+
+	@Test
+	public void testWriteInt32() throws IOException {
+		w.writeInt32(0);
+		w.writeInt32(-1);
+		w.writeInt32(-2147483648);
+		w.writeInt32(2147483647);
+		checkContents("FB00000000" + "FBFFFFFFFF" +
+				"FB80000000" + "FB7FFFFFFF");
+	}
+
+	@Test
+	public void testWriteInt64() throws IOException {
+		w.writeInt64(0L);
+		w.writeInt64(-1L);
+		w.writeInt64(-9223372036854775808L);
+		w.writeInt64(9223372036854775807L);
+		checkContents("FA0000000000000000" + "FAFFFFFFFFFFFFFFFF" +
+				"FA8000000000000000" + "FA7FFFFFFFFFFFFFFF");
+	}
+
+	@Test
+	public void testWriteIntAny() throws IOException {
+		w.writeIntAny(0L); // uint7
+		w.writeIntAny(127L); // uint7
+		w.writeIntAny(-1L); // int8
+		w.writeIntAny(128L); // int16
+		w.writeIntAny(32767L); // int16
+		w.writeIntAny(32768L); // int32
+		w.writeIntAny(2147483647L); // int32
+		w.writeIntAny(2147483648L); // int64
+		checkContents("00" + "7F" + "FDFF" + "FC0080" + "FC7FFF" +
+				"FB00008000" + "FB7FFFFFFF" + "FA0000000080000000");
+	}
+
+	@Test
+	public void testWriteFloat32() throws IOException {
+		// http://babbage.cs.qc.edu/IEEE-754/Decimal.html
+		// 1 bit for sign, 8 for exponent, 23 for significand 
+		w.writeFloat32(0F); // 0 0 0 -> 0x00000000
+		w.writeFloat32(1F); // 0 127 1 -> 0x3F800000
+		w.writeFloat32(2F); // 0 128 1 -> 0x40000000
+		w.writeFloat32(-1F); // 1 127 1 -> 0xBF800000
+		w.writeFloat32(-0F); // 1 0 0 -> 0x80000000
+		// http://steve.hollasch.net/cgindex/coding/ieeefloat.html
+		w.writeFloat32(Float.NEGATIVE_INFINITY); // 1 255 0 -> 0xFF800000
+		w.writeFloat32(Float.POSITIVE_INFINITY); // 0 255 0 -> 0x7F800000
+		w.writeFloat32(Float.NaN); // 0 255 1 -> 0x7FC00000
+		checkContents("F900000000" + "F93F800000" + "F940000000" +
+				"F9BF800000" + "F980000000" + "F9FF800000" +
+				"F97F800000" + "F97FC00000");
+	}
+
+	@Test
+	public void testWriteFloat64() throws IOException {
+		// 1 bit for sign, 11 for exponent, 52 for significand 
+		w.writeFloat64(0.0); // 0 0 0 -> 0x0000000000000000
+		w.writeFloat64(1.0); // 0 1023 1 -> 0x3FF0000000000000
+		w.writeFloat64(2.0); // 0 1024 1 -> 0x4000000000000000
+		w.writeFloat64(-1.0); // 1 1023 1 -> 0xBFF0000000000000
+		w.writeFloat64(-0.0); // 1 0 0 -> 0x8000000000000000
+		w.writeFloat64(Double.NEGATIVE_INFINITY); // 1 2047 0 -> 0xFFF00000...
+		w.writeFloat64(Double.POSITIVE_INFINITY); // 0 2047 0 -> 0x7FF00000...
+		w.writeFloat64(Double.NaN); // 0 2047 1 -> 0x7FF8000000000000
+		checkContents("F80000000000000000" + "F83FF0000000000000" +
+				"F84000000000000000" + "F8BFF0000000000000" +
+				"F88000000000000000" + "F8FFF0000000000000" +
+				"F87FF0000000000000" + "F87FF8000000000000");
+	}
+
+	@Test
+	public void testWriteUtf8() throws IOException {
+		w.writeUtf8("foo");
+		// UTF-8 tag, length as uint7, UTF-8 bytes
+		checkContents("F7" + "03" + "666F6F");
+	}
+
+	@Test
+	public void testWriteRawBytes() throws IOException {
+		w.writeRaw(new byte[] {0, 1, -1, 127, -128});
+		checkContents("F6" + "05" + "0001FF7F80");
+	}
+
+	@Test
+	public void testWriteRawObject() throws IOException {
+		w.writeRaw(new RawImpl(new byte[] {0, 1, -1, 127, -128}));
+		checkContents("F6" + "05" + "0001FF7F80");
+	}
+
+	@Test
+	@SuppressWarnings({ "rawtypes", "unchecked" })
+	public void testWriteDefiniteList() throws IOException {
+		List l = new ArrayList();
+		l.add(Byte.valueOf((byte) 1)); // Written as a uint7
+		l.add("foo");
+		l.add(Long.valueOf(128L)); // Written as an int16
+		w.writeList(l, true);
+		checkContents("F5" + "03" + "01" + "F703666F6F" + "FC0080");
+	}
+	
+	@Test
+	@SuppressWarnings({ "rawtypes", "unchecked" })
+	public void testWriteDefiniteMap() throws IOException {
+		// Use LinkedHashMap to get predictable iteration order
+		Map m = new LinkedHashMap();
+		m.put("foo", Integer.valueOf(123)); // Written as a uint7
+		m.put(new RawImpl(new byte[] {}), null); // Empty array != null
+		w.writeMap(m, true);
+		checkContents("F4" + "02" + "F703666F6F" + "7B" + "F600" + "F0");
+	}
+
+	@Test
+	@SuppressWarnings({ "rawtypes", "unchecked" })
+	public void testWriteIndefiniteList() throws IOException {
+		List l = new ArrayList();
+		l.add(Byte.valueOf((byte) 1)); // Written as a uint7
+		l.add("foo");
+		l.add(Long.valueOf(128L)); // Written as an int16
+		w.writeList(l, false);
+		checkContents("F3" + "01" + "F703666F6F" + "FC0080" + "F1");
+	}
+
+	@Test
+	@SuppressWarnings({ "rawtypes", "unchecked" })
+	public void testWriteIndefiniteMap() throws IOException {
+		// Use LinkedHashMap to get predictable iteration order
+		Map m = new LinkedHashMap();
+		m.put("foo", Integer.valueOf(123)); // Written as a uint7
+		m.put(new RawImpl(new byte[] {}), null); // Empty array != null
+		w.writeMap(m, false);
+		checkContents("F2" + "F703666F6F" + "7B" + "F600" + "F0" + "F1");
+	}
+
+	@Test
+	@SuppressWarnings({ "rawtypes", "unchecked" })
+	public void testWriteNestedMapsAndLists() throws IOException {
+		Map m = new LinkedHashMap();
+		m.put("foo", Integer.valueOf(123));
+		List l = new ArrayList();
+		l.add(Byte.valueOf((byte) 1));
+		Map m1 = new LinkedHashMap();
+		m1.put(m, l);
+		w.writeMap(m1);
+		checkContents("F4" + "01" + "F4" + "01" + "F703666F6F" + "7B" +
+				"F5" + "01" + "01");
+	}
+
+	@Test
+	public void testWriteNull() throws IOException {
+		w.writeNull();
+		checkContents("F0");
+	}
+
+	private void checkContents(String hex) throws IOException {
+		out.flush();
+		out.close();
+		byte[] expected = StringUtils.fromHexString(hex);
+		assertTrue(StringUtils.toHexString(out.toByteArray()),
+				Arrays.equals(expected, out.toByteArray()));
+	}
+}