2014-07-01 21:52:12 +09:30
|
|
|
/**
|
|
|
|
* Copyright (C) 2014 Serval Project Inc.
|
|
|
|
*
|
|
|
|
* This file is part of Serval Software (http://www.servalproject.org)
|
|
|
|
*
|
|
|
|
* Serval Software is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This source code is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this source code; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
package org.servalproject.servaldna.rhizome;
|
|
|
|
|
|
|
|
import java.util.Map;
|
|
|
|
import java.util.HashMap;
|
|
|
|
import java.util.HashSet;
|
|
|
|
import java.io.UnsupportedEncodingException;
|
|
|
|
import java.io.IOException;
|
|
|
|
import java.io.InputStream;
|
2014-07-07 17:54:52 +09:30
|
|
|
import java.io.OutputStream;
|
2014-07-01 21:52:12 +09:30
|
|
|
import java.io.OutputStreamWriter;
|
2014-07-07 17:54:52 +09:30
|
|
|
import java.io.ByteArrayInputStream;
|
|
|
|
import java.io.ByteArrayOutputStream;
|
2014-07-01 21:52:12 +09:30
|
|
|
import org.servalproject.servaldna.AbstractId;
|
|
|
|
import org.servalproject.servaldna.SubscriberId;
|
|
|
|
import org.servalproject.servaldna.BundleId;
|
|
|
|
import org.servalproject.servaldna.FileHash;
|
|
|
|
import org.servalproject.servaldna.BundleKey;
|
|
|
|
|
|
|
|
public class RhizomeManifest {
|
|
|
|
|
|
|
|
public final static int TEXT_FORMAT_MAX_SIZE = 8192;
|
|
|
|
|
|
|
|
// Core fields used for routing and expiry (cannot be null)
|
|
|
|
public final BundleId id;
|
|
|
|
public final long version;
|
|
|
|
public final long filesize;
|
|
|
|
|
|
|
|
// Principal fields (can be null)
|
|
|
|
public final FileHash filehash; // null iff filesize == 0
|
|
|
|
public final SubscriberId sender;
|
|
|
|
public final SubscriberId recipient;
|
|
|
|
public final BundleKey BK;
|
|
|
|
public final Long tail; // null iff not a journal
|
|
|
|
public final Integer crypt;
|
|
|
|
|
|
|
|
// Optional fields (all can be null)
|
|
|
|
public final Long date; // can be null
|
|
|
|
public final String service;
|
|
|
|
public final String name;
|
|
|
|
|
2014-07-07 17:54:52 +09:30
|
|
|
protected HashMap<String,String> extraFields;
|
2014-07-01 21:52:12 +09:30
|
|
|
private byte[] signatureBlock;
|
|
|
|
private byte[] textFormat;
|
|
|
|
|
|
|
|
protected RhizomeManifest( BundleId id,
|
|
|
|
long version,
|
|
|
|
long filesize,
|
|
|
|
FileHash filehash,
|
|
|
|
SubscriberId sender,
|
|
|
|
SubscriberId recipient,
|
|
|
|
BundleKey BK,
|
|
|
|
Integer crypt,
|
|
|
|
Long tail,
|
|
|
|
Long date,
|
|
|
|
String service,
|
|
|
|
String name
|
|
|
|
)
|
|
|
|
{
|
|
|
|
assert id != null;
|
|
|
|
if (filesize == 0)
|
|
|
|
assert filehash == null;
|
|
|
|
else
|
|
|
|
assert filehash != null;
|
|
|
|
this.id = id;
|
|
|
|
this.version = version;
|
|
|
|
this.filesize = filesize;
|
|
|
|
this.filehash = filehash;
|
|
|
|
this.sender = sender;
|
|
|
|
this.recipient = recipient;
|
|
|
|
this.BK = BK;
|
|
|
|
this.crypt = crypt;
|
|
|
|
this.tail = tail;
|
|
|
|
this.date = date;
|
|
|
|
this.service = service;
|
|
|
|
this.name = name;
|
|
|
|
this.extraFields = null;
|
|
|
|
this.signatureBlock = null;
|
|
|
|
this.textFormat = null;
|
|
|
|
}
|
|
|
|
|
2014-07-07 17:54:52 +09:30
|
|
|
protected RhizomeManifest(RhizomeIncompleteManifest m)
|
|
|
|
{
|
|
|
|
this(m.id, m.version, m.filesize, m.filehash, m.sender, m.recipient, m.BK, m.crypt, m.tail, m.date, m.service, m.name);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Return the Rhizome manifest in its text format representation, with the signature block at
|
|
|
|
* the end if present.
|
2014-07-01 21:52:12 +09:30
|
|
|
*
|
|
|
|
* @author Andrew Bettison <andrew@servalproject.com>
|
|
|
|
*/
|
|
|
|
public byte[] toTextFormat() throws RhizomeManifestSizeException
|
|
|
|
{
|
2014-07-07 17:54:52 +09:30
|
|
|
if (this.textFormat == null) {
|
2014-07-01 21:52:12 +09:30
|
|
|
try {
|
|
|
|
ByteArrayOutputStream os = new ByteArrayOutputStream();
|
2014-07-07 17:54:52 +09:30
|
|
|
toTextFormat(os);
|
|
|
|
os.close();
|
2014-07-01 21:52:12 +09:30
|
|
|
if (os.size() > TEXT_FORMAT_MAX_SIZE)
|
|
|
|
throw new RhizomeManifestSizeException("manifest text format overflow", os.size(), TEXT_FORMAT_MAX_SIZE);
|
2014-07-07 17:54:52 +09:30
|
|
|
this.textFormat = os.toByteArray();
|
2014-07-01 21:52:12 +09:30
|
|
|
}
|
|
|
|
catch (IOException e) {
|
|
|
|
// should not happen with ByteArrayOutputStream
|
|
|
|
return new byte[0];
|
|
|
|
}
|
|
|
|
}
|
2014-07-07 17:54:52 +09:30
|
|
|
byte[] ret = new byte[this.textFormat.length];
|
|
|
|
System.arraycopy(this.textFormat, 0, ret, 0, ret.length);
|
2014-07-01 21:52:12 +09:30
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2014-07-07 17:54:52 +09:30
|
|
|
/** Write the Rhizome manifest in its text format representation to the given output stream,
|
|
|
|
* with the signature block at the end if present.
|
|
|
|
*
|
|
|
|
* @author Andrew Bettison <andrew@servalproject.com>
|
|
|
|
*/
|
|
|
|
public void toTextFormat(OutputStream os) throws IOException
|
|
|
|
{
|
|
|
|
new RhizomeIncompleteManifest(this).toTextFormat(os);
|
|
|
|
if (this.signatureBlock != null) {
|
|
|
|
os.write(0);
|
|
|
|
os.write(this.signatureBlock);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-01 21:52:12 +09:30
|
|
|
/** Construct a Rhizome manifest from its text format representation.
|
|
|
|
*
|
|
|
|
* @author Andrew Bettison <andrew@servalproject.com>
|
|
|
|
*/
|
|
|
|
static public RhizomeManifest fromTextFormat(byte[] bytes) throws RhizomeManifestParseException
|
|
|
|
{
|
|
|
|
return fromTextFormat(bytes, 0, bytes.length);
|
|
|
|
}
|
|
|
|
|
2014-07-07 17:54:52 +09:30
|
|
|
/** Construct a complete Rhizome manifest from its text format representation, including a
|
|
|
|
* trailing signature block.
|
2014-07-01 21:52:12 +09:30
|
|
|
*
|
|
|
|
* @author Andrew Bettison <andrew@servalproject.com>
|
|
|
|
*/
|
|
|
|
static public RhizomeManifest fromTextFormat(byte[] bytes, int off, int len) throws RhizomeManifestParseException
|
|
|
|
{
|
|
|
|
// The signature block follows the first nul character at the start of a line.
|
|
|
|
byte[] sigblock = null;
|
|
|
|
int proplen = len;
|
|
|
|
for (int i = 0; i < len; ++i) {
|
|
|
|
if (bytes[off + i] == 0 && (i == 0 || bytes[off + i - 1] == '\n')) {
|
|
|
|
sigblock = new byte[len - i - 1];
|
|
|
|
System.arraycopy(bytes, off + i + 1, sigblock, 0, sigblock.length);
|
|
|
|
proplen = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2014-07-07 17:54:52 +09:30
|
|
|
RhizomeIncompleteManifest im = new RhizomeIncompleteManifest();
|
2014-07-01 21:52:12 +09:30
|
|
|
try {
|
2014-07-07 17:54:52 +09:30
|
|
|
im.parseTextFormat(new ByteArrayInputStream(bytes, off, proplen));
|
2014-07-01 21:52:12 +09:30
|
|
|
}
|
2014-07-07 17:54:52 +09:30
|
|
|
catch (IOException e) {
|
2014-07-01 21:52:12 +09:30
|
|
|
}
|
2014-07-07 17:54:52 +09:30
|
|
|
if (im.id == null)
|
2014-07-01 21:52:12 +09:30
|
|
|
throw new RhizomeManifestParseException("missing 'id' field");
|
2014-07-07 17:54:52 +09:30
|
|
|
if (im.version == null)
|
2014-07-01 21:52:12 +09:30
|
|
|
throw new RhizomeManifestParseException("missing 'version' field");
|
2014-07-07 17:54:52 +09:30
|
|
|
if (im.filesize == null)
|
2014-07-01 21:52:12 +09:30
|
|
|
throw new RhizomeManifestParseException("missing 'filesize' field");
|
2014-07-07 17:54:52 +09:30
|
|
|
if (im.filesize != 0 && im.filehash == null)
|
2014-07-01 21:52:12 +09:30
|
|
|
throw new RhizomeManifestParseException("missing 'filehash' field");
|
2014-07-07 17:54:52 +09:30
|
|
|
else if (im.filesize == 0 && im.filehash != null)
|
2014-07-01 21:52:12 +09:30
|
|
|
throw new RhizomeManifestParseException("spurious 'filehash' field");
|
2014-07-07 17:54:52 +09:30
|
|
|
RhizomeManifest m = new RhizomeManifest(im);
|
2014-07-01 21:52:12 +09:30
|
|
|
m.signatureBlock = sigblock;
|
|
|
|
m.textFormat = new byte[len];
|
|
|
|
System.arraycopy(bytes, off, m.textFormat, 0, m.textFormat.length);
|
|
|
|
return m;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Convenience method: construct a Rhizome manifest from all the bytes read from a given
|
|
|
|
* InputStream.
|
|
|
|
*
|
|
|
|
* @author Andrew Bettison <andrew@servalproject.com>
|
|
|
|
*/
|
|
|
|
static public RhizomeManifest fromTextFormat(InputStream in) throws IOException, RhizomeManifestParseException
|
|
|
|
{
|
|
|
|
byte[] bytes = new byte[TEXT_FORMAT_MAX_SIZE];
|
|
|
|
int n = 0;
|
|
|
|
int offset = 0;
|
|
|
|
while (offset < bytes.length && (n = in.read(bytes, offset, bytes.length - offset)) != -1)
|
|
|
|
offset += n;
|
|
|
|
assert offset <= bytes.length;
|
|
|
|
if (offset == bytes.length)
|
|
|
|
n = in.read();
|
|
|
|
if (n != -1)
|
|
|
|
throw new RhizomeManifestParseException("input stream too long");
|
|
|
|
return fromTextFormat(bytes, 0, offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
private static boolean isFieldNameFirstChar(char c)
|
|
|
|
{
|
|
|
|
return (c >= 'A' && c <= 'Z') || (c >= 'a' && c <= 'z');
|
|
|
|
}
|
|
|
|
|
|
|
|
private static boolean isFieldNameChar(char c)
|
|
|
|
{
|
|
|
|
return (c >= 'A' && c <= 'Z') || (c >= 'a' && c <= 'z') || (c >= '0' && c <= '9');
|
|
|
|
}
|
|
|
|
|
|
|
|
private static Long parseUnsignedLong(String text) throws NumberFormatException
|
|
|
|
{
|
|
|
|
Long value = Long.valueOf(text);
|
|
|
|
if (value < 0)
|
|
|
|
throw new NumberFormatException("negative value not allowed");
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|