/vt-crypt/tags/vt-crypt-2.1.1/src/main/java/edu/vt/middleware/crypt/asymmetric/AsymmetricCli.java

http://vt-middleware.googlecode.com/ · Java · 276 lines · 156 code · 36 blank · 84 comment · 15 complexity · 5f0a4a70dbcaa38615fa364c26a92411 MD5 · raw file

  1. /*
  2. $Id$
  3. Copyright (C) 2007-2010 Virginia Tech.
  4. All rights reserved.
  5. SEE LICENSE FOR MORE INFORMATION
  6. Author: Middleware Services
  7. Email: middleware@vt.edu
  8. Version: $Revision$
  9. Updated: $Date$
  10. */
  11. package edu.vt.middleware.crypt.asymmetric;
  12. import java.io.File;
  13. import java.io.IOException;
  14. import java.security.KeyPair;
  15. import java.security.PrivateKey;
  16. import java.security.PublicKey;
  17. import edu.vt.middleware.crypt.AbstractEncryptionCli;
  18. import edu.vt.middleware.crypt.CryptException;
  19. import edu.vt.middleware.crypt.util.CryptReader;
  20. import edu.vt.middleware.crypt.util.CryptWriter;
  21. import org.apache.commons.cli.CommandLine;
  22. import org.apache.commons.cli.Option;
  23. /**
  24. * Command line interface for asymmetric encryption operations.
  25. *
  26. * @author Middleware Services
  27. * @version $Revision: 12 $
  28. */
  29. public class AsymmetricCli extends AbstractEncryptionCli
  30. {
  31. /** Generate key pair option. */
  32. protected static final String OPT_GENKEYPAIR = "genkeys";
  33. /** Output path of private key for keypair generation option. */
  34. protected static final String OPT_PRIVKEYPATH = "privkey";
  35. /** Name of operation provided by this class. */
  36. private static final String COMMAND_NAME = "pkc";
  37. /**
  38. * CLI entry point method.
  39. *
  40. * @param args Command line arguments.
  41. */
  42. public static void main(final String[] args)
  43. {
  44. new AsymmetricCli().performAction(args);
  45. }
  46. /** {@inheritDoc} */
  47. protected void initOptions()
  48. {
  49. super.initOptions();
  50. final Option genKeyPair = new Option(
  51. OPT_GENKEYPAIR,
  52. true,
  53. "generate key pair of size; " +
  54. "public key written to -out path, " +
  55. "private key written to -privkey path");
  56. genKeyPair.setArgName("bitsize");
  57. genKeyPair.setOptionalArg(false);
  58. final Option privKeyPath = new Option(
  59. OPT_PRIVKEYPATH,
  60. true,
  61. "output path of generated private key");
  62. privKeyPath.setArgName("filepath");
  63. final Option encrypt = new Option(
  64. OPT_ENCRYPT,
  65. true,
  66. "encrypt with X.509 DER-encoded public key or certificate");
  67. encrypt.setArgName("pubkeypath");
  68. encrypt.setOptionalArg(false);
  69. final Option decrypt = new Option(
  70. OPT_DECRYPT,
  71. true,
  72. "decrypt with PKCS#8 DER-encoded private key");
  73. decrypt.setArgName("privkeypath");
  74. decrypt.setOptionalArg(false);
  75. options.addOption(genKeyPair);
  76. options.addOption(privKeyPath);
  77. options.addOption(encrypt);
  78. options.addOption(decrypt);
  79. }
  80. /** {@inheritDoc} */
  81. protected void dispatch(final CommandLine line)
  82. throws Exception
  83. {
  84. if (line.hasOption(OPT_ENCRYPT)) {
  85. encrypt(line);
  86. } else if (line.hasOption(OPT_DECRYPT)) {
  87. decrypt(line);
  88. } else if (line.hasOption(OPT_GENKEYPAIR)) {
  89. genKeyPair(line);
  90. } else {
  91. printHelp();
  92. }
  93. }
  94. /**
  95. * Creates a new asymmetric encryption algorithm instance based on CLI
  96. * options.
  97. *
  98. * @param line Parsed command line arguments container.
  99. *
  100. * @return New instance of an initialized asymmetric algorithm.
  101. */
  102. protected AsymmetricAlgorithm newAlgorithm(final CommandLine line)
  103. {
  104. if (!line.hasOption(OPT_CIPHER)) {
  105. throw new IllegalArgumentException("cipher is required.");
  106. }
  107. return AsymmetricAlgorithm.newInstance(line.getOptionValue(OPT_CIPHER));
  108. }
  109. /** {@inheritDoc} */
  110. protected String getCommandName()
  111. {
  112. return COMMAND_NAME;
  113. }
  114. /**
  115. * Perform an encryption operation using data specified on the command line.
  116. *
  117. * @param line Parsed command line arguments container.
  118. *
  119. * @throws Exception On encryption errors.
  120. */
  121. protected void encrypt(final CommandLine line)
  122. throws Exception
  123. {
  124. validateOptions(line);
  125. final AsymmetricAlgorithm alg = newAlgorithm(line);
  126. alg.setKey(readPublicKey(line));
  127. encrypt(alg, getInputStream(line), getOutputStream(line));
  128. }
  129. /**
  130. * Perform a decryption operation using data specified on the command line.
  131. *
  132. * @param line Parsed command line arguments container.
  133. *
  134. * @throws Exception On encryption errors.
  135. */
  136. protected void decrypt(final CommandLine line)
  137. throws Exception
  138. {
  139. validateOptions(line);
  140. final AsymmetricAlgorithm alg = newAlgorithm(line);
  141. alg.setKey(readPrivateKey(line));
  142. decrypt(alg, getInputStream(line), getOutputStream(line));
  143. }
  144. /**
  145. * Generate a new encryption public/private key pair using CLI arguments.
  146. *
  147. * @param line Parsed command line arguments container.
  148. *
  149. * @throws Exception On encryption errors.
  150. */
  151. protected void genKeyPair(final CommandLine line)
  152. throws Exception
  153. {
  154. validateOptions(line);
  155. final AsymmetricAlgorithm alg = newAlgorithm(line);
  156. final int size = Integer.parseInt(line.getOptionValue(OPT_GENKEYPAIR));
  157. System.err.println("Generating " + alg + " key pair of " + size + " bits");
  158. final KeyPair keyPair = alg.generateKeys(size);
  159. final File pubKeyFile = new File(line.getOptionValue(OPT_OUTFILE));
  160. final File privKeyFile = new File(line.getOptionValue(OPT_PRIVKEYPATH));
  161. CryptWriter.writeEncodedKey(keyPair.getPublic(), pubKeyFile);
  162. System.err.println("Wrote X.509 DER-encoded public key to " + pubKeyFile);
  163. CryptWriter.writeEncodedKey(keyPair.getPrivate(), privKeyFile);
  164. System.err.println(
  165. "Wrote PKCS#8 DER-encoded private key to " + privKeyFile);
  166. }
  167. /**
  168. * Creates a public key from a file defined by CLI arguments.
  169. *
  170. * @param line Parsed command line arguments container.
  171. *
  172. * @return Public key used for signature verification.
  173. *
  174. * @throws Exception On IO or key format errors.
  175. */
  176. protected PublicKey readPublicKey(final CommandLine line)
  177. throws Exception
  178. {
  179. PublicKey key = null;
  180. final String alg = line.getOptionValue(OPT_CIPHER);
  181. final File keyFile = new File(line.getOptionValue(OPT_ENCRYPT));
  182. System.err.println("Reading public key from " + keyFile);
  183. try {
  184. if (keyFile.getName().endsWith(PEM_SUFFIX)) {
  185. key = CryptReader.readPemPublicKey(keyFile);
  186. } else {
  187. key = CryptReader.readPublicKey(keyFile, alg);
  188. }
  189. } catch (CryptException e) {
  190. // Maybe the file is an X.509 certificate containing the public key
  191. key = CryptReader.readCertificate(keyFile).getPublicKey();
  192. }
  193. return key;
  194. }
  195. /**
  196. * Creates a private key from a file defined by CLI arguments.
  197. *
  198. * @param line Parsed command line arguments container.
  199. *
  200. * @return Private key.
  201. *
  202. * @throws CryptException On crypto errors.
  203. * @throws IOException On I/O errors.
  204. */
  205. protected PrivateKey readPrivateKey(final CommandLine line)
  206. throws CryptException, IOException
  207. {
  208. final File keyFile = new File(line.getOptionValue(OPT_DECRYPT));
  209. if (keyFile.getName().endsWith(PEM_SUFFIX)) {
  210. return CryptReader.readPemPrivateKey(keyFile, null);
  211. } else {
  212. return
  213. CryptReader.readPrivateKey(keyFile, line.getOptionValue(OPT_CIPHER));
  214. }
  215. }
  216. /**
  217. * Validates the existence of required options for an operation.
  218. *
  219. * @param line Parsed command line arguments container.
  220. */
  221. protected void validateOptions(final CommandLine line)
  222. {
  223. if (!line.hasOption(OPT_CIPHER)) {
  224. throw new IllegalArgumentException("cipher option is required.");
  225. }
  226. if (line.hasOption(OPT_GENKEYPAIR)) {
  227. if (!line.hasOption(OPT_OUTFILE)) {
  228. throw new IllegalArgumentException(
  229. "genkeys operation requires -out for public key output path");
  230. }
  231. if (!line.hasOption(OPT_PRIVKEYPATH)) {
  232. throw new IllegalArgumentException(
  233. "genkeys operation requires -privkey for private key output path");
  234. }
  235. }
  236. }
  237. }