/* * Copyright (c) 2010, Manuel Mausz. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * - Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * - Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * - The names of the authors may not be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS * IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ import java.io.*; import javax.crypto.*; import java.util.Collection; import java.util.Iterator; import org.bouncycastle.util.encoders.Base64; import java.security.GeneralSecurityException; /* * Common static methods used by all applications * * @author Manuel Mausz (0728348) */ class Utils { static class Shutdown extends java.lang.RuntimeException { public Shutdown() { super(); } public Shutdown(String message) { super(message); } } /*==========================================================================*/ private static final String B64 = "a-zA-Z0-9/+"; /*==========================================================================*/ public static String join(final Collection objs, final String delimiter) { if (objs == null || objs.isEmpty()) return ""; Iterator iter = objs.iterator(); StringBuffer buffer = new StringBuffer(iter.next().toString()); while (iter.hasNext()) buffer.append(delimiter).append(iter.next().toString()); return buffer.toString(); } /*--------------------------------------------------------------------------*/ public static long parseHeaderNum(String[] args, int index) { long num = -1; try { if (args.length < index + 1) throw new NumberFormatException(); num = Long.parseLong(args[index]); if (num < 0) throw new NumberFormatException(); } catch(NumberFormatException e) { System.err.println("Error: Invalid network paket from peer:"); System.err.println(join(java.util.Arrays.asList(args), " ")); } return num; } /*------------------------------------------------------------------------*/ public static void sendOutput(EncObjectOutputStream out, String[] msgs) throws IOException { out.writeLine("!output " + msgs.length); for(String msg : msgs) out.writeLine("!" + msg); out.flush(); } /*------------------------------------------------------------------------*/ public static void sendOutput(EncObjectOutputStream out, String msg) throws IOException { String[] tmp = { msg }; sendOutput(out, tmp); } /*------------------------------------------------------------------------*/ public static void sendError(EncObjectOutputStream out, String[] msgs) throws IOException { out.writeLine("!error " + msgs.length); for(String msg : msgs) out.writeLine("!" + msg); out.flush(); } /*------------------------------------------------------------------------*/ public static void sendError(EncObjectOutputStream out, String msg) throws IOException { String[] tmp = { "Error: " + msg }; sendError(out, tmp); } /*==========================================================================*/ static class EncObjectInputStream extends ObjectInputStream { private Cipher cipher = null; private Mac mac = null; public EncObjectInputStream(InputStream in) throws IOException { super(in); } /*------------------------------------------------------------------------*/ public void setCipher(Cipher cipher) { this.cipher = cipher; } /*------------------------------------------------------------------------*/ public void setMAC(Mac mac) { this.mac = mac; } /*------------------------------------------------------------------------*/ private String decode(String msg) { if (msg == null) return null; if ((cipher == null && mac == null) || msg.charAt(0) == '!') return msg; if (mac != null) { msg = new String(Base64.decode(msg)); assert msg.matches("[" + B64 + "]{43}= [\\s[^\\s]]+"); int ix = msg.indexOf(' '); if (ix == -1) { System.err.println("Error: invalid hashed message format:"); System.err.println(msg); return null; } String hasht = msg.substring(0, ix); String msgtmp = msg.substring(ix + 1); String hashc = new String(Base64.encode(mac.doFinal(msgtmp.getBytes()))); if (!hasht.equals(hashc)) { System.err.println("Error: invalid MAC:"); System.err.println(msg); throw new HashError(); //return null; } msg = msgtmp; } if (cipher != null) { try { byte[] decmsg = Base64.decode(msg); return new String(cipher.doFinal(decmsg)); } catch(GeneralSecurityException e) { System.err.println("Error: Unable to decode message: " + e.getMessage()); return null; } } return msg; } /*------------------------------------------------------------------------*/ public String readUTF() throws IOException { return decode(super.readUTF()); } /*------------------------------------------------------------------------*/ @SuppressWarnings("deprecation") public String readLine() throws IOException { return decode(super.readLine()); } } /*==========================================================================*/ static class EncObjectOutputStream extends ObjectOutputStream { private Cipher cipher = null; private Mac mac = null; public EncObjectOutputStream(OutputStream out) throws IOException { super(out); } /*------------------------------------------------------------------------*/ public void setCipher(Cipher cipher) { this.cipher = cipher; } /*------------------------------------------------------------------------*/ public void setMAC(Mac mac) { this.mac = mac; } /*------------------------------------------------------------------------*/ private byte[] encode(byte[] msg) { try { return Base64.encode(cipher.doFinal(msg)); } catch(GeneralSecurityException e) { System.err.println("Error: Unable to encode message: " + e.getMessage()); } return null; } /*------------------------------------------------------------------------*/ public void writeUTF(String msg) throws IOException { if (cipher == null && mac == null) { super.writeUTF(msg); return; } if (mac != null) { byte[] hash = mac.doFinal(msg.getBytes()); msg = new String(Base64.encode(hash)) + " " + msg; assert msg.matches("[" + B64 + "]{43}= [\\s[^\\s]]+"); } if (cipher != null) { super.writeUTF(new String(encode(msg.getBytes()))); return; } String hashedmsg = new String(Base64.encode(msg.getBytes())); super.writeUTF(hashedmsg); return; } /*------------------------------------------------------------------------*/ public void writeLine(String msg) throws IOException { if (cipher == null && mac == null) { super.writeBytes(msg + "\n"); return; } if (mac != null) { byte[] hash = mac.doFinal(msg.getBytes()); msg = new String(Base64.encode(hash)) + " " + msg; assert msg.matches("[" + B64 + "]{43}= [\\s[^\\s]]+"); } if (cipher != null) { super.writeBytes(new String(encode(msg.getBytes())) + "\n"); return; } super.writeBytes(new String(Base64.encode(msg.getBytes())) + "\n"); return; } } /*==========================================================================*/ static class HashError extends java.lang.RuntimeException { public HashError() { super(); } } }