001    /*
002     * Cumulus4j - Securing your data in the cloud - http://cumulus4j.org
003     * Copyright (C) 2011 NightLabs Consulting GmbH
004     *
005     * This program is free software: you can redistribute it and/or modify
006     * it under the terms of the GNU Affero General Public License as
007     * published by the Free Software Foundation, either version 3 of the
008     * License, or (at your option) any later version.
009     *
010     * This program is distributed in the hope that it will be useful,
011     * but WITHOUT ANY WARRANTY; without even the implied warranty of
012     * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
013     * GNU Affero General Public License for more details.
014     *
015     * You should have received a copy of the GNU Affero General Public License
016     * along with this program.  If not, see <http://www.gnu.org/licenses/>.
017     */
018    package org.cumulus4j.keystore;
019    
020    import java.io.DataOutputStream;
021    import java.io.File;
022    import java.io.FileInputStream;
023    import java.io.FileNotFoundException;
024    import java.io.FileOutputStream;
025    import java.io.IOException;
026    import java.io.OutputStream;
027    import java.lang.ref.WeakReference;
028    import java.security.GeneralSecurityException;
029    import java.security.SecureRandom;
030    import java.security.spec.KeySpec;
031    import java.util.ArrayList;
032    import java.util.Arrays;
033    import java.util.Collections;
034    import java.util.Date;
035    import java.util.HashMap;
036    import java.util.LinkedList;
037    import java.util.List;
038    import java.util.Map;
039    import java.util.Set;
040    import java.util.SortedSet;
041    import java.util.Timer;
042    import java.util.TimerTask;
043    import java.util.TreeSet;
044    import java.util.UUID;
045    
046    import javax.crypto.SecretKey;
047    import javax.crypto.SecretKeyFactory;
048    import javax.crypto.spec.PBEKeySpec;
049    
050    import org.bouncycastle.crypto.CryptoException;
051    import org.bouncycastle.crypto.params.KeyParameter;
052    import org.bouncycastle.crypto.params.ParametersWithIV;
053    import org.cumulus4j.crypto.Cipher;
054    import org.cumulus4j.crypto.CipherOperationMode;
055    import org.cumulus4j.crypto.CryptoRegistry;
056    import org.cumulus4j.keystore.prop.LongProperty;
057    import org.cumulus4j.keystore.prop.Property;
058    import org.slf4j.Logger;
059    import org.slf4j.LoggerFactory;
060    
061    /**
062     * <p>
063     * <code>KeyStore</code> is a storage facility for cryptographic keys.
064     * </p>
065     * <p>
066     * An instance of <code>KeyStore</code> manages a file in the local file system, in which it stores
067     * the keys used by the Cumulus4j-DataNucleus-plug-in in an encrypted form. All data written to the
068     * file is encrypted, hence plain data never touches the local file system (except for
069     * <a target="_blank" href="http://en.wikipedia.org/wiki/Swap_space">swapping</a>!).
070     * </p>
071     * <p>
072     * For every read/write operation, the <code>KeyStore</code> requires a user to authenticate via a
073     * user-name and a password. The password is used to encrypt/decrypt an internally used master-key
074     * which is then used to encrypt/decrypt the actual keys used by the Cumulus4j-DataNucleus-plug-in.
075     * Due to this internal master key, a user can be added or deleted and a user's password can be
076     * changed without the need of decrypting and encrypting all the contents of the KeyStore.
077     * </p>
078     * <p>
079     * By default, a <code>KeyStore</code> {@link #generateKey(String, char[]) generates keys} with a size
080     * of 256 bit. This can be controlled, however, by specifying the system property
081     * {@value #SYSTEM_PROPERTY_KEY_SIZE} (e.g. passing the argument "-Dcumulus4j.KeyStore.keySize=128"
082     * to the <code>java</code> command line will switch to 128-bit-keys).
083     * </p>
084     * <p>
085     * <b>Important:</b> As the master key is generated when the first
086     * {@link #createUser(String, char[], String, char[]) user is created} and is then not changed anymore, you must therefore
087     * specify the desired key-size already at the moment when you initialise the key store (i.e. create the first user). If
088     * you change the key-size later, it will affect only those keys that are created later.
089     * </p>
090     * <p>
091     * Note, that the "Java Cryptography Extension (JCE) Unlimited Strength Jurisdiction Policy Files" does not
092     * need to be installed for very strong cryptography, because we don't use the JCE (see {@link Cipher}).
093     * </p>
094     * <h3>File format of the key store file (version 1)</h3>
095     * <p>
096     * <table border="1" width="100%">
097     *      <tbody>
098     *      <tr>
099     *              <td align="right" valign="top"><b>Bytes</b></td><td valign="top"><b>Descrition</b></td>
100     *      </tr>
101     *      <tr>
102     *              <td align="right" valign="top">17</td><td valign="top">Header "Cumulus4jKeyStore" (ASCII encoded)</td>
103     *      </tr>
104     *      <tr>
105     *              <td align="right" valign="top">4</td><td valign="top">int: File version</td>
106     *      </tr>
107     *      <tr>
108     *              <td align="right" valign="top">4</td><td valign="top">int: Number of entries in 'Block A' to follow.</td>
109     *      </tr>
110     *      <tr>
111     *              <td colspan="2">
112     *              <table bgcolor="#F0F0F0" border="1" width="100%">
113     *                      <tbody>
114     *                      <tr><td bgcolor="#D0D0D0" colspan="2"><b>Block A: String constants</b></td></tr>
115     *                      <tr>
116     *                              <td colspan="2">
117     * In order to reduce the file size (and thus increase the write speed), various
118     * strings like encryption algorithm, checksum algorithm and the like are not written
119     * again and again for every key, but instead only once here. In every key, these
120     * Strings are then referenced instead by their position-index (zero-based).
121     *                              </td>
122     *                      </tr>
123     *
124     *                      <tr>
125     *                              <td align="right" valign="top"><b>Bytes</b></td><td valign="top"><b>Descrition</b></td>
126     *                      </tr>
127     *                      <tr>
128     *                              <td align="right" valign="top">2</td><td valign="top">short <i>len</i>: Number of bytes to follow (written by {@link DataOutputStream#writeUTF(String)}).</td>
129     *                      </tr>
130     *                      <tr>
131     *                              <td align="right" valign="top"><i>len</i></td><td valign="top">String: Constant&apos;s value (written by {@link DataOutputStream#writeUTF(String)}).</td>
132     *                      </tr>
133     *                      </tbody>
134     *              </table>
135     *              </td>
136     *      </tr>
137     *
138     *
139     *      <tr>
140     *              <td align="right" valign="top">4</td><td valign="top">int: Number of entries in 'Block B' to follow.</td>
141     *      </tr>
142     *  <tr>
143     *              <td colspan="2">
144     *              <table bgcolor="#F0F0F0" border="1" width="100%">
145     *                      <tbody>
146     *                      <tr><td bgcolor="#D0D0D0" colspan="2"><b>Block B: User-key-map</b></td></tr>
147     *
148     *                      <tr>
149     *                              <td colspan="2">
150     *                                      For every user, the master-key is stored encrypted with the user's password in this block.
151     *                              </td>
152     *                      </tr>
153     *
154     *                      <tr>
155     *                              <td align="right" valign="top"><b>Bytes</b></td><td valign="top"><b>Descrition</b></td>
156     *                      </tr>
157     *
158     *                      <tr>
159     *                              <td align="right" valign="top">2</td><td valign="top">short <i>len1</i>: User name: Number of bytes to follow (written by {@link DataOutputStream#writeUTF(String)}).</td>
160     *                      </tr>
161     *                      <tr>
162     *                              <td align="right" valign="top"><i>len1</i></td><td valign="top">String: User name (written by {@link DataOutputStream#writeUTF(String)}).</td>
163     *                      </tr>
164     *
165     *                      <tr>
166     *                              <td align="right" valign="top">4</td><td valign="top">int: Key size for the password-based key (in bits! i.e. usually 128 or 256).</td>
167     *                      </tr>
168     *                      <tr>
169     *                              <td align="right" valign="top">4</td><td valign="top">int: Iteration count for the password-based key.</td>
170     *                      </tr>
171     *                      <tr>
172     *                              <td align="right" valign="top">4</td><td valign="top">int: Reference to the name of the key-generator-algorithm for creating the password-based key (index in the list of 'Block A').</td>
173     *                      </tr>
174     *
175     *                      <tr>
176     *                              <td align="right" valign="top">2</td><td valign="top">UNSIGNED short <i>len2</i>: Salt: Number of bytes to follow (written by {@link KeyStoreUtil#writeByteArrayWithShortLengthHeader(DataOutputStream, byte[])}).</td>
177     *                      </tr>
178     *                      <tr>
179     *                              <td align="right" valign="top"><i>len2</i></td><td valign="top">byte[]: Salt to be used when generating the password-based key (written by {@link KeyStoreUtil#writeByteArrayWithShortLengthHeader(DataOutputStream, byte[])}).</td>
180     *                      </tr>
181     *
182     *                      <!-- BEGIN written by {@link AbstractEncryptedData#write(DataOutputStream, Map)} -->
183     *                              <tr>
184     *                                      <td align="right" valign="top">4</td><td valign="top">int: Reference to the name of the encryption algorithm used to encrypt this record's data (index in the list of 'Block A').</td>
185     *                              </tr>
186     *
187     *                              <tr>
188     *                                      <td align="right" valign="top">2</td><td valign="top">UNSIGNED short <i>lenIV</i>: IV: Number of bytes to follow (written by {@link KeyStoreUtil#writeByteArrayWithShortLengthHeader(DataOutputStream, byte[])}).</td>
189     *                              </tr>
190     *                              <tr>
191     *                                      <td align="right" valign="top"><i>lenIV</i></td><td valign="top">byte[]: The actual IV (initialisation vector) used to encrypt the key's data (written by {@link KeyStoreUtil#writeByteArrayWithShortLengthHeader(DataOutputStream, byte[])}).</td>
192     *                              </tr>
193     *
194     *                              <tr>
195     *                                      <td align="right" valign="top">4</td><td valign="top">int: Reference to the name of the <a target="_blank" href="http://en.wikipedia.org/wiki/Message_authentication_code">MAC</a> algorithm used to authenticate this record's data (index in the list of 'Block A').</td>
196     *                              </tr>
197     *
198     *                              <tr>
199     *                                      <td align="right" valign="top">2</td><td valign="top">short <i>lenMACKey</i>: MAC key: Number of bytes in the MAC's key.</td>
200     *                              </tr>
201     *                              <tr>
202     *                                      <td align="right" valign="top">2</td><td valign="top">short <i>lenMACIV</i>: MAC IV: Number of bytes in the MAC's IV.</td>
203     *                              </tr>
204     *                              <tr>
205     *                                      <td align="right" valign="top">2</td><td valign="top">short <i>lenMAC</i>: MAC: Number of bytes in the MAC.</td>
206     *                              </tr>
207     *
208     *                              <tr>
209     *                                      <td colspan="2">
210     *                                              <table bgcolor="#E0E0E0" border="1" width="100%">
211     *                                                      <tbody>
212     *                                                              <tr><td bgcolor="#C0C0C0" colspan="2"><b>ENCRYPTED</b></td></tr>
213     *                                                              <tr>
214     *                                                                      <td align="right" valign="top"><i>lenMACKey</i></td><td valign="top">MAC key: The actual MAC's key (random).</td>
215     *                                                              </tr>
216     *                                                              <tr>
217     *                                                                      <td align="right" valign="top"><i>lenMACIV</i></td><td valign="top">MAC IV: The actual MAC's IV (random).</td>
218     *                                                              </tr>
219     *                                                              <tr>
220     *                                                                      <td align="right" valign="top"><i>all until MAC</i></td><td valign="top">The actual data (payload).</td>
221     *                                                              </tr>
222     *                                                              <tr>
223     *                                                                      <td align="right" valign="top"><i>lenMAC</i></td><td valign="top">MAC: The actual MAC.</td>
224     *                                                              </tr>
225     *                                                      </tbody>
226     *                                              </table>
227     *                                      </td>
228     *                              </tr>
229     *
230     *                      <!-- END written by {@link AbstractEncryptedData#write(DataOutputStream, Map)} -->
231     *
232     *                      </tbody>
233     *              </table>
234     *              </td>
235     *      </tr>
236     *
237     *
238     *      <tr>
239     *              <td align="right" valign="top">4</td><td valign="top">int: Number of entries in 'Block C' to follow.</td>
240     *      </tr>
241     *      <tr>
242     *              <td colspan="2">
243     *              <table bgcolor="#F0F0F0" border="1" width="100%">
244     *                      <tbody>
245     *                      <tr><td bgcolor="#D0D0D0" colspan="2"><b>Block C: Key-ID-key-map</b></td></tr>
246     *
247     *                      <tr>
248     *                              <td colspan="2">
249     *                                      This block contains the actual keys. Every key is encrypted with the master-key.
250     *                              </td>
251     *                      </tr>
252     *
253     *                      <tr>
254     *                              <td align="right" valign="top"><b>Bytes</b></td><td valign="top"><b>Descrition</b></td>
255     *                      </tr>
256     *
257     *                      <tr>
258     *                              <td align="right" valign="top">8</td><td valign="top">long: Key identifier.</td>
259     *                      </tr>
260     *
261     *                      <!-- BEGIN written by {@link AbstractEncryptedData#write(DataOutputStream, Map)} -->
262     *                              <tr>
263     *                                      <td align="right" valign="top">4</td><td valign="top">int: Reference to the name of the encryption algorithm used to encrypt this record's data (index in the list of 'Block A').</td>
264     *                              </tr>
265     *
266     *                              <tr>
267     *                                      <td align="right" valign="top">2</td><td valign="top">UNSIGNED short <i>lenIV</i>: IV: Number of bytes to follow (written by {@link KeyStoreUtil#writeByteArrayWithShortLengthHeader(DataOutputStream, byte[])}).</td>
268     *                              </tr>
269     *                              <tr>
270     *                                      <td align="right" valign="top"><i>lenIV</i></td><td valign="top">byte[]: The actual IV (initialisation vector) used to encrypt the key's data (written by {@link KeyStoreUtil#writeByteArrayWithShortLengthHeader(DataOutputStream, byte[])}).</td>
271     *                              </tr>
272     *
273     *                              <tr>
274     *                                      <td align="right" valign="top">4</td><td valign="top">int: Reference to the name of the MAC algorithm used to authenticate this record's data (index in the list of 'Block A').</td>
275     *                              </tr>
276     *
277     *                              <tr>
278     *                                      <td align="right" valign="top">2</td><td valign="top">short <i>lenMACKey</i>: MAC key: Number of bytes in the MAC's key.</td>
279     *                              </tr>
280     *                              <tr>
281     *                                      <td align="right" valign="top">2</td><td valign="top">short <i>lenMACIV</i>: MAC IV: Number of bytes in the MAC's IV.</td>
282     *                              </tr>
283     *                              <tr>
284     *                                      <td align="right" valign="top">2</td><td valign="top">short <i>lenMAC</i>: MAC: Number of bytes in the MAC.</td>
285     *                              </tr>
286     *
287     *                              <tr>
288     *                                      <td colspan="2">
289     *                                              <table bgcolor="#E0E0E0" border="1" width="100%">
290     *                                                      <tbody>
291     *                                                              <tr><td bgcolor="#C0C0C0" colspan="2"><b>ENCRYPTED</b></td></tr>
292     *                                                              <tr>
293     *                                                                      <td align="right" valign="top"><i>lenMACKey</i></td><td valign="top">MAC key: The actual MAC's key (random).</td>
294     *                                                              </tr>
295     *                                                              <tr>
296     *                                                                      <td align="right" valign="top"><i>lenMACIV</i></td><td valign="top">MAC IV: The actual MAC's IV (random).</td>
297     *                                                              </tr>
298     *                                                              <tr>
299     *                                                                      <td align="right" valign="top"><i>all until MAC</i></td><td valign="top">The actual data (payload).</td>
300     *                                                              </tr>
301     *                                                              <tr>
302     *                                                                      <td align="right" valign="top"><i>lenMAC</i></td><td valign="top">MAC: The actual MAC.</td>
303     *                                                              </tr>
304     *                                                      </tbody>
305     *                                              </table>
306     *                                      </td>
307     *                              </tr>
308     *
309     *                      <!-- END written by {@link AbstractEncryptedData#write(DataOutputStream, Map)} -->
310     *
311     *                      </tbody>
312     *              </table>
313     *              </td>
314     *      </tr>
315     *
316     *
317     *      <tr>
318     *              <td align="right" valign="top">4</td><td valign="top">int: Number of entries in 'Block D' to follow.</td>
319     *      </tr>
320     *      <tr>
321     *              <td colspan="2">
322     *              <table bgcolor="#F0F0F0" border="1" width="100%">
323     *                      <tbody>
324     *                      <tr><td bgcolor="#D0D0D0" colspan="2"><b>Block D: Properties</b></td></tr>
325     *                      <tr>
326     *                              <td colspan="2">
327     * See {@link Property} for details about what this block is used for.
328     *                              </td>
329     *                      </tr>
330     *                      <tr>
331     *                              <td align="right" valign="top"><b>Bytes</b></td><td valign="top"><b>Descrition</b></td>
332     *                      </tr>
333     *
334     *                      <tr>
335     *                              <td align="right" valign="top">2</td><td valign="top">short <i>len1</i>: Property name: Number of bytes to follow (written by {@link DataOutputStream#writeUTF(String)}).</td>
336     *                      </tr>
337     *                      <tr>
338     *                              <td align="right" valign="top"><i>len1</i></td><td valign="top">String: Property name (written by {@link DataOutputStream#writeUTF(String)}).</td>
339     *                      </tr>
340     *
341     *                      <tr>
342     *                              <td align="right" valign="top">4</td><td valign="top">int: Reference to the fully qualified class name of the {@link Property} (index in the list of 'Block A').</td>
343     *                      </tr>
344     *
345     *                      <!-- BEGIN written by {@link AbstractEncryptedData#write(DataOutputStream, Map)} -->
346     *                              <tr>
347     *                                      <td align="right" valign="top">4</td><td valign="top">int: Reference to the name of the encryption algorithm used to encrypt this record's data (index in the list of 'Block A').</td>
348     *                              </tr>
349     *
350     *                              <tr>
351     *                                      <td align="right" valign="top">2</td><td valign="top">UNSIGNED short <i>lenIV</i>: IV: Number of bytes to follow (written by {@link KeyStoreUtil#writeByteArrayWithShortLengthHeader(DataOutputStream, byte[])}).</td>
352     *                              </tr>
353     *                              <tr>
354     *                                      <td align="right" valign="top"><i>lenIV</i></td><td valign="top">byte[]: The actual IV (initialisation vector) used to encrypt the key's data (written by {@link KeyStoreUtil#writeByteArrayWithShortLengthHeader(DataOutputStream, byte[])}).</td>
355     *                              </tr>
356     *
357     *                              <tr>
358     *                                      <td align="right" valign="top">4</td><td valign="top">int: Reference to the name of the MAC algorithm used to authenticate this record's data (index in the list of 'Block A').</td>
359     *                              </tr>
360     *
361     *                              <tr>
362     *                                      <td align="right" valign="top">2</td><td valign="top">short <i>lenMACKey</i>: MAC key: Number of bytes in the MAC's key.</td>
363     *                              </tr>
364     *                              <tr>
365     *                                      <td align="right" valign="top">2</td><td valign="top">short <i>lenMACIV</i>: MAC IV: Number of bytes in the MAC's IV.</td>
366     *                              </tr>
367     *                              <tr>
368     *                                      <td align="right" valign="top">2</td><td valign="top">short <i>lenMAC</i>: MAC: Number of bytes in the MAC.</td>
369     *                              </tr>
370     *
371     *                              <tr>
372     *                                      <td colspan="2">
373     *                                              <table bgcolor="#E0E0E0" border="1" width="100%">
374     *                                                      <tbody>
375     *                                                              <tr><td bgcolor="#C0C0C0" colspan="2"><b>ENCRYPTED</b></td></tr>
376     *                                                              <tr>
377     *                                                                      <td align="right" valign="top"><i>lenMACKey</i></td><td valign="top">MAC key: The actual MAC's key (random).</td>
378     *                                                              </tr>
379     *                                                              <tr>
380     *                                                                      <td align="right" valign="top"><i>lenMACIV</i></td><td valign="top">MAC IV: The actual MAC's IV (random).</td>
381     *                                                              </tr>
382     *                                                              <tr>
383     *                                                                      <td align="right" valign="top"><i>all until MAC</i></td><td valign="top">The actual data (payload).</td>
384     *                                                              </tr>
385     *                                                              <tr>
386     *                                                                      <td align="right" valign="top"><i>lenMAC</i></td><td valign="top">MAC: The actual MAC.</td>
387     *                                                              </tr>
388     *                                                      </tbody>
389     *                                              </table>
390     *                                      </td>
391     *                              </tr>
392     *
393     *                      <!-- END written by {@link AbstractEncryptedData#write(DataOutputStream, Map)} -->
394     *
395     *                      <tr>
396     *                              <td align="right" valign="top">20</td><td valign="top">SHA1 checksum over the complete file except for the header "Cumulus4jKeyStore", i.e. from the file version at byte offset 17 (including) till here (excluding).</td>
397     *                      </tr>
398     *                      </tbody>
399     *              </table>
400     *              </td>
401     *      </tr>
402     *
403     *      </tbody>
404     * </table>
405     * </p>
406     *
407     * @author Marco หงุ่ยตระกูล-Schulze - marco at nightlabs dot de
408     */
409    public class KeyStore
410    {
411            static final Logger logger = LoggerFactory.getLogger(KeyStore.class);
412    
413    //      private static final BouncyCastleProvider bouncyCastleProvider = new BouncyCastleProvider();
414    //      static {
415    //              Security.insertProviderAt(bouncyCastleProvider, 2);
416    //
417    //              KeyGenerator kg;
418    //              try {
419    //                      kg = KeyGenerator.getInstance("AES");
420    //              } catch (NoSuchAlgorithmException e) {
421    //                      logger.warn("KeyGenerator.getInstance(\"AES\") failed: " + e, e);
422    //                      kg = null;
423    //              }
424    //
425    //              if (kg == null || kg.getProvider() != bouncyCastleProvider)
426    //                      logger.warn("BouncyCastleProvider was NOT registered!!!");
427    //      }
428    
429            /**
430             * <p>
431             * System property to control the size of the keys {@link #generateKey(String, char[]) generated}. This
432             * includes not only the actual keys for the main encryption/decryption (in the database), but also the
433             * master key used to protect the file managed by the <code>KeyStore</code>.
434             * </p>
435             * <p>
436             * By default (if the system property {@value #SYSTEM_PROPERTY_KEY_SIZE} is not specified), keys will have a size of 256 bit.
437             * </p>
438             * <p>
439             * Note, that specifying the system property does not change any old keys - only new keys are generated
440             * with the currently active key size. Therefore, if you want to ensure that the internal master key is
441             * only 128 bit long, you have to make sure that the proper key size is specified when the first
442             * {@link #createUser(String, char[], String, char[]) user is created}!
443             * </p>
444             */
445            public static final String SYSTEM_PROPERTY_KEY_SIZE = "cumulus4j.KeyStore" + ".keySize";
446    
447            /**
448             * <p>
449             * System property to control the encryption algorithm that is used to encrypt data within the key-store. Whenever a new user is
450             * created or a new key is generated, data has to be encrypted (note that the encryption does not happen directly
451             * before data is written to the file, but already most data in memory is encrypted!).
452             * </p>
453             * <p>
454             * By default (if the system property {@value #SYSTEM_PROPERTY_ENCRYPTION_ALGORITHM} is not specified),
455             * "Twofish/GCM/NoPadding" is used. For example, to switch to "AES/CFB/NoPadding", you'd have
456             * to specify the command line argument "-Dcumulus4j.KeyStore.encryptionAlgorithm=AES/CFB/NoPadding".
457             * </p>
458             * <p>
459             * See <a target="_blank" href="http://cumulus4j.org/1.0.2/documentation/supported-algorithms.html">this document</a>
460             * for further information about what values are supported.
461             * </p>
462             * <p>
463             * <b>Important:</b> The default MAC algorithm is "NONE", which is a very bad choice for most encryption algorithms!
464             * Therefore, you must change the MAC algorithm via the system property {@value #SYSTEM_PROPERTY_MAC_ALGORITHM}
465             * if you change the encryption algorithm!
466             * </p>
467             */
468            public static final String SYSTEM_PROPERTY_ENCRYPTION_ALGORITHM = "cumulus4j.KeyStore" + ".encryptionAlgorithm";
469    
470            /**
471             * <p>
472             * System property to control the <a target="_blank" href="http://en.wikipedia.org/wiki/Message_authentication_code">MAC</a>
473             * algorithm that is used to protect the data within the key-store against manipulation.
474             * </p>
475             * <p>
476             * Whenever data is encrypted, this MAC algorithm is used to calculate a MAC over the original plain-text-data.
477             * The MAC is then stored together with the plain-text-data within the encrypted area.
478             * When data is decrypted, the MAC is calculated again over the decrypted plain-text-data and compared to the
479             * original MAC in order to make sure (1) that data was correctly decrypted [i.e. the password provided by the user
480             * is correct] and (2) that the data in the key-store was not manipulated by an attacker.
481             * </p>
482             * <p>
483             * The MAC algorithm used during encryption is stored in the encryption-record's meta-data in order
484             * to use the correct algorithm during decryption, no matter what current MAC algorithm is configured.
485             * Therefore, you can safely change this setting at any time - it will affect future encryption
486             * operations, only.
487             * </p>
488             * <p>
489             * Some block cipher modes (e.g. <a target="_blank" href="http://en.wikipedia.org/wiki/Galois/Counter_Mode">GCM</a>) already include authentication
490             * and therefore no MAC is necessary. In this case, you can specify the MAC algorithm {@value #MAC_ALGORITHM_NONE}.
491             * </p>
492             * <p>
493             * <b>Important:</b> If you specify the MAC algorithm "NONE" and use an encryption algorithm without
494             * authentication, the key store will not be able to detect a wrong password and instead return
495             * corrupt data!!! Be VERY careful with the MAC algorithm "NONE"!!!
496             * </p>
497             * <p>
498             * The default value (used when this system property is not specified) is "NONE", because the default
499             * encryption algorithm is "Twofish/GCM/NoPadding", which (due to "GCM") does not require an additional
500             * MAC.
501             * </p>
502             */
503            public static final String SYSTEM_PROPERTY_MAC_ALGORITHM = "cumulus4j.KeyStore" + ".macAlgorithm";
504    
505            /**
506             * <p>
507             * Constant for deactivating the <a target="_blank" href="http://en.wikipedia.org/wiki/Message_authentication_code">MAC</a>.
508             * </p>
509             * <p>
510             * <b>Important: Deactivating the MAC is dangerous!</b> Choose this value only, if you are absolutely
511             * sure that your {@link #SYSTEM_PROPERTY_ENCRYPTION_ALGORITHM encryption algorithm} already
512             * provides authentication - like <a target="_blank" href="http://en.wikipedia.org/wiki/Galois/Counter_Mode">GCM</a>
513             * does for example.
514             * </p>
515             * @see #SYSTEM_PROPERTY_MAC_ALGORITHM
516             */
517            public static final String MAC_ALGORITHM_NONE = "NONE";
518    
519            private static final String KEY_STORE_PROPERTY_NAME_NEXT_KEY_ID = "nextKeyID";
520    
521            private SecureRandom secureRandom = new SecureRandom();
522    
523            private static Timer expireCacheEntryTimer = new Timer(true);
524    
525            private TimerTask expireCacheEntryTimerTask = new ExipreCacheEntryTimerTask(this);
526    
527            private KeyStoreData keyStoreData = new KeyStoreData();
528    
529            private static class ExipreCacheEntryTimerTask extends TimerTask
530            {
531                    private static final Logger logger = LoggerFactory.getLogger(ExipreCacheEntryTimerTask.class);
532    
533                    private WeakReference<KeyStore> keyStoreRef;
534    
535                    public ExipreCacheEntryTimerTask(KeyStore keyStore)
536                    {
537                            if (keyStore == null)
538                                    throw new IllegalArgumentException("keyStore == null");
539    
540                            this.keyStoreRef = new WeakReference<KeyStore>(keyStore);
541                    }
542    
543                    @Override
544                    public void run()
545                    {
546                            try {
547                                    KeyStore keyStore = keyStoreRef.get();
548                                    if (keyStore == null) {
549                                            logger.info("run: KeyStore has been garbage-collected. Removing this ExipreCacheEntryTimerTask.");
550                                            this.cancel();
551                                            return;
552                                    }
553    
554                                    Date removeCachedEntriesOlderThanThisDate = new Date(System.currentTimeMillis() - 3L * 60L * 1000L); // TODO make this configurable!
555    
556                                    LinkedList<String> userNamesToExpire = new LinkedList<String>();
557                                    synchronized (keyStore) {
558                                            for (CachedMasterKey cmk : keyStore.cache_userName2cachedMasterKey.values()) {
559                                                    if (cmk.getLastUse().before(removeCachedEntriesOlderThanThisDate))
560                                                            userNamesToExpire.add(cmk.getUserName());
561                                            }
562                                    }
563    
564                                    for (String userName : userNamesToExpire) {
565                                            logger.info("run: Expiring cache for user '{}'.", userName);
566                                            keyStore.clearCache(userName);
567                                    }
568    
569                                    if (logger.isDebugEnabled()) {
570                                            synchronized (keyStore) {
571                                                    logger.debug("run: {} users left in cache.", keyStore.cache_userName2cachedMasterKey.size());
572                                            }
573                                    }
574                            } catch (Throwable x) {
575                                    // The TimerThread is cancelled, if a task throws an exception. Furthermore, they are not logged at all.
576                                    // Since we do not want the TimerThread to die, we catch everything (Throwable - not only Exception) and log
577                                    // it here. IMHO there's nothing better we can do. Marco :-)
578                                    logger.error("run: " + x, x);
579                            }
580                    }
581            }
582    
583            /**
584             * Gets the key-size that is currently configured. Therefore, this method checks, if the
585             * system property {@value #SYSTEM_PROPERTY_KEY_SIZE} has been specified, and if so returns its value.
586             * If not, it falls back to 256.
587             *
588             * @return the current key-size.
589             */
590            int getKeySize()
591            {
592                    int ks = keySize;
593    
594                    if (ks == 0) {
595                            String keySizePropName = SYSTEM_PROPERTY_KEY_SIZE;
596                            String keySizePropValue = System.getProperty(keySizePropName);
597                            if (keySizePropValue == null || keySizePropValue.trim().isEmpty()) {
598                                    ks = 256; // default value, if the property was not defined.
599                                    logger.info("getKeySize: System property '{}' is not set. Using default key size ({} bit).", keySizePropName, ks);
600                            }
601                            else {
602                                    try {
603                                            ks = Integer.parseInt(keySizePropValue.trim());
604                                    } catch (NumberFormatException x) {
605                                            NumberFormatException n = new NumberFormatException("Value of system property '" + keySizePropName + "' is not a valid integer!");
606                                            n.initCause(x);
607                                            throw n;
608                                    }
609                                    if (ks < 1)
610                                            throw new IllegalStateException("Value of system property '" + keySizePropName + "' is " + keySize + " but must be >= 1!!!");
611    
612                                    logger.info("getKeySize: System property '{}' is set to {} bit. Using this key size.", keySizePropName, ks);
613                            }
614                            keySize = ks;
615                    }
616    
617                    return ks;
618            }
619            private int keySize = 0;
620    
621    
622            String getEncryptionAlgorithm()
623            {
624                    String ea = encryptionAlgorithm;
625    
626                    if (ea == null) {
627                            String encryptionAlgorithmPropName = SYSTEM_PROPERTY_ENCRYPTION_ALGORITHM;
628                            String encryptionAlgorithmPropValue = System.getProperty(encryptionAlgorithmPropName);
629                            if (encryptionAlgorithmPropValue == null || encryptionAlgorithmPropValue.trim().isEmpty()) {
630                                    ea = "Twofish/GCM/NoPadding"; // default value, if the property was not defined.
631    //                              ea = "Twofish/CBC/PKCS5Padding"; // default value, if the property was not defined.
632    //                              ea = "AES/CBC/PKCS5Padding"; // default value, if the property was not defined.
633    //                              ea = "AES/CFB/NoPadding"; // default value, if the property was not defined.
634                                    logger.info("getEncryptionAlgorithm: System property '{}' is not set. Using default algorithm '{}'.", encryptionAlgorithmPropName, ea);
635                            }
636                            else {
637                                    ea = encryptionAlgorithmPropValue.trim();
638                                    logger.info("getEncryptionAlgorithm: System property '{}' is set to '{}'. Using this encryption algorithm.", encryptionAlgorithmPropName, ea);
639                            }
640                            encryptionAlgorithm = ea;
641                    }
642    
643                    return ea;
644            }
645            private String encryptionAlgorithm = null;
646    
647    
648            String getMACAlgorithm()
649            {
650                    String ma = macAlgorithm;
651    
652                    if (ma == null) {
653                            String macAlgorithmPropName = SYSTEM_PROPERTY_MAC_ALGORITHM;
654                            String macAlgorithmPropValue = System.getProperty(macAlgorithmPropName);
655                            if (macAlgorithmPropValue == null || macAlgorithmPropValue.trim().isEmpty()) {
656                                    ma = MAC_ALGORITHM_NONE; // default value, if the property was not defined.
657                                    logger.info("getMACAlgorithm: System property '{}' is not set. Using default MAC algorithm '{}'.", macAlgorithmPropName, ma);
658                            }
659                            else {
660                                    ma = macAlgorithmPropValue.trim();
661                                    logger.info("getMACAlgorithm: System property '{}' is set to '{}'. Using this MAC algorithm.", macAlgorithmPropName, ma);
662                            }
663                            macAlgorithm = ma;
664                    }
665    
666                    return ma;
667            }
668            private String macAlgorithm = null;
669    
670    
671            byte[] generateKey(int keySize)
672            {
673                    byte[] result = new byte[(keySize + 7) / 8];
674                    secureRandom.nextBytes(result);
675                    return result;
676            }
677    
678            byte[] generateKey()
679            {
680                    return generateKey(getKeySize());
681    //              return new SecretKeySpec(
682    //                              generateKey(getKeySize()),
683    //                              getBaseAlgorithm(getEncryptionAlgorithm())
684    //              );
685            }
686    
687            private File keyStoreFile;
688    
689            /**
690             * <p>
691             * Create a new instance of <code>KeyStore</code>.
692             * </p>
693             * <p>
694             * If the file specified by <code>keyStoreFile</code> exists, it is read into memory. If it does not exist,
695             * an empty <code>KeyStore</code> is created and written to this file.
696             * </p>
697             *
698             * @param keyStoreFile the file to be read (if existing) or created. Note that temporary files (and later maybe backup files, too)
699             * are created in the same directory (i.e. in {@link File#getParentFile() keyStoreFile.getParentFile()}).
700             * @throws IOException if reading from or writing to the local file-system failed.
701             */
702            public KeyStore(File keyStoreFile) throws IOException
703            {
704                    if (keyStoreFile == null)
705                            throw new IllegalArgumentException("keyStoreFile == null");
706    
707                    this.keyStoreFile = keyStoreFile;
708    
709                    if (!keyStoreFile.getParentFile().isDirectory())
710                            throw new FileNotFoundException("Path does not exist or is not a directory: " + keyStoreFile.getParentFile().getAbsolutePath());
711    
712                    // In case the old file was already deleted, but the new not yet renamed, we check, if a new file
713                    // exists and the old file is missing - in this case, we load the new file.
714                    File newKeyStoreFile = getNewKeyStoreFile();
715                    if (!keyStoreFile.exists() && newKeyStoreFile.exists())
716                            keyStoreFile = newKeyStoreFile;
717    
718                    FileInputStream in = keyStoreFile.length() == 0 ? null : new FileInputStream(keyStoreFile);
719                    if (in != null) {
720                            try {
721                                    keyStoreData.readFromStream(in);
722                            } finally {
723                                    in.close();
724                            }
725                    }
726                    else
727                            storeToFile(); // create the file (empty) already now, if it does not exist.
728    
729                    expireCacheEntryTimer.schedule(expireCacheEntryTimerTask, 60000, 60000); // TODO make this configurable
730            }
731    
732            File getNewKeyStoreFile()
733            {
734                    return new File(keyStoreFile.getParentFile(), keyStoreFile.getName() + ".new");
735            }
736    
737            /**
738             * Determine if this <code>KeyStore</code> is completely empty. As soon as the first user has been
739             * created, this method will return <code>false</code>.
740             *
741             * @return <code>true</code> if this <code>KeyStore</code> contains neither any user nor any key, i.e. is totally empty;
742             * <code>false</code> otherwise.
743             */
744            public synchronized boolean isEmpty()
745            {
746                    return keyStoreData.user2keyMap.isEmpty();
747            }
748    
749            synchronized long nextKeyID(String authUserName, char[] authPassword) throws AuthenticationException
750            {
751                    LongProperty property = getProperty(authUserName, authPassword, LongProperty.class, KEY_STORE_PROPERTY_NAME_NEXT_KEY_ID);
752                    if (property.getValue() == null)
753                            property.setValue(1L);
754    
755                    long result = property.getValue();
756                    property.setValue(result + 1);
757                    _setProperty(authUserName, authPassword, property);
758                    return result;
759            }
760    
761            private Map<String, CachedMasterKey> cache_userName2cachedMasterKey = new HashMap<String, CachedMasterKey>();
762    
763            public synchronized int getMasterKeySize(String authUserName, char[] authPassword)
764            throws AuthenticationException
765            {
766                    MasterKey masterKey = getMasterKey(authUserName, authPassword);
767                    return masterKey.getEncoded().length * 8;
768            }
769    
770            /**
771             * Authenticate and get the master-key. If there is a cache-entry existing, directly return this
772             * (after comparing the password); otherwise decrypt the master-key using the given password.
773             *
774             * @param authUserName the user from whose slot to take and decrypt the master-key.
775             * @param authPassword the password with which to try to decrypt the master-key.
776             * @return the decrypted, plain master-key.
777             * @throws AuthenticationException if the specified <code>authUserName</code> does not exist or the specified <code>authPassword</code>
778             * is not correct for the given <code>authUserName</code>.
779             */
780            synchronized MasterKey getMasterKey(String authUserName, char[] authPassword)
781            throws AuthenticationException
782            {
783    //              logger.trace("getMasterKey: authUserName={} authPassword={}", authUserName, new String(authPassword));
784    
785                    CachedMasterKey cachedMasterKey = cache_userName2cachedMasterKey.get(authUserName);
786                    MasterKey result = cachedMasterKey == null ? null : cachedMasterKey.getMasterKey();
787                    if (result != null && Arrays.equals(authPassword, cachedMasterKey.getPassword())) {
788                            cachedMasterKey.updateLastUse();
789                            return result;
790                    }
791                    result = null;
792    
793                    EncryptedMasterKey encryptedKey = keyStoreData.user2keyMap.get(authUserName);
794                    if (encryptedKey == null)
795                            logger.warn("getMasterKey: Unknown userName: {}", authUserName); // NOT throw exception here to not disclose the true reason of the AuthenticationException - see below
796                    else {
797                            PlaintextDataAndMAC plaintextDataAndMAC;
798                            try {
799                                    Cipher cipher = getCipherForUserPassword(
800                                                    authPassword,
801                                                    encryptedKey.getPasswordBasedKeySize(),
802                                                    encryptedKey.getPasswordBasedIterationCount(),
803                                                    encryptedKey.getPasswordBasedKeyGeneratorAlgorithm(),
804                                                    encryptedKey.getSalt(),
805                                                    encryptedKey.getEncryptionIV(), encryptedKey.getEncryptionAlgorithm(),
806                                                    CipherOperationMode.DECRYPT
807                                    );
808                                    byte[] decrypted = cipher.doFinal(encryptedKey.getEncryptedData());
809    
810                                    plaintextDataAndMAC = new PlaintextDataAndMAC(decrypted, encryptedKey);
811                            } catch (CryptoException x) {
812                                    logger.warn("getMasterKey: Caught CryptoException indicating a wrong password for user \"{}\"!", authUserName);
813                                    plaintextDataAndMAC = null;
814                            } catch (GeneralSecurityException x) {
815                                    throw new RuntimeException(x);
816                            }
817    
818                            try {
819                                    if (plaintextDataAndMAC != null && plaintextDataAndMAC.verifyMAC())
820                                            result = new MasterKey(plaintextDataAndMAC.getData());
821                                    else
822                                            logger.warn("getMasterKey: Wrong password for user \"{}\"! MAC verification failed.", authUserName);
823                            } catch (GeneralSecurityException x) {
824                                    throw new RuntimeException(x);
825                            }
826                    }
827    
828                    // We check only once at the end of this method if we could successfully authenticate and otherwise
829                    // throw a AuthenticationException. If we threw the AuthenticationException at different locations (even with the same
830                    // message), and attacker might know from the stack trace (=> line number) whether the user-name
831                    // or the password was wrong. This information will be logged, but not disclosed in the exception.
832                    // Marco :-)
833                    if (result == null)
834                            throw new AuthenticationException("Unknown user \"" + authUserName + "\" or wrong password!");
835    
836                    cache_userName2cachedMasterKey.put(authUserName, new CachedMasterKey(authUserName, authPassword, result));
837                    return result;
838            }
839    
840            private Cipher getCipherForUserPassword(
841                            char[] password,
842                            int passwordBasedKeySize, int passwordBasedIterationCount, String passwordBasedKeyGeneratorAlgorithm,
843                            byte[] salt, byte[] iv, String algorithm, CipherOperationMode opmode) throws GeneralSecurityException
844            {
845                    if (iv == null) {
846                            if (CipherOperationMode.ENCRYPT != opmode)
847                                    throw new IllegalArgumentException("iv must not be null when decrypting!");
848                    }
849                    else {
850                            if (CipherOperationMode.ENCRYPT == opmode)
851                                    throw new IllegalArgumentException("iv must be null when encrypting!");
852                    }
853    
854                    if (algorithm == null) {
855                            if (CipherOperationMode.ENCRYPT != opmode)
856                                    throw new IllegalArgumentException("algorithm must not be null when decrypting!");
857    
858                            algorithm = getEncryptionAlgorithm();
859                    }
860    
861                    SecretKeyFactory factory = SecretKeyFactory.getInstance(passwordBasedKeyGeneratorAlgorithm);
862    
863                    KeySpec spec = new PBEKeySpec(password, salt, passwordBasedIterationCount, passwordBasedKeySize);
864                    SecretKey secretKey = factory.generateSecret(spec);
865    
866                    Cipher cipher = CryptoRegistry.sharedInstance().createCipher(algorithm);
867    
868                    if (iv == null) {
869                            iv = new byte[cipher.getIVSize()];
870                            secureRandom.nextBytes(iv);
871                    }
872    
873                    cipher.init(opmode, new ParametersWithIV(new KeyParameter(secretKey.getEncoded()), iv));
874    
875                    return cipher;
876            }
877    
878    //      private String getBaseAlgorithm(String algorithm)
879    //      {
880    //              int slashIdx = algorithm.indexOf('/');
881    //              if (slashIdx < 0)
882    //                      return algorithm;
883    //
884    //              return algorithm.substring(0, slashIdx);
885    //      }
886    
887            private Cipher getCipherForMasterKey(MasterKey masterKey, byte[] iv, String algorithm, CipherOperationMode opmode) throws GeneralSecurityException
888            {
889                    if (iv == null) {
890                            if (CipherOperationMode.ENCRYPT != opmode)
891                                    throw new IllegalArgumentException("iv must not be null when decrypting!");
892                    }
893                    else {
894                            if (CipherOperationMode.ENCRYPT == opmode)
895                                    throw new IllegalArgumentException("iv must be null when encrypting!");
896                    }
897    
898                    if (algorithm == null) {
899                            if (CipherOperationMode.ENCRYPT != opmode)
900                                    throw new IllegalArgumentException("algorithm must not be null when decrypting!");
901    
902                            algorithm = getEncryptionAlgorithm();
903                    }
904    
905                    Cipher cipher = CryptoRegistry.sharedInstance().createCipher(algorithm);
906    
907                    if (iv == null) {
908                            iv = new byte[cipher.getIVSize()];
909                            secureRandom.nextBytes(iv);
910                    }
911                    cipher.init(opmode, new ParametersWithIV(new KeyParameter(masterKey.getEncoded()), iv));
912                    return cipher;
913            }
914    
915            /**
916             * <p>
917             * Generate a new key and store it to the file.
918             * </p>
919             * <p>
920             * The new key will be generated with the size specified by the
921             * system property {@value #SYSTEM_PROPERTY_KEY_SIZE} and encrypted with the
922             * master-key and the encryption-algorithm specified by the
923             * system property {@value #SYSTEM_PROPERTY_ENCRYPTION_ALGORITHM}.
924             * </p>
925             *
926             * @param authUserName the authenticated user authorizing this action.
927             * @param authPassword the password for authenticating the user specified by <code>authUserName</code>.
928             * @return the newly created key.
929             * @throws AuthenticationException if the specified <code>authUserName</code> does not exist or the specified <code>authPassword</code>
930             * is not correct for the given <code>authUserName</code>.
931             * @throws IOException if writing to the local file-system failed.
932             */
933            public synchronized GeneratedKey generateKey(String authUserName, char[] authPassword)
934            throws AuthenticationException, IOException
935            {
936                    long keyID = nextKeyID(authUserName, authPassword);
937                    byte[] key = generateKey();
938                    GeneratedKey generatedKey = new GeneratedKey(keyID, key);
939                    _setKey(authUserName, authPassword, keyID, key);
940                    storeToFile();
941                    return generatedKey;
942            }
943    
944            /**
945             * <p>
946             * Generate <code>qty</code> new keys and store them to the file.
947             * </p>
948             * <p>
949             * This method behaves like {@link #generateKey(String, char[])} but is much
950             * faster when multiple keys have to be generated (bulk operation).
951             * </p>
952             *
953             * @param authUserName the authenticated user authorizing this action.
954             * @param authPassword the password for authenticating the user specified by <code>authUserName</code>.
955             * @param qty the number of keys to be generated. If 0, the method will do nothing and return
956             * an empty list, if &lt; 0, an {@link IllegalArgumentException} will be thrown.
957             * @return a list of generated keys; never <code>null</code>.
958             * @throws AuthenticationException if the specified <code>authUserName</code> does not exist or the specified <code>authPassword</code>
959             * is not correct for the given <code>authUserName</code>.
960             * @throws IOException if writing to the local file-system failed.
961             */
962            public synchronized List<GeneratedKey> generateKeys(String authUserName, char[] authPassword, int qty)
963            throws AuthenticationException, IOException
964            {
965                    if (qty < 0)
966                            throw new IllegalArgumentException("qty < 0");
967    
968                    List<GeneratedKey> result = new ArrayList<GeneratedKey>(qty);
969                    for (int i = 0; i < qty; ++i) {
970                            long keyID = nextKeyID(authUserName, authPassword);
971                            byte[] key = generateKey();
972                            GeneratedKey generatedKey = new GeneratedKey(keyID, key);
973                            _setKey(authUserName, authPassword, keyID, key);
974                            result.add(generatedKey);
975                    }
976                    storeToFile();
977                    return result;
978            }
979    
980            /**
981             * <p>
982             * Create a new user.
983             * </p>
984             * <p>
985             * Before the <code>KeyStore</code> can be used (i.e. before most methods work), this method has to be called
986             * to create the first user. When the first user is created, the internal master-key is generated, which will
987             * then not be changed anymore (double-check that the {@link #SYSTEM_PROPERTY_KEY_SIZE key-size} is set correctly at
988             * this time).
989             * </p>
990             *
991             * @param authUserName the authenticated user authorizing this action. If the very first user is created, this value
992             * is ignored and can be <code>null</code>.
993             * @param authPassword the password for authenticating the user specified by <code>authUserName</code>. If the very first user is created, this value
994             * is ignored and can be <code>null</code>.
995             * @param userName the name of the user to be created.
996             * @param password the password of the new user.
997             * @throws AuthenticationException if the specified <code>authUserName</code> does not exist or the specified <code>authPassword</code>
998             * is not correct for the given <code>authUserName</code>.
999             * @throws UserAlreadyExistsException if a user with the name specified by <code>userName</code> already exists.
1000             * @throws IOException if writing to the local file-system failed.
1001             */
1002            public synchronized void createUser(String authUserName, char[] authPassword, String userName, char[] password)
1003            throws AuthenticationException, UserAlreadyExistsException, IOException
1004            {
1005                    if (userName == null)
1006                            throw new IllegalArgumentException("userName must not be null!");
1007    
1008                    if (password == null)
1009                            throw new IllegalArgumentException("password must not be null!");
1010    
1011                    MasterKey masterKey;
1012    
1013                    if (isEmpty()) {
1014                            byte[] key = generateKey();
1015                            masterKey = new MasterKey(key);
1016                            // Unfortunately, we cannot clear the sensitive data from the key instance, because
1017                            // there is no nice way to do this (we could only do very ugly reflection-based stuff).
1018                            // But fortunately, this happens only the very first time a new, empty KeyStore is created.
1019                            // With an existing KeyStore we won't come here and our MasterKey can [and will] be cleared.
1020                            // Marco :-)
1021                            logger.info("createUser: Created master-key with a size of {} bits. This key will not be modified for this key-store anymore.", key.length * 8);
1022                    }
1023                    else
1024                            masterKey = getMasterKey(authUserName, authPassword);
1025    
1026                    if (keyStoreData.user2keyMap.containsKey(userName))
1027                            throw new UserAlreadyExistsException("User '" + userName + "' already exists!");
1028    
1029                    setUser(masterKey, userName, password);
1030            }
1031    
1032            synchronized void setUser(MasterKey masterKey, String userName, char[] password)
1033            throws IOException
1034            {
1035                    byte[] plainMasterKeyData = masterKey.getEncoded();
1036    
1037                    byte[] salt = new byte[8]; // Are 8 bytes salt salty (i.e. secure) enough?
1038                    secureRandom.nextBytes(salt);
1039                    try {
1040                            int passwordBasedKeySize = getKeySize();
1041                            int passwordBasedIterationCount = 1024; // TODO make configurable!
1042                            String passwordBasedKeyGeneratorAlgorithm = "PBKDF2WithHmacSHA1"; // TODO make configurable
1043    
1044                            Cipher cipher = getCipherForUserPassword(
1045                                            password,
1046                                            passwordBasedKeySize,
1047                                            passwordBasedIterationCount,
1048                                            passwordBasedKeyGeneratorAlgorithm,
1049                                            salt, null, null, CipherOperationMode.ENCRYPT
1050                            );
1051    
1052                            PlaintextDataAndMAC plaintextDataAndMAC = new PlaintextDataAndMAC(plainMasterKeyData, getMACAlgorithm());
1053                            byte[] encrypted = cipher.doFinal(plaintextDataAndMAC.toByteArray());
1054    
1055                            byte[] iv = ((ParametersWithIV)cipher.getParameters()).getIV();
1056    
1057                            EncryptedMasterKey encryptedKey = new EncryptedMasterKey(
1058                                            keyStoreData,
1059                                            userName,
1060                                            passwordBasedKeySize,
1061                                            passwordBasedIterationCount,
1062                                            keyStoreData.stringConstant(passwordBasedKeyGeneratorAlgorithm),
1063                                            salt,
1064                                            keyStoreData.stringConstant(cipher.getTransformation()),
1065                                            iv,
1066                                            keyStoreData.stringConstant(plaintextDataAndMAC.getMACAlgorithm()),
1067                                            (short)plaintextDataAndMAC.getMACKey().length,
1068                                            (short)plaintextDataAndMAC.getMACIV().length,
1069                                            (short)plaintextDataAndMAC.getMAC().length, encrypted
1070                            );
1071                            keyStoreData.user2keyMap.put(userName, encryptedKey);
1072                            usersCache = null;
1073                    } catch (CryptoException e) {
1074                            throw new RuntimeException(e);
1075                    } catch (GeneralSecurityException e) {
1076                            throw new RuntimeException(e);
1077                    }
1078    
1079                    storeToFile();
1080            }
1081    
1082            synchronized void storeToFile() throws IOException
1083            {
1084                    File newKeyStoreFile = getNewKeyStoreFile();
1085                    boolean deleteNewKeyStoreFile = true;
1086                    try {
1087                            OutputStream out = new FileOutputStream(newKeyStoreFile);
1088                            try {
1089                                    keyStoreData.writeToStream(out);
1090                            } finally {
1091                                    out.close();
1092                            }
1093    
1094                            deleteNewKeyStoreFile = false;
1095                            keyStoreFile.delete();
1096                            newKeyStoreFile.renameTo(keyStoreFile);
1097                    } finally {
1098                            if (deleteNewKeyStoreFile) {
1099                                    try {
1100                                            newKeyStoreFile.delete();
1101                                    } catch (Exception x) {
1102                                            logger.warn("Deleting the newKeyStoreFile failed!", x);
1103                                    }
1104                            }
1105                    }
1106            }
1107    
1108            /**
1109             * <p>
1110             * Get all users who can authenticate at this <code>KeyStore</code>.
1111             * </p>
1112             *
1113             * @param authUserName the authenticated user authorizing this action.
1114             * @param authPassword the password for authenticating the user specified by <code>authUserName</code>.
1115             * @return a read-only {@link Set} of all user-names known to this <code>KeyStore</code>. This
1116             * <code>Set</code> is an unmodifiable copy of the internally used data and therefore is both thread-safe
1117             * and iteration-safe (i.e. it can be iterated while simultaneously users are {@link #deleteUser(String, char[], String) deleted}).
1118             * @throws AuthenticationException if the specified <code>authUserName</code> does not exist or the specified <code>authPassword</code>
1119             * is not correct for the given <code>authUserName</code>.
1120             */
1121            public synchronized SortedSet<String> getUsers(String authUserName, char[] authPassword)
1122            throws AuthenticationException
1123            {
1124                    // The following getMasterKey(...) is no real protection, because the information returned by this method
1125                    // is currently not protected, but this way, we already have the right arguments to later encrypt this
1126                    // information, too - if we ever want to.
1127                    // Marco :-)
1128                    getMasterKey(authUserName, authPassword);
1129    
1130                    SortedSet<String> users = usersCache;
1131                    if (users == null) {
1132                            users = Collections.unmodifiableSortedSet(new TreeSet<String>(keyStoreData.user2keyMap.keySet()));
1133                            usersCache = users;
1134                    }
1135    
1136                    return users;
1137            }
1138    
1139            private SortedSet<String> usersCache = null;
1140    
1141            /**
1142             * <p>
1143             * Delete the user specified by <code>userName</code>.
1144             * </p>
1145             * <p>
1146             * Deleting the authenticated user himself (i.e. <code>authUserName == userName</code>) is possible,
1147             * as long as it is not the last user.
1148             * </p>
1149             *
1150             * @param authUserName the name of the principal, i.e. the user authorizing this operation.
1151             * @param authPassword the password of the principal.
1152             * @param userName the name of the user to be deleted.
1153             * @throws AuthenticationException if the specified <code>authUserName</code> does not exist or the specified <code>authPassword</code>
1154             * is not correct for the given <code>authUserName</code>.
1155             * @throws UserNotFoundException if there is no user with the name specified by <code>userName</code>.
1156             * @throws CannotDeleteLastUserException if the last user would be deleted by this method invocation (thus rendering
1157             * the <code>KeyStore</code> unusable and unrecoverable - i.e. totally lost).
1158             * @throws IOException if writing to the local file-system failed.
1159             */
1160            public synchronized void deleteUser(String authUserName, char[] authPassword, String userName)
1161            throws AuthenticationException, UserNotFoundException, CannotDeleteLastUserException, IOException
1162            {
1163                    // The following getMasterKey(...) is no real protection, because a user can be deleted without
1164                    // authenticating on the file-base (as this doesn't require to decrypt data, currently), but
1165                    // this way, we already have the right arguments here and might later encrypt the required infos.
1166                    // Marco :-)
1167                    getMasterKey(authUserName, authPassword);
1168    
1169                    EncryptedMasterKey encryptedKey = keyStoreData.user2keyMap.get(userName);
1170                    if (encryptedKey == null)
1171                            throw new UserNotFoundException("The user \"" + userName + "\" does not exist!");
1172    
1173                    if (keyStoreData.user2keyMap.size() == 1)
1174                            throw new CannotDeleteLastUserException("You cannot delete the last user and \"" + userName + "\" is the last user!");
1175    
1176                    clearCache(userName);
1177                    keyStoreData.user2keyMap.remove(userName);
1178                    usersCache = null;
1179    
1180                    storeToFile();
1181            }
1182    
1183            /**
1184             * <p>
1185             * Change a user's password.
1186             * </p>
1187             * <p>
1188             * The user identified by <code>userName</code> will have the new password specified by
1189             * <code>newPassword</code> immediately after this method. Authenticating this user with
1190             * his old password will fail afterwards.
1191             * </p>
1192             *
1193             * @param authUserName the authenticated user authorizing this action.
1194             * @param authPassword the password for authenticating the user specified by <code>authUserName</code>.
1195             * @param userName the user whose password is to be changed. This can be the same as <code>authUserName</code>.
1196             * @param newPassword the new password.
1197             * @throws AuthenticationException if the specified <code>authUserName</code> does not exist or the specified <code>authPassword</code>
1198             * is not correct for the given <code>authUserName</code>.
1199             * @throws UserNotFoundException if there is no user with the name specified by <code>userName</code>.
1200             * @throws IOException if writing to the local file-system failed.
1201             */
1202            public synchronized void changeUserPassword(String authUserName, char[] authPassword, String userName, char[] newPassword)
1203            throws AuthenticationException, UserNotFoundException, IOException
1204            {
1205                    MasterKey masterKey = getMasterKey(authUserName, authPassword);
1206    
1207                    if (!keyStoreData.user2keyMap.containsKey(userName))
1208                            throw new UserNotFoundException("User '" + userName + "' does not exist!");
1209    
1210                    setUser(masterKey, userName, newPassword);
1211            }
1212    
1213            /**
1214             * Get the key identified by the given <code>keyID</code>.
1215             *
1216             * @param authUserName the authenticated user authorizing this action.
1217             * @param authPassword the password for authenticating the user specified by <code>authUserName</code>.
1218             * @param keyID the identifier of the key to get.
1219             * @return the key associated with the given identifier; never <code>null</code> (if there is no key for the given <code>keyID</code>,
1220             * a {@link KeyNotFoundException} is thrown).
1221             * @throws AuthenticationException if the specified <code>authUserName</code> does not exist or the specified <code>authPassword</code>
1222             * is not correct for the given <code>authUserName</code>.
1223             * @throws KeyNotFoundException if the specified <code>keyID</code> does not reference any existing key. Note, that the
1224             * authentication process occurs before any lookup and therefore a {@link KeyNotFoundException} indicates a correct authentication
1225             * (otherwise the {@link AuthenticationException} would have been thrown before).
1226             */
1227            public synchronized byte[] getKey(String authUserName, char[] authPassword, long keyID)
1228            throws AuthenticationException, KeyNotFoundException
1229            {
1230                    MasterKey masterKey = getMasterKey(authUserName, authPassword);
1231                    EncryptedKey encryptedKey = keyStoreData.keyID2keyMap.get(keyID);
1232                    if (encryptedKey == null)
1233                            throw new KeyNotFoundException("There is no key with keyID=" + keyID + "!");
1234    
1235                    try {
1236                            Cipher cipher = getCipherForMasterKey(
1237                                            masterKey,
1238                                            encryptedKey.getEncryptionIV(),
1239                                            encryptedKey.getEncryptionAlgorithm(),
1240                                            CipherOperationMode.DECRYPT
1241                            );
1242                            byte[] decrypted = cipher.doFinal(encryptedKey.getEncryptedData());
1243    
1244                            PlaintextDataAndMAC plaintextDataAndMAC = new PlaintextDataAndMAC(decrypted, encryptedKey);
1245                            if (!plaintextDataAndMAC.verifyMAC())
1246                                    throw new IllegalStateException("MAC mismatch!!! This means, the decryption key was wrong!");
1247    
1248                            return plaintextDataAndMAC.getData();
1249                    } catch (CryptoException e) {
1250                            throw new RuntimeException(e);
1251                    } catch (GeneralSecurityException e) {
1252                            throw new RuntimeException(e);
1253                    }
1254            }
1255    
1256            public synchronized SortedSet<Long> getKeyIDs(String authUserName, char[] authPassword)
1257            throws AuthenticationException
1258            {
1259                    getMasterKey(authUserName, authPassword);
1260                    SortedSet<Long> result = new TreeSet<Long>(keyStoreData.keyID2keyMap.keySet());
1261                    return result;
1262            }
1263    
1264            private void _setKey(String authUserName, char[] authPassword, long keyID, byte[] key)
1265            throws AuthenticationException
1266            {
1267                    MasterKey masterKey = getMasterKey(authUserName, authPassword);
1268    
1269                    try {
1270                            PlaintextDataAndMAC plaintextDataAndMAC = new PlaintextDataAndMAC(key, getMACAlgorithm());
1271    
1272                            Cipher cipher = getCipherForMasterKey(masterKey, null, null, CipherOperationMode.ENCRYPT);
1273                            byte[] iv = ((ParametersWithIV)cipher.getParameters()).getIV();
1274                            byte[] encrypted = cipher.doFinal(plaintextDataAndMAC.toByteArray());
1275    
1276                            EncryptedKey encryptedKey = new EncryptedKey(
1277                                            keyStoreData,
1278                                            keyID,
1279                                            keyStoreData.stringConstant(cipher.getTransformation()),
1280                                            iv,
1281                                            plaintextDataAndMAC.getMACAlgorithm(),
1282                                            (short)plaintextDataAndMAC.getMACKey().length,
1283                                            (short)plaintextDataAndMAC.getMACIV().length,
1284                                            (short)plaintextDataAndMAC.getMAC().length, encrypted
1285                            );
1286                            keyStoreData.keyID2keyMap.put(keyID, encryptedKey);
1287                    } catch (CryptoException e) {
1288                            throw new RuntimeException(e);
1289                    } catch (GeneralSecurityException e) {
1290                            throw new RuntimeException(e);
1291                    }
1292            }
1293    
1294            /**
1295             * <p>
1296             * Get a named property.
1297             * </p>
1298             * <p>
1299             * The <code>KeyStore</code> supports managing arbitrary properties in the form of
1300             * name-value-pairs. The names are plain-text, but the values are encrypted.
1301             * A property-value can be of any type for which a subclass of
1302             * {@link org.cumulus4j.keystore.prop.Property} exists.
1303             * </p>
1304             * <p>
1305             * This method will always return an instance of the given <code>propertyType</code>, no matter,
1306             * if the property exists in this <code>KeyStore</code> or not. If the property does not exist,
1307             * its {@link Property#getValue() value} will be <code>null</code>.
1308             * </p>
1309             * <p>
1310             * <b>Important:</b> Never directly instantiate a {@link Property}-subclass. Always use this method
1311             * as a factory for property instances.
1312             * </p>
1313             *
1314             * @param <P> the type of the property.
1315             * @param authUserName the authenticated user authorizing this action.
1316             * @param authPassword the password for authenticating the user specified by <code>authUserName</code>.
1317             * @param propertyType the type of the property; must not be <code>null</code>. If the property does not yet exist,
1318             * every type can be specified. If the property already exists, this type must match the type of the property.
1319             * If they do not match, an {@link IllegalArgumentException} is thrown.
1320             * @param name the unique name of the property; must not be <code>null</code>.
1321             * @return the property; never <code>null</code>. If the property does not yet exist, a new, empty property is returned.
1322             * @throws AuthenticationException if the specified <code>authUserName</code> does not exist or the specified <code>authPassword</code>
1323             * is not correct for the given <code>authUserName</code>.
1324             * @see #setProperty(String, char[], Property)
1325             * @see #removeProperty(String, char[], String)
1326             */
1327            public synchronized <P extends Property<?>> P getProperty(String authUserName, char[] authPassword, Class<P> propertyType, String name)
1328            throws AuthenticationException
1329            {
1330                    MasterKey masterKey = getMasterKey(authUserName, authPassword);
1331    
1332                    if (name == null)
1333                            throw new IllegalArgumentException("name == null");
1334    
1335                    EncryptedProperty encryptedProperty = keyStoreData.name2propertyMap.get(name);
1336    
1337                    P result;
1338                    try {
1339                            result = propertyType.newInstance();
1340                    } catch (InstantiationException e) {
1341                            throw new RuntimeException(e);
1342                    } catch (IllegalAccessException e) {
1343                            throw new RuntimeException(e);
1344                    }
1345                    result.setName(name);
1346                    result.setXxx(propertyXxx);
1347    
1348                    if (encryptedProperty != null) {
1349                            if (!propertyType.equals(encryptedProperty.getType()))
1350                                    throw new IllegalArgumentException("propertyType != encryptedProperty.type :: " + propertyType.getClass().getName() + " != " + encryptedProperty.getType().getName());
1351    
1352                            try {
1353                                    Cipher cipher = getCipherForMasterKey(
1354                                                    masterKey,
1355                                                    encryptedProperty.getEncryptionIV(),
1356                                                    encryptedProperty.getEncryptionAlgorithm(),
1357                                                    CipherOperationMode.DECRYPT
1358                                    );
1359                                    byte[] decrypted = cipher.doFinal(encryptedProperty.getEncryptedData());
1360    
1361                                    PlaintextDataAndMAC plaintextDataAndMAC = new PlaintextDataAndMAC(decrypted, encryptedProperty);
1362                                    if (!plaintextDataAndMAC.verifyMAC())
1363                                            throw new IllegalStateException("MAC mismatch!!! This means, the decryption key was wrong!");
1364    
1365                                    result.setValueEncoded(plaintextDataAndMAC.getData());
1366                            } catch (CryptoException e) {
1367                                    throw new RuntimeException(e);
1368                            } catch (GeneralSecurityException e) {
1369                                    throw new RuntimeException(e);
1370                            }
1371                    }
1372    
1373                    return result;
1374            }
1375    
1376            public synchronized SortedSet<Property<?>> getProperties(String authUserName, char[] authPassword)
1377            throws AuthenticationException
1378            {
1379                    SortedSet<Property<?>> result = new TreeSet<Property<?>>();
1380                    for (Map.Entry<String, EncryptedProperty> me : keyStoreData.name2propertyMap.entrySet()) {
1381                            Property<?> property = getProperty(authUserName, authPassword, me.getValue().getType(), me.getKey());
1382                            result.add(property);
1383                    }
1384                    return result;
1385            }
1386    
1387            /**
1388             * <p>
1389             * Remove a property.
1390             * </p>
1391             * <p>
1392             * If the property with the given name does not exist, this method won't do anything.
1393             * </p>
1394             *
1395             * @param authUserName the authenticated user authorizing this action.
1396             * @param authPassword the password for authenticating the user specified by <code>authUserName</code>.
1397             * @param name the unique name of the property; must not be <code>null</code>.
1398             * @return whether the property was removed, i.e. whether this <code>KeyStore</code> was changed by the operation.
1399             * @throws AuthenticationException if the specified <code>authUserName</code> does not exist or the specified <code>authPassword</code>
1400             * is not correct for the given <code>authUserName</code>.
1401             * @throws IOException if writing to the local file-system failed.
1402             * @see #getProperty(String, char[], Class, String)
1403             * @see #setProperty(String, char[], Property)
1404             */
1405            public synchronized boolean removeProperty(String authUserName, char[] authPassword, String name)
1406            throws AuthenticationException, IOException
1407            {
1408                    boolean removed = _removeProperty(authUserName, authPassword, name);
1409    
1410                    if (removed)
1411                            storeToFile();
1412    
1413                    return removed;
1414            }
1415    
1416            boolean _removeProperty(String authUserName, char[] authPassword, String name)
1417            throws AuthenticationException
1418            {
1419                    getMasterKey(authUserName, authPassword);
1420                    return keyStoreData.name2propertyMap.remove(name) != null;
1421            }
1422    
1423            private UUID propertyXxx = UUID.randomUUID();
1424    
1425            /**
1426             * <p>
1427             * Set a property.
1428             * </p>
1429             * <p>
1430             * If the property's {@link Property#getValue() value} is <code>null</code>, the property is
1431             * {@link #removeProperty(String, char[], String) removed} instead.
1432             * </p>
1433             * <p>
1434             * If a property with the same {@link Property#getName() name} already exists, it is overwritten.
1435             * </p>
1436             * <p>
1437             * The property's value is encrypted with the internal master-key. The property's name is stored
1438             * in plain (unencrypted) form.
1439             * </p>
1440             *
1441             * @param authUserName the authenticated user authorizing this action.
1442             * @param authPassword the password for authenticating the user specified by <code>authUserName</code>.
1443             * @param property the property to set. Do not instantiate any property directly!
1444             * Use {@link #getProperty(String, char[], Class, String)} instead!
1445             * @throws AuthenticationException if the specified <code>authUserName</code> does not exist or the specified <code>authPassword</code>
1446             * is not correct for the given <code>authUserName</code>.
1447             * @throws IOException if writing to the local file-system failed.
1448             * @see #getProperty(String, char[], Class, String)
1449             * @see #removeProperty(String, char[], String)
1450             */
1451            public synchronized void setProperty(String authUserName, char[] authPassword, Property<?> property)
1452            throws AuthenticationException, IOException
1453            {
1454                    _setProperty(authUserName, authPassword, property);
1455                    storeToFile();
1456            }
1457    
1458            private void _setProperty(String authUserName, char[] authPassword, Property<?> property)
1459            throws AuthenticationException
1460            {
1461                    MasterKey masterKey = getMasterKey(authUserName, authPassword);
1462    
1463                    if (property == null)
1464                            throw new IllegalArgumentException("property == null");
1465    
1466                    if (!propertyXxx.equals(property.getXxx()))
1467                            throw new IllegalArgumentException("property was not created by this KeyStore! You should use 'getProperty(...)' instead of 'new SomeProperty(...)'!!! And you should never store properties unencrypted somewhere outside!");
1468    
1469                    if (property.getName() == null)
1470                            throw new IllegalArgumentException("property.name == null");
1471    
1472                    keyStoreData.stringConstant(property.getClass().getName());
1473    
1474                    byte[] plainValueEncoded = property.getValueEncoded();
1475                    if (plainValueEncoded == null) {
1476                            _removeProperty(authUserName, authPassword, property.getName());
1477                    }
1478                    else {
1479                            try {
1480                                    PlaintextDataAndMAC plaintextDataAndMAC = new PlaintextDataAndMAC(plainValueEncoded, getMACAlgorithm());
1481    
1482                                    Cipher cipher = getCipherForMasterKey(masterKey, null, null, CipherOperationMode.ENCRYPT);
1483                                    byte[] encrypted = cipher.doFinal(plaintextDataAndMAC.toByteArray());
1484                                    byte[] iv = ((ParametersWithIV)cipher.getParameters()).getIV();
1485    
1486                                    @SuppressWarnings("unchecked")
1487                                    Class<? extends Property<?>> propertyType = (Class<? extends Property<?>>) property.getClass();
1488                                    EncryptedProperty encryptedProperty = new EncryptedProperty(
1489                                                    keyStoreData, property.getName(),
1490                                                    propertyType,
1491                                                    keyStoreData.stringConstant(cipher.getTransformation()),
1492                                                    iv,
1493                                                    plaintextDataAndMAC.getMACAlgorithm(),
1494                                                    (short)plaintextDataAndMAC.getMACKey().length,
1495                                                    (short)plaintextDataAndMAC.getMACIV().length,
1496                                                    (short)plaintextDataAndMAC.getMAC().length, encrypted
1497                                    );
1498                                    keyStoreData.name2propertyMap.put(encryptedProperty.getName(), encryptedProperty);
1499                            } catch (CryptoException e) {
1500                                    throw new RuntimeException(e);
1501                            } catch (GeneralSecurityException e) {
1502                                    throw new RuntimeException(e);
1503                            }
1504                    }
1505            }
1506    
1507            /**
1508             * <p>
1509             * Clear all cached data for the specified user name.
1510             * </p>
1511             * <p>
1512             * Every time, a user
1513             * calls a method requiring <code>authUserName</code> and <code>authPassword</code>,
1514             * either an authentication process happens, or a previously cached authentication
1515             * result (i.e. a decrypted master-key) is used. In order to speed things up, authentication results are cached for a
1516             * limited time. After this time elapses, the data is cleared by a timer. If a user wants (for security reasons)
1517             * remove the cached data from the memory earlier, he can call this method.
1518             * </p>
1519             *
1520             * @param userName the user for which to clear all the cached data. <code>null</code> to clear the complete cache for all users.
1521             */
1522            public synchronized void clearCache(String userName)
1523            {
1524                    if (userName == null) {
1525                            for(CachedMasterKey cachedMasterKey : cache_userName2cachedMasterKey.values())
1526                                    cachedMasterKey.clear();
1527    
1528                            cache_userName2cachedMasterKey.clear();
1529                    }
1530                    else {
1531                            CachedMasterKey cachedMasterKey = cache_userName2cachedMasterKey.remove(userName);
1532                            if (cachedMasterKey != null)
1533                                    cachedMasterKey.clear();
1534                    }
1535            }
1536    
1537            @Override
1538            protected void finalize() throws Throwable {
1539                    clearCache(null);
1540                    super.finalize();
1541            }
1542    }