Author: rgoers
Date: Tue Aug 11 08:12:27 2009
New Revision: 803022

URL: http://svn.apache.org/viewvc?rev=803022&view=rev
Log:
Allow passwords in the url to be encrypted

Added:
    
commons/proper/vfs/trunk/core/src/main/java/org/apache/commons/vfs/util/EncryptDecrypt.java
    
commons/proper/vfs/trunk/core/src/test/java/org/apache/commons/vfs/util/EncryptDecryptTest.java
Modified:
    
commons/proper/vfs/trunk/core/src/main/java/org/apache/commons/vfs/provider/HostFileNameParser.java
    
commons/proper/vfs/trunk/core/src/main/java/org/apache/commons/vfs/provider/http/HttpClientFactory.java

Modified: 
commons/proper/vfs/trunk/core/src/main/java/org/apache/commons/vfs/provider/HostFileNameParser.java
URL: 
http://svn.apache.org/viewvc/commons/proper/vfs/trunk/core/src/main/java/org/apache/commons/vfs/provider/HostFileNameParser.java?rev=803022&r1=803021&r2=803022&view=diff
==============================================================================
--- 
commons/proper/vfs/trunk/core/src/main/java/org/apache/commons/vfs/provider/HostFileNameParser.java
 (original)
+++ 
commons/proper/vfs/trunk/core/src/main/java/org/apache/commons/vfs/provider/HostFileNameParser.java
 Tue Aug 11 08:12:27 2009
@@ -19,6 +19,7 @@
 import org.apache.commons.vfs.FileName;
 import org.apache.commons.vfs.FileSystemException;
 import org.apache.commons.vfs.FileType;
