2014-06-27 07:59:57 +00:00
|
|
|
/**
|
|
|
|
* 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;
|
|
|
|
|
2014-07-01 12:22:12 +00:00
|
|
|
import java.lang.reflect.InvocationTargetException;
|
|
|
|
import java.util.Map;
|
|
|
|
import java.util.List;
|
2014-06-27 07:59:57 +00:00
|
|
|
import java.io.IOException;
|
2014-07-01 12:22:12 +00:00
|
|
|
import java.io.InputStream;
|
2014-06-27 07:59:57 +00:00
|
|
|
import java.io.InputStreamReader;
|
|
|
|
import java.net.HttpURLConnection;
|
|
|
|
import org.servalproject.json.JSONTokeniser;
|
|
|
|
import org.servalproject.json.JSONInputException;
|
2014-07-01 12:22:12 +00:00
|
|
|
import org.servalproject.servaldna.BundleId;
|
|
|
|
import org.servalproject.servaldna.SubscriberId;
|
|
|
|
import org.servalproject.servaldna.BundleSecret;
|
|
|
|
import org.servalproject.servaldna.ServalDHttpConnectionFactory;
|
|
|
|
import org.servalproject.servaldna.ServalDInterfaceException;
|
|
|
|
import org.servalproject.servaldna.ServalDFailureException;
|
2014-06-27 07:59:57 +00:00
|
|
|
|
|
|
|
public class RhizomeCommon
|
|
|
|
{
|
2014-07-01 12:22:12 +00:00
|
|
|
|
|
|
|
protected static InputStream receiveResponse(HttpURLConnection conn, int expected_response_code) throws IOException, ServalDInterfaceException
|
2014-06-27 07:59:57 +00:00
|
|
|
{
|
|
|
|
int[] expected_response_codes = { expected_response_code };
|
2014-07-01 12:22:12 +00:00
|
|
|
return receiveResponse(conn, expected_response_codes);
|
2014-06-27 07:59:57 +00:00
|
|
|
}
|
|
|
|
|
2014-07-01 12:22:12 +00:00
|
|
|
protected static InputStream receiveResponse(HttpURLConnection conn, int[] expected_response_codes) throws IOException, ServalDInterfaceException
|
2014-06-27 07:59:57 +00:00
|
|
|
{
|
2014-07-01 12:22:12 +00:00
|
|
|
for (int code: expected_response_codes) {
|
|
|
|
if (conn.getResponseCode() == code)
|
|
|
|
return conn.getInputStream();
|
|
|
|
}
|
2014-06-27 07:59:57 +00:00
|
|
|
if (!conn.getContentType().equals("application/json"))
|
|
|
|
throw new ServalDInterfaceException("unexpected HTTP Content-Type: " + conn.getContentType());
|
|
|
|
if (conn.getResponseCode() == HttpURLConnection.HTTP_FORBIDDEN) {
|
|
|
|
JSONTokeniser json = new JSONTokeniser(new InputStreamReader(conn.getErrorStream(), "US-ASCII"));
|
|
|
|
Status status = decodeRestfulStatus(json);
|
|
|
|
throw new ServalDInterfaceException("unexpected Rhizome failure, \"" + status.message + "\"");
|
|
|
|
}
|
|
|
|
throw new ServalDInterfaceException("unexpected HTTP response code: " + conn.getResponseCode());
|
|
|
|
}
|
|
|
|
|
2014-07-01 12:22:12 +00:00
|
|
|
protected static JSONTokeniser receiveRestfulResponse(HttpURLConnection conn, int expected_response_code) throws IOException, ServalDInterfaceException
|
|
|
|
{
|
|
|
|
int[] expected_response_codes = { expected_response_code };
|
|
|
|
return receiveRestfulResponse(conn, expected_response_codes);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected static JSONTokeniser receiveRestfulResponse(HttpURLConnection conn, int[] expected_response_codes) throws IOException, ServalDInterfaceException
|
|
|
|
{
|
|
|
|
InputStream in = receiveResponse(conn, expected_response_codes);
|
|
|
|
if (!conn.getContentType().equals("application/json"))
|
|
|
|
throw new ServalDInterfaceException("unexpected HTTP Content-Type: " + conn.getContentType());
|
|
|
|
return new JSONTokeniser(new InputStreamReader(in, "US-ASCII"));
|
|
|
|
}
|
|
|
|
|
2014-06-27 07:59:57 +00:00
|
|
|
private static class Status {
|
|
|
|
public String message;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected static Status decodeRestfulStatus(JSONTokeniser json) throws IOException, ServalDInterfaceException
|
|
|
|
{
|
|
|
|
try {
|
|
|
|
Status status = new Status();
|
|
|
|
json.consume(JSONTokeniser.Token.START_OBJECT);
|
|
|
|
json.consume("http_status_code");
|
|
|
|
json.consume(JSONTokeniser.Token.COLON);
|
|
|
|
json.consume(Integer.class);
|
|
|
|
json.consume(JSONTokeniser.Token.COMMA);
|
|
|
|
status.message = json.consume("http_status_message");
|
|
|
|
json.consume(JSONTokeniser.Token.COLON);
|
|
|
|
String message = json.consume(String.class);
|
|
|
|
json.consume(JSONTokeniser.Token.END_OBJECT);
|
|
|
|
json.consume(JSONTokeniser.Token.EOF);
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
catch (JSONInputException e) {
|
|
|
|
throw new ServalDInterfaceException("malformed JSON status response", e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-01 12:22:12 +00:00
|
|
|
public static RhizomeManifestBundle rhizomeManifest(ServalDHttpConnectionFactory connector, BundleId bid) throws IOException, ServalDInterfaceException
|
|
|
|
{
|
|
|
|
HttpURLConnection conn = connector.newServalDHttpConnection("/restful/rhizome/" + bid.toHex() + ".rhm");
|
|
|
|
conn.connect();
|
|
|
|
InputStream in = RhizomeCommon.receiveResponse(conn, HttpURLConnection.HTTP_OK);
|
|
|
|
if (!conn.getContentType().equals("rhizome-manifest/text"))
|
|
|
|
throw new ServalDInterfaceException("unexpected HTTP Content-Type: " + conn.getContentType());
|
|
|
|
RhizomeManifest manifest;
|
|
|
|
try {
|
|
|
|
manifest = RhizomeManifest.fromTextFormat(in);
|
|
|
|
}
|
|
|
|
catch (RhizomeManifestParseException e) {
|
|
|
|
throw new ServalDInterfaceException("malformed manifest from daemon", e);
|
|
|
|
}
|
|
|
|
finally {
|
|
|
|
in.close();
|
|
|
|
}
|
|
|
|
Map<String,List<String>> headers = conn.getHeaderFields();
|
|
|
|
for (Map.Entry<String,List<String>> e: headers.entrySet()) {
|
|
|
|
for (String v: e.getValue()) {
|
|
|
|
System.err.println("received header " + e.getKey() + ": " + v);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
long insertTime = headerUnsignedLong(conn, "Serval-Rhizome-Bundle-Inserttime");
|
|
|
|
SubscriberId author = header(conn, "Serval-Rhizome-Bundle-Author", SubscriberId.class);
|
|
|
|
BundleSecret secret = header(conn, "Serval-Rhizome-Bundle-Secret", BundleSecret.class);
|
|
|
|
return new RhizomeManifestBundle(manifest, insertTime, author, secret);
|
|
|
|
}
|
|
|
|
|
|
|
|
private static String headerString(HttpURLConnection conn, String header) throws ServalDInterfaceException
|
|
|
|
{
|
|
|
|
String str = conn.getHeaderField(header);
|
|
|
|
if (str == null)
|
|
|
|
throw new ServalDInterfaceException("missing header field: " + header);
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static int headerInteger(HttpURLConnection conn, String header) throws ServalDInterfaceException
|
|
|
|
{
|
|
|
|
String str = headerString(conn, header);
|
|
|
|
try {
|
|
|
|
return Integer.parseInt(str);
|
|
|
|
}
|
|
|
|
catch (NumberFormatException e) {
|
|
|
|
}
|
|
|
|
throw new ServalDInterfaceException("invalid header field: " + header + ": " + str);
|
|
|
|
}
|
|
|
|
|
|
|
|
private static long headerUnsignedLong(HttpURLConnection conn, String header) throws ServalDInterfaceException
|
|
|
|
{
|
|
|
|
String str = headerString(conn, header);
|
|
|
|
try {
|
|
|
|
long value = Long.parseLong(str);
|
|
|
|
if (value >= 0)
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
catch (NumberFormatException e) {
|
|
|
|
}
|
|
|
|
throw new ServalDInterfaceException("invalid header field: " + header + ": " + str);
|
|
|
|
}
|
|
|
|
|
|
|
|
private static <T> T header(HttpURLConnection conn, String header, Class<T> cls) throws ServalDInterfaceException
|
|
|
|
{
|
|
|
|
String str = headerString(conn, header);
|
|
|
|
try {
|
|
|
|
return (T) cls.getConstructor(String.class).newInstance(str);
|
|
|
|
}
|
|
|
|
catch (InvocationTargetException e) {
|
|
|
|
throw new ServalDInterfaceException("invalid header field: " + header + ": " + str, e.getTargetException());
|
|
|
|
}
|
|
|
|
catch (Exception e) {
|
|
|
|
throw new ServalDInterfaceException("invalid header field: " + header + ": " + str, e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-27 07:59:57 +00:00
|
|
|
}
|