/* * Copyright (C) 2009 Google Inc. All rights reserved. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.google.polo.pairing; import com.google.polo.exception.PoloException; import java.math.BigInteger; import java.security.cert.Certificate; import javax.net.ssl.SSLPeerUnverifiedException; import javax.net.ssl.SSLSession; /** * Utility methods of general usefulness to the Polo library. */ public class PoloUtil { /** * Returns the peer {@link Certificate} for an {@link SSLSession}. * * @throws PoloException if the peer certificate could not be obtained * from the {@link SSLSession}. * @return the {@link Certificate} of the peer */ public static Certificate getPeerCert(SSLSession session) throws PoloException { try { // Peer certificate Certificate[] certs = session.getPeerCertificates(); if (certs == null || certs.length < 1) { throw new PoloException("No peer certificate."); } return certs[0]; } catch (SSLPeerUnverifiedException e) { throw new PoloException(e); } } /** * Return the local {@link Certificate} for an {@link SSLSession}. * * @throws PoloException if the local certificate could not be obtained * from the {@link SSLSession} * @return the {@link Certificate} of the peer */ public static Certificate getLocalCert(SSLSession session) throws PoloException { Certificate[] certs = session.getLocalCertificates(); if (certs == null || certs.length < 1) { throw new PoloException("No local certificate."); } return certs[0]; } /** * Converts an array of bytes to a string of hexadecimal characters. * Leading null bytes are preserved in the output. * <p> * The input byte stream is assumed to be a positive, two's complement * representation of an integer. The return value is the hexadecimal string * representation of this value. * * @param bytes the bytes to convert * @return the string representation */ public static String bytesToHexString(byte[] bytes) { if (bytes == null || bytes.length == 0) { return ""; } BigInteger bigint = new BigInteger(1, bytes); int formatLen = bytes.length * 2; return String.format("%0" + formatLen + "x", bigint); } /** * Converts a string of hex characters to a byte array. * * @param hexstr the string of hex characters * @return a byte array representation */ public static byte[] hexStringToBytes(String hexstr) { if (hexstr == null || hexstr.length() == 0 || (hexstr.length() % 2) != 0) { throw new IllegalArgumentException("Bad input string."); } byte[] result = new byte[hexstr.length() / 2]; for (int i=0; i < result.length; i++) { result[i] = (byte) Integer.parseInt(hexstr.substring(2 * i, 2 * (i + 1)), 16); } return result; } /** * Converts an integer value to the big endian 4-byte representation. */ public static final byte[] intToBigEndianIntBytes(int intVal) { byte[] outBuf = new byte[4]; outBuf[0] = (byte)((intVal >> 24) & 0xff); outBuf[1] = (byte)((intVal >> 16) & 0xff); outBuf[2] = (byte)((intVal >> 8) & 0xff); outBuf[3] = (byte)(intVal & 0xff); return outBuf; } /** * Converts a 4-byte array of bytes to an unsigned long value. */ public static final long intBigEndianBytesToLong(byte[] input) { assert (input.length == 4); long ret = (long)(input[0]) & 0xff; ret <<= 8; ret |= (long)(input[1]) & 0xff; ret <<= 8; ret |= (long)(input[2]) & 0xff; ret <<= 8; ret |= (long)(input[3]) & 0xff; return ret; } }