/src/org/apache/james/mime4j/storage/CipherStorageProvider.java

https://github.com/jca02266/k9mail · Java · 177 lines · 102 code · 25 blank · 50 comment · 6 complexity · beebbd8460dbce69e70cbd23d3e6ba6f MD5 · raw file

  1. /****************************************************************
  2. * Licensed to the Apache Software Foundation (ASF) under one *
  3. * or more contributor license agreements. See the NOTICE file *
  4. * distributed with this work for additional information *
  5. * regarding copyright ownership. The ASF licenses this file *
  6. * to you under the Apache License, Version 2.0 (the *
  7. * "License"); you may not use this file except in compliance *
  8. * with the License. You may obtain a copy of the License at *
  9. * *
  10. * http://www.apache.org/licenses/LICENSE-2.0 *
  11. * *
  12. * Unless required by applicable law or agreed to in writing, *
  13. * software distributed under the License is distributed on an *
  14. * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY *
  15. * KIND, either express or implied. See the License for the *
  16. * specific language governing permissions and limitations *
  17. * under the License. *
  18. ****************************************************************/
  19. package org.apache.james.mime4j.storage;
  20. import java.io.IOException;
  21. import java.io.InputStream;
  22. import java.security.GeneralSecurityException;
  23. import java.security.NoSuchAlgorithmException;
  24. import javax.crypto.Cipher;
  25. import javax.crypto.CipherInputStream;
  26. import javax.crypto.CipherOutputStream;
  27. import javax.crypto.KeyGenerator;
  28. import javax.crypto.spec.SecretKeySpec;
  29. /**
  30. * A {@link StorageProvider} that transparently scrambles and unscrambles the
  31. * data stored by another <code>StorageProvider</code>.
  32. *
  33. * <p>
  34. * Example usage:
  35. *
  36. * <pre>
  37. * StorageProvider mistrusted = new TempFileStorageProvider();
  38. * StorageProvider enciphered = new CipherStorageProvider(mistrusted);
  39. * StorageProvider provider = new ThresholdStorageProvider(enciphered);
  40. * DefaultStorageProvider.setInstance(provider);
  41. * </pre>
  42. */
  43. public class CipherStorageProvider extends AbstractStorageProvider {
  44. private final StorageProvider backend;
  45. private final String algorithm;
  46. private final KeyGenerator keygen;
  47. /**
  48. * Creates a new <code>CipherStorageProvider</code> for the given back-end
  49. * using the Blowfish cipher algorithm.
  50. *
  51. * @param backend
  52. * back-end storage strategy to encrypt.
  53. */
  54. public CipherStorageProvider(StorageProvider backend) {
  55. this(backend, "Blowfish");
  56. }
  57. /**
  58. * Creates a new <code>CipherStorageProvider</code> for the given back-end
  59. * and cipher algorithm.
  60. *
  61. * @param backend
  62. * back-end storage strategy to encrypt.
  63. * @param algorithm
  64. * the name of the symmetric block cipher algorithm such as
  65. * "Blowfish", "AES" or "RC2".
  66. */
  67. public CipherStorageProvider(StorageProvider backend, String algorithm) {
  68. if (backend == null)
  69. throw new IllegalArgumentException();
  70. try {
  71. this.backend = backend;
  72. this.algorithm = algorithm;
  73. this.keygen = KeyGenerator.getInstance(algorithm);
  74. } catch (NoSuchAlgorithmException e) {
  75. throw new IllegalArgumentException(e);
  76. }
  77. }
  78. public StorageOutputStream createStorageOutputStream() throws IOException {
  79. SecretKeySpec skeySpec = getSecretKeySpec();
  80. return new CipherStorageOutputStream(backend
  81. .createStorageOutputStream(), algorithm, skeySpec);
  82. }
  83. private SecretKeySpec getSecretKeySpec() {
  84. byte[] raw = keygen.generateKey().getEncoded();
  85. return new SecretKeySpec(raw, algorithm);
  86. }
  87. private static final class CipherStorageOutputStream extends
  88. StorageOutputStream {
  89. private final StorageOutputStream storageOut;
  90. private final String algorithm;
  91. private final SecretKeySpec skeySpec;
  92. private final CipherOutputStream cipherOut;
  93. public CipherStorageOutputStream(StorageOutputStream out,
  94. String algorithm, SecretKeySpec skeySpec) throws IOException {
  95. try {
  96. this.storageOut = out;
  97. this.algorithm = algorithm;
  98. this.skeySpec = skeySpec;
  99. Cipher cipher = Cipher.getInstance(algorithm);
  100. cipher.init(Cipher.ENCRYPT_MODE, skeySpec);
  101. this.cipherOut = new CipherOutputStream(out, cipher);
  102. } catch (GeneralSecurityException e) {
  103. throw (IOException) new IOException().initCause(e);
  104. }
  105. }
  106. @Override
  107. public void close() throws IOException {
  108. super.close();
  109. cipherOut.close();
  110. }
  111. @Override
  112. protected void write0(byte[] buffer, int offset, int length)
  113. throws IOException {
  114. cipherOut.write(buffer, offset, length);
  115. }
  116. @Override
  117. protected Storage toStorage0() throws IOException {
  118. // cipherOut has already been closed because toStorage calls close
  119. Storage encrypted = storageOut.toStorage();
  120. return new CipherStorage(encrypted, algorithm, skeySpec);
  121. }
  122. }
  123. private static final class CipherStorage implements Storage {
  124. private Storage encrypted;
  125. private final String algorithm;
  126. private final SecretKeySpec skeySpec;
  127. public CipherStorage(Storage encrypted, String algorithm,
  128. SecretKeySpec skeySpec) {
  129. this.encrypted = encrypted;
  130. this.algorithm = algorithm;
  131. this.skeySpec = skeySpec;
  132. }
  133. public void delete() {
  134. if (encrypted != null) {
  135. encrypted.delete();
  136. encrypted = null;
  137. }
  138. }
  139. public InputStream getInputStream() throws IOException {
  140. if (encrypted == null)
  141. throw new IllegalStateException("storage has been deleted");
  142. try {
  143. Cipher cipher = Cipher.getInstance(algorithm);
  144. cipher.init(Cipher.DECRYPT_MODE, skeySpec);
  145. InputStream in = encrypted.getInputStream();
  146. return new CipherInputStream(in, cipher);
  147. } catch (GeneralSecurityException e) {
  148. throw (IOException) new IOException().initCause(e);
  149. }
  150. }
  151. }
  152. }