Package org.certificatetransparency.ctlog.serialization

Source Code of org.certificatetransparency.ctlog.serialization.CryptoDataLoader

package org.certificatetransparency.ctlog.serialization;

import com.google.common.base.Joiner;
import com.google.common.io.Files;

import org.apache.commons.codec.binary.Base64;
import org.certificatetransparency.ctlog.UnsupportedCryptoPrimitiveException;

import java.io.BufferedInputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.nio.charset.Charset;
import java.security.KeyFactory;
import java.security.NoSuchAlgorithmException;
import java.security.PublicKey;
import java.security.cert.Certificate;
import java.security.cert.CertificateException;
import java.security.cert.CertificateFactory;
import java.security.spec.InvalidKeySpecException;
import java.security.spec.X509EncodedKeySpec;
import java.util.Arrays;
import java.util.Collection;
import java.util.List;

/**
* Class for reading various crypto structures off disk.
*/
public class CryptoDataLoader {
  /**
   * Returns a list of certificates from an input stream of PEM-encoded certs.
   * @param pemStream input stream with PEM bytes
   * @return A list of certificates in the PEM file.
   */
  static List<Certificate> parseCertificates(InputStream pemStream) {
    CertificateFactory factory;
    try {
      factory = CertificateFactory.getInstance("X.509");
    } catch (CertificateException e) {
      throw new UnsupportedCryptoPrimitiveException("Failure getting X.509 factory", e);
    }

    try {
      Collection<? extends Certificate> certs = factory.generateCertificates(pemStream);
      Certificate[] toReturn = certs.toArray(new Certificate[]{});
      return Arrays.asList(toReturn);
    } catch (CertificateException e) {
      throw new InvalidInputException("Not a valid PEM stream", e);
    }
  }

  /**
   * Parses a PEM-encoded file containing a list of certificates.
   *
   * @param pemCertsFile File to parse.
   * @return A list of certificates from the certificates in the file.
   * @throws FileNotFoundException If the file is not present.
   */
  public static List<Certificate> certificatesFromFile(File pemCertsFile) {
    try {
      return parseCertificates(new BufferedInputStream(new FileInputStream(pemCertsFile)));
    } catch (FileNotFoundException e) {
      throw new InvalidInputException(
          String.format("Could not find certificate chain file %s.", pemCertsFile),
          e);
    }
  }

  static PublicKey parsePublicKey(List<String> pemLines) {
    //TODO(eranm): Filter out non PEM blocks.
    // Are the contents PEM-encoded?
    boolean correctHeader = pemLines.get(0).equals("-----BEGIN PUBLIC KEY-----");
    boolean correctFooter = pemLines.get(pemLines.size() - 1)
        .equals("-----END PUBLIC KEY-----");
    if (!correctHeader || !correctFooter) {
      throw new IllegalArgumentException(
          String.format("Input is not a PEM-encoded key file: " + pemLines));
    }

    // The contents are PEM encoded - first and last lines are header and footer.
    String b64string = Joiner.on("").join(pemLines.subList(1, pemLines.size() - 1));
    // Extract public key
    X509EncodedKeySpec spec = new X509EncodedKeySpec(Base64.decodeBase64(b64string));
    // Note: EC KeyFactory does not exist in openjdk, only Oracle's JDK.
    KeyFactory kf = null;
    try {
      kf = KeyFactory.getInstance("EC");
      return kf.generatePublic(spec);
    } catch (NoSuchAlgorithmException e) {
      // EC is known to be missing from openjdk; Oracle's JDK must be used.
      throw new UnsupportedCryptoPrimitiveException("EC support missing", e);
    } catch (InvalidKeySpecException e) {
      throw new InvalidInputException("Log public key is invalid", e);
    }
  }

  /**
   * Load EC public key from a PEM file.
   * @param pemFile File containing the key.
   * @return Public key represented by this file.
   */
  public static PublicKey keyFromFile(File pemFile) {
    try {
      return parsePublicKey(Files.readLines(pemFile, Charset.defaultCharset()));
    } catch (IOException e) {
      throw new InvalidInputException(
          String.format("Error reading input file %s", pemFile), e);
    }
  }
}
TOP

Related Classes of org.certificatetransparency.ctlog.serialization.CryptoDataLoader

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.