+import org.apache.commons.vfs.util.EncryptDecrypt;
 
 /**
  * Implementation for any url based filesystem.<br />
@@ -122,6 +123,18 @@
         auth.userName = UriParser.decode(userName);
         auth.password = UriParser.decode(password);
 
+        if (auth.password != null && auth.password.startsWith("{") && 
auth.password.endsWith("}"))
+        {
+            try
+            {
+                auth.password = 
EncryptDecrypt.decrypt(auth.password.substring(1, auth.password.length()-1));
+            }
+            catch (Exception ex)
+            {
+                throw new FileSystemException("Unable to decrypt password", 
ex);
+            }
+        }
+
         // Extract hostname, and normalise (lowercase)
         final String hostName = extractHostName(name);
         if (hostName == null)

Modified: 
commons/proper/vfs/trunk/core/src/main/java/org/apache/commons/vfs/provider/http/HttpClientFactory.java
URL: 
http://svn.apache.org/viewvc/commons/proper/vfs/trunk/core/src/main/java/org/apache/commons/vfs/provider/http/HttpClientFactory.java?rev=803022&r1=803021&r2=803022&view=diff
==============================================================================
--- 
commons/proper/vfs/trunk/core/src/main/java/org/apache/commons/vfs/provider/http/HttpClientFactory.java
 (original)
+++ 
commons/proper/vfs/trunk/core/src/main/java/org/apache/commons/vfs/provider/http/HttpClientFactory.java
 Tue Aug 11 08:12:27 2009
@@ -74,7 +74,7 @@
                 String proxyHost = builder.getProxyHost(fileSystemOptions);
                 int proxyPort = builder.getProxyPort(fileSystemOptions);
 
-                if (proxyHost != null && proxyPort > 0)
+                if (proxyHost != null && proxyHost.length() > 0 && proxyPort > 
0)
                 {
                     config.setProxy(proxyHost, proxyPort);
                 }

Added: 
commons/proper/vfs/trunk/core/src/main/java/org/apache/commons/vfs/util/EncryptDecrypt.java
URL: 
http://svn.apache.org/viewvc/commons/proper/vfs/trunk/core/src/main/java/org/apache/commons/vfs/util/EncryptDecrypt.java?rev=803022&view=auto
==============================================================================
--- 
commons/proper/vfs/trunk/core/src/main/java/org/apache/commons/vfs/util/EncryptDecrypt.java
 (added)
+++ 
commons/proper/vfs/trunk/core/src/main/java/org/apache/commons/vfs/util/EncryptDecrypt.java
 Tue Aug 11 08:12:27 2009
@@ -0,0 +1,149 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements.  See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License.  You may obtain a copy of the License at
+ *
+ *      http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package org.apache.commons.vfs.util;
+
+import javax.crypto.Cipher;
+import javax.crypto.spec.SecretKeySpec;
+
+/**
+ * Allows passwords to be encrypted and decrypted.
+ */
+public class EncryptDecrypt
+{
+    private static char[] hexChars =
+            {'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 
'D', 'E', 'F'};
+
+    private static byte[] keyBytes = {0x41, 0x70, 0x61, 0x63, 0x68, 0x65, 
0x43, 0x6F, 0x6D, 0x6D,
+            0x6F, 0x6E, 0x73, 0x56, 0x46, 0x53};
+
+    private static int INDEX_NOT_FOUND = -1;
+
+    /**
+     * This class can be called with "encrypt" password as the arguments where 
encrypt is
+     * a literal and password is replaced with the clear text password to be 
encrypted.
+     *
+     * @param args The arguments in the form "command" parm1, parm2.
+     * @throws Exception If an error occurs.
+     */
+    public static void main(String[] args) throws Exception
+    {
+        if (args.length != 2 || !(args[0].equals("encrypt")))
+        {
+            System.err.println("Usage: \"EncryptDecrypt encrypt\" password");
+            System.err.println("     password : The clear text password to 
encrypt");
+            System.exit(0);
+        }
+
+        if (args[0].equals("encrypt"))
+        {
+            System.out.println(encrypt(args[1]));
+        }
+    }
+
+    /**
+     * Encrypt the plain text password.
+     * @param plainKey The password.
+     * @return The encrypted password String.
+     * @throws Exception If an error occurs.
+     */
+    public static String encrypt(String plainKey) throws Exception
+    {
+        byte[] input = plainKey.getBytes();
+        SecretKeySpec key = new SecretKeySpec(keyBytes, "AES");
+
+        Cipher cipher = Cipher.getInstance("AES");
+
+        // encryption pass
+        cipher.init(Cipher.ENCRYPT_MODE, key);
+
+        byte[] cipherText = new byte[cipher.getOutputSize(input.length)];
+        int ctLength = cipher.update(input, 0, input.length, cipherText, 0);
+        ctLength += cipher.doFinal(cipherText, ctLength);
+        return encode(cipherText);
+    }
+
+    /**
+     * Decrypts the password.
+     * @param encryptedKey the encrypted password.
+     * @return The plain text password.
+     * @throws Exception If an error occurs.
+     */
+    public static String decrypt(String encryptedKey) throws Exception
+    {
+        SecretKeySpec key = new SecretKeySpec(keyBytes, "AES");
+        Cipher cipher = Cipher.getInstance("AES");
+        cipher.init(Cipher.DECRYPT_MODE, key);
+        byte[] decoded = decode(encryptedKey);
+        byte[] plainText = new byte[cipher.getOutputSize(decoded.length)];
+        int ptLength = cipher.update(decoded, 0, decoded.length, plainText, 0);
+        ptLength += cipher.doFinal(plainText, ptLength);
+        return new String(plainText).substring(0, ptLength);
+    }
+
+    private static String encode(byte[] bytes)
+    {
+        StringBuffer builder = new StringBuffer();
+
+        for (int i=0; i < bytes.length; ++i)
+        {
+            builder.append(hexChars[(bytes[i] >> 4) & 0x0f]);
+            builder.append(hexChars[bytes[i] & 0x0f]);
+        }
+        return builder.toString();
+    }
+
+    private static byte[] decode(String str)
+    {
+        int length = str.length() / 2;
+        byte[] decoded = new byte[length];
+        char[] chars = str.toCharArray();
+        int index = 0;
+        for (int i = 0; i < chars.length; ++i)
+        {
+            int id1 = indexOf(hexChars, chars[i]);
+            if (id1 == -1)
+            {
+                throw new IllegalArgumentException("Character " + chars[i] + " 
at position " + i +
+                        " is not a valid hexideciam character");
+            }
+            int id2 = indexOf(hexChars, chars[++i]);
+            if (id2 == -1)
+            {
+                throw new IllegalArgumentException("Character " + chars[i] + " 
at position " + i +
+                        " is not a valid hexideciam character");
+            }
+            decoded[index++] = (byte) ((id1 << 4) | id2);
+        }
+        return decoded;
+    }
+
+    private static int indexOf(char[] array, char valueToFind)
+    {
+        if (array == null)
+        {
+            return INDEX_NOT_FOUND;
+        }
+        for (int i = 0; i < array.length; i++)
+        {
+            if (valueToFind == array[i])
+            {
+                return i;
+            }
+        }
+        return INDEX_NOT_FOUND;
+    }
+}

Added: 
commons/proper/vfs/trunk/core/src/test/java/org/apache/commons/vfs/util/EncryptDecryptTest.java
URL: 
http://svn.apache.org/viewvc/commons/proper/vfs/trunk/core/src/test/java/org/apache/commons/vfs/util/EncryptDecryptTest.java?rev=803022&view=auto
==============================================================================
--- 
commons/proper/vfs/trunk/core/src/test/java/org/apache/commons/vfs/util/EncryptDecryptTest.java
 (added)
+++ 
commons/proper/vfs/trunk/core/src/test/java/org/apache/commons/vfs/util/EncryptDecryptTest.java
 Tue Aug 11 08:12:27 2009
@@ -0,0 +1,39 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements.  See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License.  You may obtain a copy of the License at
+ *
+ *      http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+package org.apache.commons.vfs.util;
+
+import junit.framework.TestCase;
+
+/**
+ *
+ */
+public class EncryptDecryptTest extends TestCase
+{
+
+    public void testEncryptDecrypt() throws Exception
+    {
+        String source = "Qryp2!t&tpR59";
+        String expected = "914589F049CE3E8E3BB1A41BEAE12A9C";
+
+        String encrypted = EncryptDecrypt.encrypt(source);
+        assertEquals(expected, encrypted);
+
+        String decrypted = EncryptDecrypt.decrypt(encrypted);
+        assertEquals(source, decrypted);
+
+    }
+}


Reply via email to