#!/usr/bin/env compileAndGo
# -*- coding: utf-8 mode: java -*- vim:sw=4:sts=4:et:ai:si:sta:fenc=utf-8
compiler=javac
mainClass=upassword
compileAndGo

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.PrintWriter;
import java.io.StringWriter;
import java.io.UnsupportedEncodingException;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.security.SecureRandom;
import java.util.Arrays;
import java.util.List;
import java.util.Random;
import java.util.regex.Pattern;

import javax.crypto.Cipher;
import javax.crypto.KeyGenerator;
import javax.crypto.SecretKey;
import javax.crypto.SecretKeyFactory;
import javax.crypto.spec.PBEKeySpec;
import javax.crypto.spec.SecretKeySpec;

public class upassword {
    public static final String UTF_8 = "UTF-8";

    public static final int MIN_LEN = 8;

    public static final int MIN_UPPER = 0;

    public static final int MIN_LOWER = 0;

    public static final int MIN_ALPHA = 2;

    public static final int MIN_NUMBER = 0;

    public static final int MIN_SYMBOL = 0;

    public static final int MIN_SPECIAL = 2;

    public static final boolean ALLOW_MULTIBYTES = false;

    public static final String getSummary(Exception e) {
        StringWriter sw = new StringWriter();
        PrintWriter pw = new PrintWriter(sw);
        e.printStackTrace(pw);
        pw.flush();
        return sw.toString();
    }

    public static final void println(String text) {
        System.out.println(text);
        System.out.flush();
    }

    public static final StringBuilder toHex(StringBuilder sb, byte b) {
        if (sb == null) return null;
        int l = b & 0x0F, u = (b & 0xF0) >> 4;
        u += u < 10? '0': 'A' - 10;
        l += l < 10? '0': 'A' - 10;
        return sb.append((char)u).append((char)l);
    }

    public static final String toHex(byte[] ba) {
        if (ba == null) return null;
        StringBuilder sb = new StringBuilder(2 * ba.length);
        for (int i = 0; i < ba.length; i++) {
            toHex(sb, ba[i]);
        }
        return sb.toString();
    }

    public static final boolean isHex(String str) {
        int l = str.length();
        if (l % 2 != 0) return false;
        for (int i = 0; i < l; i++) {
            char c = str.charAt(i);
            if (!(c >= '0' && c <= '9' || c >= 'A' && c <= 'F' || c >= 'a' && c <= 'f')) {
                return false;
            }
        }
        return true;
    }

    public static final byte[] fromHex(String str) {
        if (str == null) return null;
        int l = str.length();
        byte[] ba = new byte[l / 2];
        int j = 0;
        for (int i = 0; i < l; i += 2) {
            byte b = (byte)Short.parseShort(str.substring(i, i + 2), 16);
            ba[j++] = b;
        }
        return ba;
    }

    public static final boolean strIsempty(String str) {
        return str == null || str.length() == 0;
    }

    public static final boolean strEquals(String s1, String s2) {
        return s1 == s2 || (s1 != null && s1.equals(s2));
    }

    public static final String strNotnull(String str) {
        return str == null? "": str;
    }

    public static final String strSubstr(String str, int start, int end) {
        if (str == null) return null;

        int l = str.length();
        if (l > 0) {
            while (start < 0)
                start += l;
            while (end < 0)
                end += l;
        } else {
            if (start < 0) start = 0;
            if (end < 0) end = 0;
        }
        if (start >= l || start >= end) return "";
        if (end >= l + 1) end = l;

        return str.substring(start, end);
    }

    public static final String strSubstr(String str, int start) {
        if (str == null) return null;
        return strSubstr(str, start, str.length());
    }

    // ------------------------------------------------------------------------
    public static class Base64 {
        public final static int NO_OPTIONS = 0;

        public final static int ENCODE = 1;

        public final static int DECODE = 0;

        public final static int GZIP = 2;

        public final static int DONT_BREAK_LINES = 8;

        private final static int MAX_LINE_LENGTH = 76;

        private final static byte EQUALS_SIGN = (byte)'=';

        private final static byte NEW_LINE = (byte)'\n';

        private final static String PREFERRED_ENCODING = "UTF-8";

        private final static byte[] ALPHABET;

        private final static byte[] _NATIVE_ALPHABET = {
                (byte)'A',
                (byte)'B',
                (byte)'C',
                (byte)'D',
                (byte)'E',
                (byte)'F',
                (byte)'G',
                (byte)'H',
                (byte)'I',
                (byte)'J',
                (byte)'K',
                (byte)'L',
                (byte)'M',
                (byte)'N',
                (byte)'O',
                (byte)'P',
                (byte)'Q',
                (byte)'R',
                (byte)'S',
                (byte)'T',
                (byte)'U',
                (byte)'V',
                (byte)'W',
                (byte)'X',
                (byte)'Y',
                (byte)'Z',
                (byte)'a',
                (byte)'b',
                (byte)'c',
                (byte)'d',
                (byte)'e',
                (byte)'f',
                (byte)'g',
                (byte)'h',
                (byte)'i',
                (byte)'j',
                (byte)'k',
                (byte)'l',
                (byte)'m',
                (byte)'n',
                (byte)'o',
                (byte)'p',
                (byte)'q',
                (byte)'r',
                (byte)'s',
                (byte)'t',
                (byte)'u',
                (byte)'v',
                (byte)'w',
                (byte)'x',
                (byte)'y',
                (byte)'z',
                (byte)'0',
                (byte)'1',
                (byte)'2',
                (byte)'3',
                (byte)'4',
                (byte)'5',
                (byte)'6',
                (byte)'7',
                (byte)'8',
                (byte)'9',
                (byte)'+',
                (byte)'/'};
        static {
            byte[] __bytes;
            try {
                __bytes = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"
                        .getBytes(PREFERRED_ENCODING);
            } catch (java.io.UnsupportedEncodingException use) {
                __bytes = _NATIVE_ALPHABET;
            }
            ALPHABET = __bytes;
        }

        private final static byte[] DECODABET = {
                -9,
                -9,
                -9,
                -9,
                -9,
                -9,
                -9,
                -9,
                -9,
                -5,
                -5,
                -9,
                -9,
                -5,
                -9,
                -9,
                -9,
                -9,
                -9,
                -9,
                -9,
                -9,
                -9,
                -9,
                -9,
                -9,
                -9,
                -9,
                -9,
                -9,
                -9,
                -9,
                -5,
                -9,
                -9,
                -9,
                -9,
                -9,
                -9,
                -9,
                -9,
                -9,
                -9,
                62,
                -9,
                -9,
                -9,
                63,
                52,
                53,
                54,
                55,
                56,
                57,
                58,
                59,
                60,
                61,
                -9,
                -9,
                -9,
                -1,
                -9,
                -9,
                -9,
                0,
                1,
                2,
                3,
                4,
                5,
                6,
                7,
                8,
                9,
                10,
                11,
                12,
                13,
                14,
                15,
                16,
                17,
                18,
                19,
                20,
                21,
                22,
                23,
                24,
                25,
                -9,
                -9,
                -9,
                -9,
                -9,
                -9,
                26,
                27,
                28,
                29,
                30,
                31,
                32,
                33,
                34,
                35,
                36,
                37,
                38,
                39,
                40,
                41,
                42,
                43,
                44,
                45,
                46,
                47,
                48,
                49,
                50,
                51,
                -9,
                -9,
                -9,
                -9};

        private final static byte WHITE_SPACE_ENC = -5;

        private final static byte EQUALS_SIGN_ENC = -1;

        private Base64() {
        }

        private static byte[] encode3to4(byte[] b4, byte[] threeBytes, int numSigBytes) {
            encode3to4(threeBytes, 0, numSigBytes, b4, 0);
            return b4;
        }

        private static byte[] encode3to4(byte[] source, int srcOffset, int numSigBytes,
                byte[] destination, int destOffset) {
            int inBuff = (numSigBytes > 0? ((source[srcOffset] << 24) >>> 8): 0)
                    | (numSigBytes > 1? ((source[srcOffset + 1] << 24) >>> 16): 0)
                    | (numSigBytes > 2? ((source[srcOffset + 2] << 24) >>> 24): 0);
            switch (numSigBytes) {
            case 3:
                destination[destOffset] = ALPHABET[(inBuff >>> 18)];
                destination[destOffset + 1] = ALPHABET[(inBuff >>> 12) & 0x3f];
                destination[destOffset + 2] = ALPHABET[(inBuff >>> 6) & 0x3f];
                destination[destOffset + 3] = ALPHABET[(inBuff) & 0x3f];
                return destination;
            case 2:
                destination[destOffset] = ALPHABET[(inBuff >>> 18)];
                destination[destOffset + 1] = ALPHABET[(inBuff >>> 12) & 0x3f];
                destination[destOffset + 2] = ALPHABET[(inBuff >>> 6) & 0x3f];
                destination[destOffset + 3] = EQUALS_SIGN;
                return destination;
            case 1:
                destination[destOffset] = ALPHABET[(inBuff >>> 18)];
                destination[destOffset + 1] = ALPHABET[(inBuff >>> 12) & 0x3f];
                destination[destOffset + 2] = EQUALS_SIGN;
                destination[destOffset + 3] = EQUALS_SIGN;
                return destination;
            default:
                return destination;
            }
        }

        public static String encodeObject(java.io.Serializable serializableObject) {
            return encodeObject(serializableObject, NO_OPTIONS);
        }

        public static String encodeObject(java.io.Serializable serializableObject, int options) {
            java.io.ByteArrayOutputStream baos = null;
            java.io.OutputStream b64os = null;
            java.io.ObjectOutputStream oos = null;
            java.util.zip.GZIPOutputStream gzos = null;
            int gzip = (options & GZIP);
            int dontBreakLines = (options & DONT_BREAK_LINES);
            try {
                baos = new java.io.ByteArrayOutputStream();
                b64os = new Base64.OutputStream(baos, ENCODE | dontBreakLines);
                if (gzip == GZIP) {
                    gzos = new java.util.zip.GZIPOutputStream(b64os);
                    oos = new java.io.ObjectOutputStream(gzos);
                } else oos = new java.io.ObjectOutputStream(b64os);
                oos.writeObject(serializableObject);
            } catch (java.io.IOException e) {
                e.printStackTrace();
                return null;
            } finally {
                try {
                    oos.close();
                } catch (Exception e) {
                }
                try {
                    gzos.close();
                } catch (Exception e) {
                }
                try {
                    b64os.close();
                } catch (Exception e) {
                }
                try {
                    baos.close();
                } catch (Exception e) {
                }
            }
            try {
                return new String(baos.toByteArray(), PREFERRED_ENCODING);
            } catch (java.io.UnsupportedEncodingException uue) {
                return new String(baos.toByteArray());
            }
        }

        public static String encodeBytes(byte[] source) {
            return encodeBytes(source, 0, source.length, NO_OPTIONS);
        }

        public static String encodeBytes(byte[] source, int options) {
            return encodeBytes(source, 0, source.length, options);
        }

        public static String encodeBytes(byte[] source, int off, int len) {
            return encodeBytes(source, off, len, NO_OPTIONS);
        }

        public static String encodeBytes(byte[] source, int off, int len, int options) {
            int dontBreakLines = (options & DONT_BREAK_LINES);
            int gzip = (options & GZIP);
            if (gzip == GZIP) {
                java.io.ByteArrayOutputStream baos = null;
                java.util.zip.GZIPOutputStream gzos = null;
                Base64.OutputStream b64os = null;
                try {
                    baos = new java.io.ByteArrayOutputStream();
                    b64os = new Base64.OutputStream(baos, ENCODE | dontBreakLines);
                    gzos = new java.util.zip.GZIPOutputStream(b64os);
                    gzos.write(source, off, len);
                    gzos.close();
                } catch (java.io.IOException e) {
                    e.printStackTrace();
                    return null;
                } finally {
                    try {
                        gzos.close();
                    } catch (Exception e) {
                    }
                    try {
                        b64os.close();
                    } catch (Exception e) {
                    }
                    try {
                        baos.close();
                    } catch (Exception e) {
                    }
                }
                try {
                    return new String(baos.toByteArray(), PREFERRED_ENCODING);
                } catch (java.io.UnsupportedEncodingException uue) {
                    return new String(baos.toByteArray());
                }
            } else {
                boolean breakLines = dontBreakLines == 0;
                int len43 = len * 4 / 3;
                byte[] outBuff = new byte[(len43) + ((len % 3) > 0? 4: 0)
                        + (breakLines? (len43 / MAX_LINE_LENGTH): 0)];
                int d = 0;
                int e = 0;
                int len2 = len - 2;
                int lineLength = 0;
                for (; d < len2; d += 3, e += 4) {
                    encode3to4(source, d + off, 3, outBuff, e);
                    lineLength += 4;
                    if (breakLines && lineLength == MAX_LINE_LENGTH) {
                        outBuff[e + 4] = NEW_LINE;
                        e++;
                        lineLength = 0;
                    }
                }
                if (d < len) {
                    encode3to4(source, d + off, len - d, outBuff, e);
                    e += 4;
                }
                try {
                    return new String(outBuff, 0, e, PREFERRED_ENCODING);
                } catch (java.io.UnsupportedEncodingException uue) {
                    return new String(outBuff, 0, e);
                }
            }
        }

        private static int decode4to3(byte[] source, int srcOffset, byte[] destination,
                int destOffset) {
            if (source[srcOffset + 2] == EQUALS_SIGN) {
                int outBuff = ((DECODABET[source[srcOffset]] & 0xFF) << 18)
                        | ((DECODABET[source[srcOffset + 1]] & 0xFF) << 12);
                destination[destOffset] = (byte)(outBuff >>> 16);
                return 1;
            } else if (source[srcOffset + 3] == EQUALS_SIGN) {
                int outBuff = ((DECODABET[source[srcOffset]] & 0xFF) << 18)
                        | ((DECODABET[source[srcOffset + 1]] & 0xFF) << 12)
                        | ((DECODABET[source[srcOffset + 2]] & 0xFF) << 6);
                destination[destOffset] = (byte)(outBuff >>> 16);
                destination[destOffset + 1] = (byte)(outBuff >>> 8);
                return 2;
            } else {
                try {
                    int outBuff = ((DECODABET[source[srcOffset]] & 0xFF) << 18)
                            | ((DECODABET[source[srcOffset + 1]] & 0xFF) << 12)
                            | ((DECODABET[source[srcOffset + 2]] & 0xFF) << 6)
                            | ((DECODABET[source[srcOffset + 3]] & 0xFF));
                    destination[destOffset] = (byte)(outBuff >> 16);
                    destination[destOffset + 1] = (byte)(outBuff >> 8);
                    destination[destOffset + 2] = (byte)(outBuff);
                    return 3;
                } catch (Exception e) {
                    System.out.println("" + source[srcOffset] + ": "
                            + (DECODABET[source[srcOffset]]));
                    System.out.println("" + source[srcOffset + 1] + ": "
                            + (DECODABET[source[srcOffset + 1]]));
                    System.out.println("" + source[srcOffset + 2] + ": "
                            + (DECODABET[source[srcOffset + 2]]));
                    System.out.println("" + source[srcOffset + 3] + ": "
                            + (DECODABET[source[srcOffset + 3]]));
                    return -1;
                }
            }
        }

        public static byte[] decode(byte[] source, int off, int len) {
            int len34 = len * 3 / 4;
            byte[] outBuff = new byte[len34];
            int outBuffPosn = 0;
            byte[] b4 = new byte[4];
            int b4Posn = 0;
            int i = 0;
            byte sbiCrop = 0;
            byte sbiDecode = 0;
            for (i = off; i < off + len; i++) {
                sbiCrop = (byte)(source[i] & 0x7f);
                sbiDecode = DECODABET[sbiCrop];
                if (sbiDecode >= WHITE_SPACE_ENC) {
                    if (sbiDecode >= EQUALS_SIGN_ENC) {
                        b4[b4Posn++] = sbiCrop;
                        if (b4Posn > 3) {
                            outBuffPosn += decode4to3(b4, 0, outBuff, outBuffPosn);
                            b4Posn = 0;
                            if (sbiCrop == EQUALS_SIGN) break;
                        }
                    }
                } else {
                    throw new IllegalArgumentException("Bad Base64 input character at " + i + ": "
                            + source[i] + "(decimal)");
                }
            }
            byte[] out = new byte[outBuffPosn];
            System.arraycopy(outBuff, 0, out, 0, outBuffPosn);
            return out;
        }

        public static byte[] decode(String s) {
            byte[] bytes;
            try {
                bytes = s.getBytes(PREFERRED_ENCODING);
            } catch (java.io.UnsupportedEncodingException uee) {
                bytes = s.getBytes();
            }
            bytes = decode(bytes, 0, bytes.length);
            if (bytes != null && bytes.length >= 4) {
                int head = ((int)bytes[0] & 0xff) | ((bytes[1] << 8) & 0xff00);
                if (java.util.zip.GZIPInputStream.GZIP_MAGIC == head) {
                    java.io.ByteArrayInputStream bais = null;
                    java.util.zip.GZIPInputStream gzis = null;
                    java.io.ByteArrayOutputStream baos = null;
                    byte[] buffer = new byte[2048];
                    int length = 0;
                    try {
                        baos = new java.io.ByteArrayOutputStream();
                        bais = new java.io.ByteArrayInputStream(bytes);
                        gzis = new java.util.zip.GZIPInputStream(bais);
                        while ((length = gzis.read(buffer)) >= 0) {
                            baos.write(buffer, 0, length);
                        }
                        bytes = baos.toByteArray();
                    } catch (java.io.IOException e) {
                    } finally {
                        try {
                            baos.close();
                        } catch (Exception e) {
                        }
                        try {
                            gzis.close();
                        } catch (Exception e) {
                        }
                        try {
                            bais.close();
                        } catch (Exception e) {
                        }
                    }
                }
            }
            return bytes;
        }

        public static Object decodeToObject(String encodedObject) {
            byte[] objBytes = decode(encodedObject);
            java.io.ByteArrayInputStream bais = null;
            java.io.ObjectInputStream ois = null;
            Object obj = null;
            try {
                bais = new java.io.ByteArrayInputStream(objBytes);
                ois = new java.io.ObjectInputStream(bais);
                obj = ois.readObject();
            } catch (java.io.IOException e) {
                e.printStackTrace();
                obj = null;
            } catch (java.lang.ClassNotFoundException e) {
                e.printStackTrace();
                obj = null;
            } finally {
                try {
                    bais.close();
                } catch (Exception e) {
                }
                try {
                    ois.close();
                } catch (Exception e) {
                }
            }
            return obj;
        }

        public static boolean encodeToFile(byte[] dataToEncode, String filename) {
            boolean success = false;
            Base64.OutputStream bos = null;
            try {
                bos = new Base64.OutputStream(new java.io.FileOutputStream(filename), Base64.ENCODE);
                bos.write(dataToEncode);
                success = true;
            } catch (java.io.IOException e) {
                success = false;
            } finally {
                try {
                    bos.close();
                } catch (Exception e) {
                }
            }
            return success;
        }

        public static boolean decodeToFile(String dataToDecode, String filename) {
            boolean success = false;
            Base64.OutputStream bos = null;
            try {
                bos = new Base64.OutputStream(new java.io.FileOutputStream(filename), Base64.DECODE);
                bos.write(dataToDecode.getBytes(PREFERRED_ENCODING));
                success = true;
            } catch (java.io.IOException e) {
                success = false;
            } finally {
                try {
                    bos.close();
                } catch (Exception e) {
                }
            }
            return success;
        }

        public static byte[] decodeFromFile(String filename) {
            byte[] decodedData = null;
            Base64.InputStream bis = null;
            try {
                java.io.File file = new java.io.File(filename);
                byte[] buffer = null;
                int length = 0;
                int numBytes = 0;
                if (file.length() > Integer.MAX_VALUE) {
                    throw new IllegalArgumentException(
                            "File is too big for this convenience method (" + file.length()
                                    + " bytes).");
                }
                buffer = new byte[(int)file.length()];
                bis = new Base64.InputStream(new java.io.BufferedInputStream(
                        new java.io.FileInputStream(file)), Base64.DECODE);
                while ((numBytes = bis.read(buffer, length, 4096)) >= 0)
                    length += numBytes;
                decodedData = new byte[length];
                System.arraycopy(buffer, 0, decodedData, 0, length);
            } catch (java.io.IOException e) {
                throw new IllegalArgumentException("Error decoding from file " + filename);
            } finally {
                try {
                    bis.close();
                } catch (Exception e) {
                }
            }
            return decodedData;
        }

        public static String encodeFromFile(String filename) {
            String encodedData = null;
            Base64.InputStream bis = null;
            try {
                java.io.File file = new java.io.File(filename);
                byte[] buffer = new byte[(int)(file.length() * 1.4)];
                int length = 0;
                int numBytes = 0;
                bis = new Base64.InputStream(new java.io.BufferedInputStream(
                        new java.io.FileInputStream(file)), Base64.ENCODE);
                while ((numBytes = bis.read(buffer, length, 4096)) >= 0)
                    length += numBytes;
                encodedData = new String(buffer, 0, length, Base64.PREFERRED_ENCODING);
            } catch (java.io.IOException e) {
                System.err.println("Error encoding from file " + filename);
            } finally {
                try {
                    bis.close();
                } catch (Exception e) {
                }
            }
            return encodedData;
        }

        public static class InputStream extends java.io.FilterInputStream {
            private boolean encode;

            private int position;

            private byte[] buffer;

            private int bufferLength;

            private int numSigBytes;

            private int lineLength;

            private boolean breakLines;

            public InputStream(java.io.InputStream in) {
                this(in, DECODE);
            }

            public InputStream(java.io.InputStream in, int options) {
                super(in);
                this.breakLines = (options & DONT_BREAK_LINES) != DONT_BREAK_LINES;
                this.encode = (options & ENCODE) == ENCODE;
                this.bufferLength = encode? 4: 3;
                this.buffer = new byte[bufferLength];
                this.position = -1;
                this.lineLength = 0;
            }

            public int read() throws java.io.IOException {
                if (position < 0) {
                    if (encode) {
                        byte[] b3 = new byte[3];
                        int numBinaryBytes = 0;
                        for (int i = 0; i < 3; i++) {
                            try {
                                int b = in.read();
                                if (b >= 0) {
                                    b3[i] = (byte)b;
                                    numBinaryBytes++;
                                }
                            } catch (java.io.IOException e) {
                                if (i == 0) throw e;
                            }
                        }
                        if (numBinaryBytes > 0) {
                            encode3to4(b3, 0, numBinaryBytes, buffer, 0);
                            position = 0;
                            numSigBytes = 4;
                        } else {
                            return -1;
                        }
                    } else {
                        byte[] b4 = new byte[4];
                        int i = 0;
                        for (i = 0; i < 4; i++) {
                            int b = 0;
                            do {
                                b = in.read();
                            } while (b >= 0 && DECODABET[b & 0x7f] <= WHITE_SPACE_ENC);
                            if (b < 0) break;
                            b4[i] = (byte)b;
                        }
                        if (i == 4) {
                            numSigBytes = decode4to3(b4, 0, buffer, 0);
                            position = 0;
                        } else if (i == 0) {
                            return -1;
                        } else {
                            throw new java.io.IOException("Improperly padded Base64 input.");
                        }
                    }
                }
                if (position >= 0) {
                    if ( /* !encode && */position >= numSigBytes) return -1;
                    if (encode && breakLines && lineLength >= MAX_LINE_LENGTH) {
                        lineLength = 0;
                        return '\n';
                    } else {
                        lineLength++;
                        int b = buffer[position++];
                        if (position >= bufferLength) position = -1;
                        return b & 0xFF;
                    }
                } else {
                    throw new java.io.IOException("Error in Base64 code reading stream.");
                }
            }

            public int read(byte[] dest, int off, int len) throws java.io.IOException {
                int i;
                int b;
                for (i = 0; i < len; i++) {
                    b = read();
                    if (b >= 0) dest[off + i] = (byte)b;
                    else if (i == 0) return -1;
                    else break;
                }
                return i;
            }
        }

        public static class OutputStream extends java.io.FilterOutputStream {
            private boolean encode;

            private int position;

            private byte[] buffer;

            private int bufferLength;

            private int lineLength;

            private boolean breakLines;

            private byte[] b4;

            private boolean suspendEncoding;

            public OutputStream(java.io.OutputStream out) {
                this(out, ENCODE);
            }

            public OutputStream(java.io.OutputStream out, int options) {
                super(out);
                this.breakLines = (options & DONT_BREAK_LINES) != DONT_BREAK_LINES;
                this.encode = (options & ENCODE) == ENCODE;
                this.bufferLength = encode? 3: 4;
                this.buffer = new byte[bufferLength];
                this.position = 0;
                this.lineLength = 0;
                this.suspendEncoding = false;
                this.b4 = new byte[4];
            }

            public void write(int theByte) throws java.io.IOException {
                if (suspendEncoding) {
                    super.out.write(theByte);
                    return;
                }
                if (encode) {
                    buffer[position++] = (byte)theByte;
                    if (position >= bufferLength) {
                        out.write(encode3to4(b4, buffer, bufferLength));
                        lineLength += 4;
                        if (breakLines && lineLength >= MAX_LINE_LENGTH) {
                            out.write(NEW_LINE);
                            lineLength = 0;
                        }
                        position = 0;
                    }
                } else {
                    if (DECODABET[theByte & 0x7f] > WHITE_SPACE_ENC) {
                        buffer[position++] = (byte)theByte;
                        if (position >= bufferLength) {
                            int len = Base64.decode4to3(buffer, 0, b4, 0);
                            out.write(b4, 0, len);
                            position = 0;
                        }
                    } else if (DECODABET[theByte & 0x7f] != WHITE_SPACE_ENC) {
                        throw new java.io.IOException("Invalid character in Base64 data.");
                    }
                }
            }

            public void write(byte[] theBytes, int off, int len) throws java.io.IOException {
                if (suspendEncoding) {
                    super.out.write(theBytes, off, len);
                    return;
                }
                for (int i = 0; i < len; i++) {
                    write(theBytes[off + i]);
                }
            }

            public void flushBase64() throws java.io.IOException {
                if (position > 0) {
                    if (encode) {
                        out.write(encode3to4(b4, buffer, position));
                        position = 0;
                    } else {
                        throw new java.io.IOException("Base64 input not properly padded.");
                    }
                }
            }

            public void close() throws java.io.IOException {
                flushBase64();
                super.close();
                buffer = null;
                out = null;
            }

            public void suspendEncoding() throws java.io.IOException {
                flushBase64();
                this.suspendEncoding = true;
            }

            public void resumeEncoding() {
                this.suspendEncoding = false;
            }
        }
    }

    // ------------------------------------------------------------------------
    public static class DES {
        private int[] encryptKeys = new int[32];

        private int[] decryptKeys = new int[32];

        private int[] tempInts = new int[2];

        public DES() {
        }

        public DES(byte[] key) {
            if (key.length == 7) {
                byte[] key8 = new byte[8];
                makeSMBKey(key, key8);
                setKey(key8);
            } else {
                setKey(key);
            }
        }

        public static void makeSMBKey(byte[] key7, byte[] key8) {
            int i;
            key8[0] = (byte)((key7[0] >> 1) & 0xff);
            key8[1] = (byte)((((key7[0] & 0x01) << 6) | (((key7[1] & 0xff) >> 2) & 0xff)) & 0xff);
            key8[2] = (byte)((((key7[1] & 0x03) << 5) | (((key7[2] & 0xff) >> 3) & 0xff)) & 0xff);
            key8[3] = (byte)((((key7[2] & 0x07) << 4) | (((key7[3] & 0xff) >> 4) & 0xff)) & 0xff);
            key8[4] = (byte)((((key7[3] & 0x0F) << 3) | (((key7[4] & 0xff) >> 5) & 0xff)) & 0xff);
            key8[5] = (byte)((((key7[4] & 0x1F) << 2) | (((key7[5] & 0xff) >> 6) & 0xff)) & 0xff);
            key8[6] = (byte)((((key7[5] & 0x3F) << 1) | (((key7[6] & 0xff) >> 7) & 0xff)) & 0xff);
            key8[7] = (byte)(key7[6] & 0x7F);
            for (i = 0; i < 8; i++) {
                key8[i] = (byte)(key8[i] << 1);
            }
        }

        public void setKey(byte[] key) {
            deskey(key, true, encryptKeys);
            deskey(key, false, decryptKeys);
        }

        private void deskey(byte[] keyBlock, boolean encrypting, int[] KnL) {
            int i, j, l, m, n;
            int[] pc1m = new int[56];
            int[] pcr = new int[56];
            int[] kn = new int[32];
            for (j = 0; j < 56; ++j) {
                l = pc1[j];
                m = l & 07;
                pc1m[j] = ((keyBlock[l >>> 3] & bytebit[m]) != 0)? 1: 0;
            }
            for (i = 0; i < 16; ++i) {
                if (encrypting) m = i << 1;
                else m = (15 - i) << 1;
                n = m + 1;
                kn[m] = kn[n] = 0;
                for (j = 0; j < 28; ++j) {
                    l = j + totrot[i];
                    if (l < 28) pcr[j] = pc1m[l];
                    else pcr[j] = pc1m[l - 28];
                }
                for (j = 28; j < 56; ++j) {
                    l = j + totrot[i];
                    if (l < 56) pcr[j] = pc1m[l];
                    else pcr[j] = pc1m[l - 28];
                }
                for (j = 0; j < 24; ++j) {
                    if (pcr[pc2[j]] != 0) kn[m] |= bigbyte[j];
                    if (pcr[pc2[j + 24]] != 0) kn[n] |= bigbyte[j];
                }
            }
            cookey(kn, KnL);
        }

        private void cookey(int[] raw, int KnL[]) {
            int raw0, raw1;
            int rawi, KnLi;
            int i;
            for (i = 0, rawi = 0, KnLi = 0; i < 16; ++i) {
                raw0 = raw[rawi++];
                raw1 = raw[rawi++];
                KnL[KnLi] = (raw0 & 0x00fc0000) << 6;
                KnL[KnLi] |= (raw0 & 0x00000fc0) << 10;
                KnL[KnLi] |= (raw1 & 0x00fc0000) >>> 10;
                KnL[KnLi] |= (raw1 & 0x00000fc0) >>> 6;
                ++KnLi;
                KnL[KnLi] = (raw0 & 0x0003f000) << 12;
                KnL[KnLi] |= (raw0 & 0x0000003f) << 16;
                KnL[KnLi] |= (raw1 & 0x0003f000) >>> 4;
                KnL[KnLi] |= (raw1 & 0x0000003f);
                ++KnLi;
            }
        }

        private void encrypt(byte[] clearText, int clearOff, byte[] cipherText, int cipherOff) {
            squashBytesToInts(clearText, clearOff, tempInts, 0, 2);
            des(tempInts, tempInts, encryptKeys);
            spreadIntsToBytes(tempInts, 0, cipherText, cipherOff, 2);
        }

        private void decrypt(byte[] cipherText, int cipherOff, byte[] clearText, int clearOff) {
            squashBytesToInts(cipherText, cipherOff, tempInts, 0, 2);
            des(tempInts, tempInts, decryptKeys);
            spreadIntsToBytes(tempInts, 0, clearText, clearOff, 2);
        }

        private void des(int[] inInts, int[] outInts, int[] keys) {
            int fval, work, right, leftt;
            int round;
            int keysi = 0;
            leftt = inInts[0];
            right = inInts[1];
            work = ((leftt >>> 4) ^ right) & 0x0f0f0f0f;
            right ^= work;
            leftt ^= (work << 4);
            work = ((leftt >>> 16) ^ right) & 0x0000ffff;
            right ^= work;
            leftt ^= (work << 16);
            work = ((right >>> 2) ^ leftt) & 0x33333333;
            leftt ^= work;
            right ^= (work << 2);
            work = ((right >>> 8) ^ leftt) & 0x00ff00ff;
            leftt ^= work;
            right ^= (work << 8);
            right = (right << 1) | ((right >>> 31) & 1);
            work = (leftt ^ right) & 0xaaaaaaaa;
            leftt ^= work;
            right ^= work;
            leftt = (leftt << 1) | ((leftt >>> 31) & 1);
            for (round = 0; round < 8; ++round) {
                work = (right << 28) | (right >>> 4);
                work ^= keys[keysi++];
                fval = SP7[work & 0x0000003f];
                fval |= SP5[(work >>> 8) & 0x0000003f];
                fval |= SP3[(work >>> 16) & 0x0000003f];
                fval |= SP1[(work >>> 24) & 0x0000003f];
                work = right ^ keys[keysi++];
                fval |= SP8[work & 0x0000003f];
                fval |= SP6[(work >>> 8) & 0x0000003f];
                fval |= SP4[(work >>> 16) & 0x0000003f];
                fval |= SP2[(work >>> 24) & 0x0000003f];
                leftt ^= fval;
                work = (leftt << 28) | (leftt >>> 4);
                work ^= keys[keysi++];
                fval = SP7[work & 0x0000003f];
                fval |= SP5[(work >>> 8) & 0x0000003f];
                fval |= SP3[(work >>> 16) & 0x0000003f];
                fval |= SP1[(work >>> 24) & 0x0000003f];
                work = leftt ^ keys[keysi++];
                fval |= SP8[work & 0x0000003f];
                fval |= SP6[(work >>> 8) & 0x0000003f];
                fval |= SP4[(work >>> 16) & 0x0000003f];
                fval |= SP2[(work >>> 24) & 0x0000003f];
                right ^= fval;
            }
            right = (right << 31) | (right >>> 1);
            work = (leftt ^ right) & 0xaaaaaaaa;
            leftt ^= work;
            right ^= work;
            leftt = (leftt << 31) | (leftt >>> 1);
            work = ((leftt >>> 8) ^ right) & 0x00ff00ff;
            right ^= work;
            leftt ^= (work << 8);
            work = ((leftt >>> 2) ^ right) & 0x33333333;
            right ^= work;
            leftt ^= (work << 2);
            work = ((right >>> 16) ^ leftt) & 0x0000ffff;
            leftt ^= work;
            right ^= (work << 16);
            work = ((right >>> 4) ^ leftt) & 0x0f0f0f0f;
            leftt ^= work;
            right ^= (work << 4);
            outInts[0] = right;
            outInts[1] = leftt;
        }

        public void encrypt(byte[] clearText, byte[] cipherText) {
            encrypt(clearText, 0, cipherText, 0);
        }

        public void decrypt(byte[] cipherText, byte[] clearText) {
            decrypt(cipherText, 0, clearText, 0);
        }

        public byte[] encrypt(byte[] clearText) {
            int length = clearText.length;
            if (length % 8 != 0) {
                System.out.println("Array must be a multiple of 8");
                return null;
            }
            byte[] cipherText = new byte[length];
            int count = length / 8;
            for (int i = 0; i < count; i++)
                encrypt(clearText, i * 8, cipherText, i * 8);
            return cipherText;
        }

        public byte[] decrypt(byte[] cipherText) {
            int length = cipherText.length;
            if (length % 8 != 0) {
                System.out.println("Array must be a multiple of 8");
                return null;
            }
            byte[] clearText = new byte[length];
            int count = length / 8;
            for (int i = 0; i < count; i++)
                encrypt(cipherText, i * 8, clearText, i * 8);
            return clearText;
        }

        private static byte[] bytebit = {
                (byte)0x80,
                (byte)0x40,
                (byte)0x20,
                (byte)0x10,
                (byte)0x08,
                (byte)0x04,
                (byte)0x02,
                (byte)0x01};

        private static int[] bigbyte = {
                0x800000,
                0x400000,
                0x200000,
                0x100000,
                0x080000,
                0x040000,
                0x020000,
                0x010000,
                0x008000,
                0x004000,
                0x002000,
                0x001000,
                0x000800,
                0x000400,
                0x000200,
                0x000100,
                0x000080,
                0x000040,
                0x000020,
                0x000010,
                0x000008,
                0x000004,
                0x000002,
                0x000001};

        private static byte[] pc1 = {
                (byte)56,
                (byte)48,
                (byte)40,
                (byte)32,
                (byte)24,
                (byte)16,
                (byte)8,
                (byte)0,
                (byte)57,
                (byte)49,
                (byte)41,
                (byte)33,
                (byte)25,
                (byte)17,
                (byte)9,
                (byte)1,
                (byte)58,
                (byte)50,
                (byte)42,
                (byte)34,
                (byte)26,
                (byte)18,
                (byte)10,
                (byte)2,
                (byte)59,
                (byte)51,
                (byte)43,
                (byte)35,
                (byte)62,
                (byte)54,
                (byte)46,
                (byte)38,
                (byte)30,
                (byte)22,
                (byte)14,
                (byte)6,
                (byte)61,
                (byte)53,
                (byte)45,
                (byte)37,
                (byte)29,
                (byte)21,
                (byte)13,
                (byte)5,
                (byte)60,
                (byte)52,
                (byte)44,
                (byte)36,
                (byte)28,
                (byte)20,
                (byte)12,
                (byte)4,
                (byte)27,
                (byte)19,
                (byte)11,
                (byte)3};

        private static int[] totrot = {1, 2, 4, 6, 8, 10, 12, 14, 15, 17, 19, 21, 23, 25, 27, 28};

        private static byte[] pc2 = {
                (byte)13,
                (byte)16,
                (byte)10,
                (byte)23,
                (byte)0,
                (byte)4,
                (byte)2,
                (byte)27,
                (byte)14,
                (byte)5,
                (byte)20,
                (byte)9,
                (byte)22,
                (byte)18,
                (byte)11,
                (byte)3,
                (byte)25,
                (byte)7,
                (byte)15,
                (byte)6,
                (byte)26,
                (byte)19,
                (byte)12,
                (byte)1,
                (byte)40,
                (byte)51,
                (byte)30,
                (byte)36,
                (byte)46,
                (byte)54,
                (byte)29,
                (byte)39,
                (byte)50,
                (byte)44,
                (byte)32,
                (byte)47,
                (byte)43,
                (byte)48,
                (byte)38,
                (byte)55,
                (byte)33,
                (byte)52,
                (byte)45,
                (byte)41,
                (byte)49,
                (byte)35,
                (byte)28,
                (byte)31,};

        private static int[] SP1 = {
                0x01010400,
                0x00000000,
                0x00010000,
                0x01010404,
                0x01010004,
                0x00010404,
                0x00000004,
                0x00010000,
                0x00000400,
                0x01010400,
                0x01010404,
                0x00000400,
                0x01000404,
                0x01010004,
                0x01000000,
                0x00000004,
                0x00000404,
                0x01000400,
                0x01000400,
                0x00010400,
                0x00010400,
                0x01010000,
                0x01010000,
                0x01000404,
                0x00010004,
                0x01000004,
                0x01000004,
                0x00010004,
                0x00000000,
                0x00000404,
                0x00010404,
                0x01000000,
                0x00010000,
                0x01010404,
                0x00000004,
                0x01010000,
                0x01010400,
                0x01000000,
                0x01000000,
                0x00000400,
                0x01010004,
                0x00010000,
                0x00010400,
                0x01000004,
                0x00000400,
                0x00000004,
                0x01000404,
                0x00010404,
                0x01010404,
                0x00010004,
                0x01010000,
                0x01000404,
                0x01000004,
                0x00000404,
                0x00010404,
                0x01010400,
                0x00000404,
                0x01000400,
                0x01000400,
                0x00000000,
                0x00010004,
                0x00010400,
                0x00000000,
                0x01010004};

        private static int[] SP2 = {
                0x80108020,
                0x80008000,
                0x00008000,
                0x00108020,
                0x00100000,
                0x00000020,
                0x80100020,
                0x80008020,
                0x80000020,
                0x80108020,
                0x80108000,
                0x80000000,
                0x80008000,
                0x00100000,
                0x00000020,
                0x80100020,
                0x00108000,
                0x00100020,
                0x80008020,
                0x00000000,
                0x80000000,
                0x00008000,
                0x00108020,
                0x80100000,
                0x00100020,
                0x80000020,
                0x00000000,
                0x00108000,
                0x00008020,
                0x80108000,
                0x80100000,
                0x00008020,
                0x00000000,
                0x00108020,
                0x80100020,
                0x00100000,
                0x80008020,
                0x80100000,
                0x80108000,
                0x00008000,
                0x80100000,
                0x80008000,
                0x00000020,
                0x80108020,
                0x00108020,
                0x00000020,
                0x00008000,
                0x80000000,
                0x00008020,
                0x80108000,
                0x00100000,
                0x80000020,
                0x00100020,
                0x80008020,
                0x80000020,
                0x00100020,
                0x00108000,
                0x00000000,
                0x80008000,
                0x00008020,
                0x80000000,
                0x80100020,
                0x80108020,
                0x00108000};

        private static int[] SP3 = {
                0x00000208,
                0x08020200,
                0x00000000,
                0x08020008,
                0x08000200,
                0x00000000,
                0x00020208,
                0x08000200,
                0x00020008,
                0x08000008,
                0x08000008,
                0x00020000,
                0x08020208,
                0x00020008,
                0x08020000,
                0x00000208,
                0x08000000,
                0x00000008,
                0x08020200,
                0x00000200,
                0x00020200,
                0x08020000,
                0x08020008,
                0x00020208,
                0x08000208,
                0x00020200,
                0x00020000,
                0x08000208,
                0x00000008,
                0x08020208,
                0x00000200,
                0x08000000,
                0x08020200,
                0x08000000,
                0x00020008,
                0x00000208,
                0x00020000,
                0x08020200,
                0x08000200,
                0x00000000,
                0x00000200,
                0x00020008,
                0x08020208,
                0x08000200,
                0x08000008,
                0x00000200,
                0x00000000,
                0x08020008,
                0x08000208,
                0x00020000,
                0x08000000,
                0x08020208,
                0x00000008,
                0x00020208,
                0x00020200,
                0x08000008,
                0x08020000,
                0x08000208,
                0x00000208,
                0x08020000,
                0x00020208,
                0x00000008,
                0x08020008,
                0x00020200};

        private static int[] SP4 = {
                0x00802001,
                0x00002081,
                0x00002081,
                0x00000080,
                0x00802080,
                0x00800081,
                0x00800001,
                0x00002001,
                0x00000000,
                0x00802000,
                0x00802000,
                0x00802081,
                0x00000081,
                0x00000000,
                0x00800080,
                0x00800001,
                0x00000001,
                0x00002000,
                0x00800000,
                0x00802001,
                0x00000080,
                0x00800000,
                0x00002001,
                0x00002080,
                0x00800081,
                0x00000001,
                0x00002080,
                0x00800080,
                0x00002000,
                0x00802080,
                0x00802081,
                0x00000081,
                0x00800080,
                0x00800001,
                0x00802000,
                0x00802081,
                0x00000081,
                0x00000000,
                0x00000000,
                0x00802000,
                0x00002080,
                0x00800080,
                0x00800081,
                0x00000001,
                0x00802001,
                0x00002081,
                0x00002081,
                0x00000080,
                0x00802081,
                0x00000081,
                0x00000001,
                0x00002000,
                0x00800001,
                0x00002001,
                0x00802080,
                0x00800081,
                0x00002001,
                0x00002080,
                0x00800000,
                0x00802001,
                0x00000080,
                0x00800000,
                0x00002000,
                0x00802080};

        private static int[] SP5 = {
                0x00000100,
                0x02080100,
                0x02080000,
                0x42000100,
                0x00080000,
                0x00000100,
                0x40000000,
                0x02080000,
                0x40080100,
                0x00080000,
                0x02000100,
                0x40080100,
                0x42000100,
                0x42080000,
                0x00080100,
                0x40000000,
                0x02000000,
                0x40080000,
                0x40080000,
                0x00000000,
                0x40000100,
                0x42080100,
                0x42080100,
                0x02000100,
                0x42080000,
                0x40000100,
                0x00000000,
                0x42000000,
                0x02080100,
                0x02000000,
                0x42000000,
                0x00080100,
                0x00080000,
                0x42000100,
                0x00000100,
                0x02000000,
                0x40000000,
                0x02080000,
                0x42000100,
                0x40080100,
                0x02000100,
                0x40000000,
                0x42080000,
                0x02080100,
                0x40080100,
                0x00000100,
                0x02000000,
                0x42080000,
                0x42080100,
                0x00080100,
                0x42000000,
                0x42080100,
                0x02080000,
                0x00000000,
                0x40080000,
                0x42000000,
                0x00080100,
                0x02000100,
                0x40000100,
                0x00080000,
                0x00000000,
                0x40080000,
                0x02080100,
                0x40000100};

        private static int[] SP6 = {
                0x20000010,
                0x20400000,
                0x00004000,
                0x20404010,
                0x20400000,
                0x00000010,
                0x20404010,
                0x00400000,
                0x20004000,
                0x00404010,
                0x00400000,
                0x20000010,
                0x00400010,
                0x20004000,
                0x20000000,
                0x00004010,
                0x00000000,
                0x00400010,
                0x20004010,
                0x00004000,
                0x00404000,
                0x20004010,
                0x00000010,
                0x20400010,
                0x20400010,
                0x00000000,
                0x00404010,
                0x20404000,
                0x00004010,
                0x00404000,
                0x20404000,
                0x20000000,
                0x20004000,
                0x00000010,
                0x20400010,
                0x00404000,
                0x20404010,
                0x00400000,
                0x00004010,
                0x20000010,
                0x00400000,
                0x20004000,
                0x20000000,
                0x00004010,
                0x20000010,
                0x20404010,
                0x00404000,
                0x20400000,
                0x00404010,
                0x20404000,
                0x00000000,
                0x20400010,
                0x00000010,
                0x00004000,
                0x20400000,
                0x00404010,
                0x00004000,
                0x00400010,
                0x20004010,
                0x00000000,
                0x20404000,
                0x20000000,
                0x00400010,
                0x20004010};

        private static int[] SP7 = {
                0x00200000,
                0x04200002,
                0x04000802,
                0x00000000,
                0x00000800,
                0x04000802,
                0x00200802,
                0x04200800,
                0x04200802,
                0x00200000,
                0x00000000,
                0x04000002,
                0x00000002,
                0x04000000,
                0x04200002,
                0x00000802,
                0x04000800,
                0x00200802,
                0x00200002,
                0x04000800,
                0x04000002,
                0x04200000,
                0x04200800,
                0x00200002,
                0x04200000,
                0x00000800,
                0x00000802,
                0x04200802,
                0x00200800,
                0x00000002,
                0x04000000,
                0x00200800,
                0x04000000,
                0x00200800,
                0x00200000,
                0x04000802,
                0x04000802,
                0x04200002,
                0x04200002,
                0x00000002,
                0x00200002,
                0x04000000,
                0x04000800,
                0x00200000,
                0x04200800,
                0x00000802,
                0x00200802,
                0x04200800,
                0x00000802,
                0x04000002,
                0x04200802,
                0x04200000,
                0x00200800,
                0x00000000,
                0x00000002,
                0x04200802,
                0x00000000,
                0x00200802,
                0x04200000,
                0x00000800,
                0x04000002,
                0x04000800,
                0x00000800,
                0x00200002};

        private static int[] SP8 = {
                0x10001040,
                0x00001000,
                0x00040000,
                0x10041040,
                0x10000000,
                0x10001040,
                0x00000040,
                0x10000000,
                0x00040040,
                0x10040000,
                0x10041040,
                0x00041000,
                0x10041000,
                0x00041040,
                0x00001000,
                0x00000040,
                0x10040000,
                0x10000040,
                0x10001000,
                0x00001040,
                0x00041000,
                0x00040040,
                0x10040040,
                0x10041000,
                0x00001040,
                0x00000000,
                0x00000000,
                0x10040040,
                0x10000040,
                0x10001000,
                0x00041040,
                0x00040000,
                0x00041040,
                0x00040000,
                0x10041000,
                0x00001000,
                0x00000040,
                0x10040040,
                0x00001000,
                0x00041040,
                0x10001000,
                0x00000040,
                0x10000040,
                0x10040000,
                0x10040040,
                0x10000000,
                0x00040000,
                0x10001040,
                0x00000000,
                0x10041040,
                0x00040040,
                0x10000040,
                0x10040000,
                0x10001000,
                0x10001040,
                0x00000000,
                0x10041040,
                0x00041000,
                0x00041000,
                0x00001040,
                0x00001040,
                0x00040040,
                0x10000000,
                0x10041000};

        public static void squashBytesToInts(byte[] inBytes, int inOff, int[] outInts, int outOff,
                int intLen) {
            for (int i = 0; i < intLen; ++i)
                outInts[outOff + i] = ((inBytes[inOff + i * 4] & 0xff) << 24)
                        | ((inBytes[inOff + i * 4 + 1] & 0xff) << 16)
                        | ((inBytes[inOff + i * 4 + 2] & 0xff) << 8)
                        | (inBytes[inOff + i * 4 + 3] & 0xff);
        }

        public static void spreadIntsToBytes(int[] inInts, int inOff, byte[] outBytes, int outOff,
                int intLen) {
            for (int i = 0; i < intLen; ++i) {
                outBytes[outOff + i * 4] = (byte)(inInts[inOff + i] >>> 24);
                outBytes[outOff + i * 4 + 1] = (byte)(inInts[inOff + i] >>> 16);
                outBytes[outOff + i * 4 + 2] = (byte)(inInts[inOff + i] >>> 8);
                outBytes[outOff + i * 4 + 3] = (byte)inInts[inOff + i];
            }
        }
    }

    // ------------------------------------------------------------------------
    public static class MD4 extends MessageDigest implements Cloneable {
        private static final int BLOCK_LENGTH = 64;

        private int[] context = new int[4];

        private long count;

        private byte[] buffer = new byte[BLOCK_LENGTH];

        private int[] X = new int[16];

        public MD4() {
            super("MD4");
            engineReset();
        }

        private MD4(MD4 md) {
            this();
            context = (int[])md.context.clone();
            buffer = (byte[])md.buffer.clone();
            count = md.count;
        }

        public Object clone() {
            return new MD4(this);
        }

        public void engineReset() {
            context[0] = 0x67452301;
            context[1] = 0xEFCDAB89;
            context[2] = 0x98BADCFE;
            context[3] = 0x10325476;
            count = 0L;
            for (int i = 0; i < BLOCK_LENGTH; i++)
                buffer[i] = 0;
        }

        public void engineUpdate(byte b) {
            int i = (int)(count % BLOCK_LENGTH);
            count++;
            buffer[i] = b;
            if (i == BLOCK_LENGTH - 1) transform(buffer, 0);
        }

        public void engineUpdate(byte[] input, int offset, int len) {
            if (offset < 0 || len < 0 || (long)offset + len > input.length)
                throw new ArrayIndexOutOfBoundsException();
            int bufferNdx = (int)(count % BLOCK_LENGTH);
            count += len;
            int partLen = BLOCK_LENGTH - bufferNdx;
            int i = 0;
            if (len >= partLen) {
                System.arraycopy(input, offset, buffer, bufferNdx, partLen);
                transform(buffer, 0);
                for (i = partLen; i + BLOCK_LENGTH - 1 < len; i += BLOCK_LENGTH)
                    transform(input, offset + i);
                bufferNdx = 0;
            }
            if (i < len) System.arraycopy(input, offset + i, buffer, bufferNdx, len - i);
        }

        public byte[] engineDigest() {
            int bufferNdx = (int)(count % BLOCK_LENGTH);
            int padLen = (bufferNdx < 56)? (56 - bufferNdx): (120 - bufferNdx);
            byte[] tail = new byte[padLen + 8];
            tail[0] = (byte)0x80;
            for (int i = 0; i < 8; i++)
                tail[padLen + i] = (byte)((count * 8) >>> (8 * i));
            engineUpdate(tail, 0, tail.length);
            byte[] result = new byte[16];
            for (int i = 0; i < 4; i++)
                for (int j = 0; j < 4; j++)
                    result[i * 4 + j] = (byte)(context[i] >>> (8 * j));
            engineReset();
            return result;
        }

        private void transform(byte[] block, int offset) {
            for (int i = 0; i < 16; i++)
                X[i] = (block[offset++] & 0xFF) | (block[offset++] & 0xFF) << 8
                        | (block[offset++] & 0xFF) << 16 | (block[offset++] & 0xFF) << 24;
            int A = context[0];
            int B = context[1];
            int C = context[2];
            int D = context[3];
            A = FF(A, B, C, D, X[0], 3);
            D = FF(D, A, B, C, X[1], 7);
            C = FF(C, D, A, B, X[2], 11);
            B = FF(B, C, D, A, X[3], 19);
            A = FF(A, B, C, D, X[4], 3);
            D = FF(D, A, B, C, X[5], 7);
            C = FF(C, D, A, B, X[6], 11);
            B = FF(B, C, D, A, X[7], 19);
            A = FF(A, B, C, D, X[8], 3);
            D = FF(D, A, B, C, X[9], 7);
            C = FF(C, D, A, B, X[10], 11);
            B = FF(B, C, D, A, X[11], 19);
            A = FF(A, B, C, D, X[12], 3);
            D = FF(D, A, B, C, X[13], 7);
            C = FF(C, D, A, B, X[14], 11);
            B = FF(B, C, D, A, X[15], 19);
            A = GG(A, B, C, D, X[0], 3);
            D = GG(D, A, B, C, X[4], 5);
            C = GG(C, D, A, B, X[8], 9);
            B = GG(B, C, D, A, X[12], 13);
            A = GG(A, B, C, D, X[1], 3);
            D = GG(D, A, B, C, X[5], 5);
            C = GG(C, D, A, B, X[9], 9);
            B = GG(B, C, D, A, X[13], 13);
            A = GG(A, B, C, D, X[2], 3);
            D = GG(D, A, B, C, X[6], 5);
            C = GG(C, D, A, B, X[10], 9);
            B = GG(B, C, D, A, X[14], 13);
            A = GG(A, B, C, D, X[3], 3);
            D = GG(D, A, B, C, X[7], 5);
            C = GG(C, D, A, B, X[11], 9);
            B = GG(B, C, D, A, X[15], 13);
            A = HH(A, B, C, D, X[0], 3);
            D = HH(D, A, B, C, X[8], 9);
            C = HH(C, D, A, B, X[4], 11);
            B = HH(B, C, D, A, X[12], 15);
            A = HH(A, B, C, D, X[2], 3);
            D = HH(D, A, B, C, X[10], 9);
            C = HH(C, D, A, B, X[6], 11);
            B = HH(B, C, D, A, X[14], 15);
            A = HH(A, B, C, D, X[1], 3);
            D = HH(D, A, B, C, X[9], 9);
            C = HH(C, D, A, B, X[5], 11);
            B = HH(B, C, D, A, X[13], 15);
            A = HH(A, B, C, D, X[3], 3);
            D = HH(D, A, B, C, X[11], 9);
            C = HH(C, D, A, B, X[7], 11);
            B = HH(B, C, D, A, X[15], 15);
            context[0] += A;
            context[1] += B;
            context[2] += C;
            context[3] += D;
        }

        private int FF(int a, int b, int c, int d, int x, int s) {
            int t = a + ((b & c) | (~b & d)) + x;
            return t << s | t >>> (32 - s);
        }

        private int GG(int a, int b, int c, int d, int x, int s) {
            int t = a + ((b & (c | d)) | (c & d)) + x + 0x5A827999;
            return t << s | t >>> (32 - s);
        }

        private int HH(int a, int b, int c, int d, int x, int s) {
            int t = a + (b ^ c ^ d) + x + 0x6ED9EBA1;
            return t << s | t >>> (32 - s);
        }
    }

    // ------------------------------------------------------------------------
    public static class jcrypt {
        private jcrypt() {
        }

        private static final int ITERATIONS = 16;

        private static final int con_salt[] = {
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,
                0x01,
                0x02,
                0x03,
                0x04,
                0x05,
                0x06,
                0x07,
                0x08,
                0x09,
                0x0A,
                0x0B,
                0x05,
                0x06,
                0x07,
                0x08,
                0x09,
                0x0A,
                0x0B,
                0x0C,
                0x0D,
                0x0E,
                0x0F,
                0x10,
                0x11,
                0x12,
                0x13,
                0x14,
                0x15,
                0x16,
                0x17,
                0x18,
                0x19,
                0x1A,
                0x1B,
                0x1C,
                0x1D,
                0x1E,
                0x1F,
                0x20,
                0x21,
                0x22,
                0x23,
                0x24,
                0x25,
                0x20,
                0x21,
                0x22,
                0x23,
                0x24,
                0x25,
                0x26,
                0x27,
                0x28,
                0x29,
                0x2A,
                0x2B,
                0x2C,
                0x2D,
                0x2E,
                0x2F,
                0x30,
                0x31,
                0x32,
                0x33,
                0x34,
                0x35,
                0x36,
                0x37,
                0x38,
                0x39,
                0x3A,
                0x3B,
                0x3C,
                0x3D,
                0x3E,
                0x3F,
                0x00,
                0x00,
                0x00,
                0x00,
                0x00,};

        private static final boolean shifts2[] = {
                false,
                false,
                true,
                true,
                true,
                true,
                true,
                true,
                false,
                true,
                true,
                true,
                true,
                true,
                true,
                false};

        private static final int skb[][] = {
                {
                        0x00000000,
                        0x00000010,
                        0x20000000,
                        0x20000010,
                        0x00010000,
                        0x00010010,
                        0x20010000,
                        0x20010010,
                        0x00000800,
                        0x00000810,
                        0x20000800,
                        0x20000810,
                        0x00010800,
                        0x00010810,
                        0x20010800,
                        0x20010810,
                        0x00000020,
                        0x00000030,
                        0x20000020,
                        0x20000030,
                        0x00010020,
                        0x00010030,
                        0x20010020,
                        0x20010030,
                        0x00000820,
                        0x00000830,
                        0x20000820,
                        0x20000830,
                        0x00010820,
                        0x00010830,
                        0x20010820,
                        0x20010830,
                        0x00080000,
                        0x00080010,
                        0x20080000,
                        0x20080010,
                        0x00090000,
                        0x00090010,
                        0x20090000,
                        0x20090010,
                        0x00080800,
                        0x00080810,
                        0x20080800,
                        0x20080810,
                        0x00090800,
                        0x00090810,
                        0x20090800,
                        0x20090810,
                        0x00080020,
                        0x00080030,
                        0x20080020,
                        0x20080030,
                        0x00090020,
                        0x00090030,
                        0x20090020,
                        0x20090030,
                        0x00080820,
                        0x00080830,
                        0x20080820,
                        0x20080830,
                        0x00090820,
                        0x00090830,
                        0x20090820,
                        0x20090830,},
                {
                        0x00000000,
                        0x02000000,
                        0x00002000,
                        0x02002000,
                        0x00200000,
                        0x02200000,
                        0x00202000,
                        0x02202000,
                        0x00000004,
                        0x02000004,
                        0x00002004,
                        0x02002004,
                        0x00200004,
                        0x02200004,
                        0x00202004,
                        0x02202004,
                        0x00000400,
                        0x02000400,
                        0x00002400,
                        0x02002400,
                        0x00200400,
                        0x02200400,
                        0x00202400,
                        0x02202400,
                        0x00000404,
                        0x02000404,
                        0x00002404,
                        0x02002404,
                        0x00200404,
                        0x02200404,
                        0x00202404,
                        0x02202404,
                        0x10000000,
                        0x12000000,
                        0x10002000,
                        0x12002000,
                        0x10200000,
                        0x12200000,
                        0x10202000,
                        0x12202000,
                        0x10000004,
                        0x12000004,
                        0x10002004,
                        0x12002004,
                        0x10200004,
                        0x12200004,
                        0x10202004,
                        0x12202004,
                        0x10000400,
                        0x12000400,
                        0x10002400,
                        0x12002400,
                        0x10200400,
                        0x12200400,
                        0x10202400,
                        0x12202400,
                        0x10000404,
                        0x12000404,
                        0x10002404,
                        0x12002404,
                        0x10200404,
                        0x12200404,
                        0x10202404,
                        0x12202404,},
                {
                        0x00000000,
                        0x00000001,
                        0x00040000,
                        0x00040001,
                        0x01000000,
                        0x01000001,
                        0x01040000,
                        0x01040001,
                        0x00000002,
                        0x00000003,
                        0x00040002,
                        0x00040003,
                        0x01000002,
                        0x01000003,
                        0x01040002,
                        0x01040003,
                        0x00000200,
                        0x00000201,
                        0x00040200,
                        0x00040201,
                        0x01000200,
                        0x01000201,
                        0x01040200,
                        0x01040201,
                        0x00000202,
                        0x00000203,
                        0x00040202,
                        0x00040203,
                        0x01000202,
                        0x01000203,
                        0x01040202,
                        0x01040203,
                        0x08000000,
                        0x08000001,
                        0x08040000,
                        0x08040001,
                        0x09000000,
                        0x09000001,
                        0x09040000,
                        0x09040001,
                        0x08000002,
                        0x08000003,
                        0x08040002,
                        0x08040003,
                        0x09000002,
                        0x09000003,
                        0x09040002,
                        0x09040003,
                        0x08000200,
                        0x08000201,
                        0x08040200,
                        0x08040201,
                        0x09000200,
                        0x09000201,
                        0x09040200,
                        0x09040201,
                        0x08000202,
                        0x08000203,
                        0x08040202,
                        0x08040203,
                        0x09000202,
                        0x09000203,
                        0x09040202,
                        0x09040203,},
                {
                        0x00000000,
                        0x00100000,
                        0x00000100,
                        0x00100100,
                        0x00000008,
                        0x00100008,
                        0x00000108,
                        0x00100108,
                        0x00001000,
                        0x00101000,
                        0x00001100,
                        0x00101100,
                        0x00001008,
                        0x00101008,
                        0x00001108,
                        0x00101108,
                        0x04000000,
                        0x04100000,
                        0x04000100,
                        0x04100100,
                        0x04000008,
                        0x04100008,
                        0x04000108,
                        0x04100108,
                        0x04001000,
                        0x04101000,
                        0x04001100,
                        0x04101100,
                        0x04001008,
                        0x04101008,
                        0x04001108,
                        0x04101108,
                        0x00020000,
                        0x00120000,
                        0x00020100,
                        0x00120100,
                        0x00020008,
                        0x00120008,
                        0x00020108,
                        0x00120108,
                        0x00021000,
                        0x00121000,
                        0x00021100,
                        0x00121100,
                        0x00021008,
                        0x00121008,
                        0x00021108,
                        0x00121108,
                        0x04020000,
                        0x04120000,
                        0x04020100,
                        0x04120100,
                        0x04020008,
                        0x04120008,
                        0x04020108,
                        0x04120108,
                        0x04021000,
                        0x04121000,
                        0x04021100,
                        0x04121100,
                        0x04021008,
                        0x04121008,
                        0x04021108,
                        0x04121108,},
                {
                        0x00000000,
                        0x10000000,
                        0x00010000,
                        0x10010000,
                        0x00000004,
                        0x10000004,
                        0x00010004,
                        0x10010004,
                        0x20000000,
                        0x30000000,
                        0x20010000,
                        0x30010000,
                        0x20000004,
                        0x30000004,
                        0x20010004,
                        0x30010004,
                        0x00100000,
                        0x10100000,
                        0x00110000,
                        0x10110000,
                        0x00100004,
                        0x10100004,
                        0x00110004,
                        0x10110004,
                        0x20100000,
                        0x30100000,
                        0x20110000,
                        0x30110000,
                        0x20100004,
                        0x30100004,
                        0x20110004,
                        0x30110004,
                        0x00001000,
                        0x10001000,
                        0x00011000,
                        0x10011000,
                        0x00001004,
                        0x10001004,
                        0x00011004,
                        0x10011004,
                        0x20001000,
                        0x30001000,
                        0x20011000,
                        0x30011000,
                        0x20001004,
                        0x30001004,
                        0x20011004,
                        0x30011004,
                        0x00101000,
                        0x10101000,
                        0x00111000,
                        0x10111000,
                        0x00101004,
                        0x10101004,
                        0x00111004,
                        0x10111004,
                        0x20101000,
                        0x30101000,
                        0x20111000,
                        0x30111000,
                        0x20101004,
                        0x30101004,
                        0x20111004,
                        0x30111004,},
                {
                        0x00000000,
                        0x08000000,
                        0x00000008,
                        0x08000008,
                        0x00000400,
                        0x08000400,
                        0x00000408,
                        0x08000408,
                        0x00020000,
                        0x08020000,
                        0x00020008,
                        0x08020008,
                        0x00020400,
                        0x08020400,
                        0x00020408,
                        0x08020408,
                        0x00000001,
                        0x08000001,
                        0x00000009,
                        0x08000009,
                        0x00000401,
                        0x08000401,
                        0x00000409,
                        0x08000409,
                        0x00020001,
                        0x08020001,
                        0x00020009,
                        0x08020009,
                        0x00020401,
                        0x08020401,
                        0x00020409,
                        0x08020409,
                        0x02000000,
                        0x0A000000,
                        0x02000008,
                        0x0A000008,
                        0x02000400,
                        0x0A000400,
                        0x02000408,
                        0x0A000408,
                        0x02020000,
                        0x0A020000,
                        0x02020008,
                        0x0A020008,
                        0x02020400,
                        0x0A020400,
                        0x02020408,
                        0x0A020408,
                        0x02000001,
                        0x0A000001,
                        0x02000009,
                        0x0A000009,
                        0x02000401,
                        0x0A000401,
                        0x02000409,
                        0x0A000409,
                        0x02020001,
                        0x0A020001,
                        0x02020009,
                        0x0A020009,
                        0x02020401,
                        0x0A020401,
                        0x02020409,
                        0x0A020409,},
                {
                        0x00000000,
                        0x00000100,
                        0x00080000,
                        0x00080100,
                        0x01000000,
                        0x01000100,
                        0x01080000,
                        0x01080100,
                        0x00000010,
                        0x00000110,
                        0x00080010,
                        0x00080110,
                        0x01000010,
                        0x01000110,
                        0x01080010,
                        0x01080110,
                        0x00200000,
                        0x00200100,
                        0x00280000,
                        0x00280100,
                        0x01200000,
                        0x01200100,
                        0x01280000,
                        0x01280100,
                        0x00200010,
                        0x00200110,
                        0x00280010,
                        0x00280110,
                        0x01200010,
                        0x01200110,
                        0x01280010,
                        0x01280110,
                        0x00000200,
                        0x00000300,
                        0x00080200,
                        0x00080300,
                        0x01000200,
                        0x01000300,
                        0x01080200,
                        0x01080300,
                        0x00000210,
                        0x00000310,
                        0x00080210,
                        0x00080310,
                        0x01000210,
                        0x01000310,
                        0x01080210,
                        0x01080310,
                        0x00200200,
                        0x00200300,
                        0x00280200,
                        0x00280300,
                        0x01200200,
                        0x01200300,
                        0x01280200,
                        0x01280300,
                        0x00200210,
                        0x00200310,
                        0x00280210,
                        0x00280310,
                        0x01200210,
                        0x01200310,
                        0x01280210,
                        0x01280310,},
                {
                        0x00000000,
                        0x04000000,
                        0x00040000,
                        0x04040000,
                        0x00000002,
                        0x04000002,
                        0x00040002,
                        0x04040002,
                        0x00002000,
                        0x04002000,
                        0x00042000,
                        0x04042000,
                        0x00002002,
                        0x04002002,
                        0x00042002,
                        0x04042002,
                        0x00000020,
                        0x04000020,
                        0x00040020,
                        0x04040020,
                        0x00000022,
                        0x04000022,
                        0x00040022,
                        0x04040022,
                        0x00002020,
                        0x04002020,
                        0x00042020,
                        0x04042020,
                        0x00002022,
                        0x04002022,
                        0x00042022,
                        0x04042022,
                        0x00000800,
                        0x04000800,
                        0x00040800,
                        0x04040800,
                        0x00000802,
                        0x04000802,
                        0x00040802,
                        0x04040802,
                        0x00002800,
                        0x04002800,
                        0x00042800,
                        0x04042800,
                        0x00002802,
                        0x04002802,
                        0x00042802,
                        0x04042802,
                        0x00000820,
                        0x04000820,
                        0x00040820,
                        0x04040820,
                        0x00000822,
                        0x04000822,
                        0x00040822,
                        0x04040822,
                        0x00002820,
                        0x04002820,
                        0x00042820,
                        0x04042820,
                        0x00002822,
                        0x04002822,
                        0x00042822,
                        0x04042822,},};

        private static final int SPtrans[][] = {
                {
                        0x00820200,
                        0x00020000,
                        0x80800000,
                        0x80820200,
                        0x00800000,
                        0x80020200,
                        0x80020000,
                        0x80800000,
                        0x80020200,
                        0x00820200,
                        0x00820000,
                        0x80000200,
                        0x80800200,
                        0x00800000,
                        0x00000000,
                        0x80020000,
                        0x00020000,
                        0x80000000,
                        0x00800200,
                        0x00020200,
                        0x80820200,
                        0x00820000,
                        0x80000200,
                        0x00800200,
                        0x80000000,
                        0x00000200,
                        0x00020200,
                        0x80820000,
                        0x00000200,
                        0x80800200,
                        0x80820000,
                        0x00000000,
                        0x00000000,
                        0x80820200,
                        0x00800200,
                        0x80020000,
                        0x00820200,
                        0x00020000,
                        0x80000200,
                        0x00800200,
                        0x80820000,
                        0x00000200,
                        0x00020200,
                        0x80800000,
                        0x80020200,
                        0x80000000,
                        0x80800000,
                        0x00820000,
                        0x80820200,
                        0x00020200,
                        0x00820000,
                        0x80800200,
                        0x00800000,
                        0x80000200,
                        0x80020000,
                        0x00000000,
                        0x00020000,
                        0x00800000,
                        0x80800200,
                        0x00820200,
                        0x80000000,
                        0x80820000,
                        0x00000200,
                        0x80020200,},
                {
                        0x10042004,
                        0x00000000,
                        0x00042000,
                        0x10040000,
                        0x10000004,
                        0x00002004,
                        0x10002000,
                        0x00042000,
                        0x00002000,
                        0x10040004,
                        0x00000004,
                        0x10002000,
                        0x00040004,
                        0x10042000,
                        0x10040000,
                        0x00000004,
                        0x00040000,
                        0x10002004,
                        0x10040004,
                        0x00002000,
                        0x00042004,
                        0x10000000,
                        0x00000000,
                        0x00040004,
                        0x10002004,
                        0x00042004,
                        0x10042000,
                        0x10000004,
                        0x10000000,
                        0x00040000,
                        0x00002004,
                        0x10042004,
                        0x00040004,
                        0x10042000,
                        0x10002000,
                        0x00042004,
                        0x10042004,
                        0x00040004,
                        0x10000004,
                        0x00000000,
                        0x10000000,
                        0x00002004,
                        0x00040000,
                        0x10040004,
                        0x00002000,
                        0x10000000,
                        0x00042004,
                        0x10002004,
                        0x10042000,
                        0x00002000,
                        0x00000000,
                        0x10000004,
                        0x00000004,
                        0x10042004,
                        0x00042000,
                        0x10040000,
                        0x10040004,
                        0x00040000,
                        0x00002004,
                        0x10002000,
                        0x10002004,
                        0x00000004,
                        0x10040000,
                        0x00042000,},
                {
                        0x41000000,
                        0x01010040,
                        0x00000040,
                        0x41000040,
                        0x40010000,
                        0x01000000,
                        0x41000040,
                        0x00010040,
                        0x01000040,
                        0x00010000,
                        0x01010000,
                        0x40000000,
                        0x41010040,
                        0x40000040,
                        0x40000000,
                        0x41010000,
                        0x00000000,
                        0x40010000,
                        0x01010040,
                        0x00000040,
                        0x40000040,
                        0x41010040,
                        0x00010000,
                        0x41000000,
                        0x41010000,
                        0x01000040,
                        0x40010040,
                        0x01010000,
                        0x00010040,
                        0x00000000,
                        0x01000000,
                        0x40010040,
                        0x01010040,
                        0x00000040,
                        0x40000000,
                        0x00010000,
                        0x40000040,
                        0x40010000,
                        0x01010000,
                        0x41000040,
                        0x00000000,
                        0x01010040,
                        0x00010040,
                        0x41010000,
                        0x40010000,
                        0x01000000,
                        0x41010040,
                        0x40000000,
                        0x40010040,
                        0x41000000,
                        0x01000000,
                        0x41010040,
                        0x00010000,
                        0x01000040,
                        0x41000040,
                        0x00010040,
                        0x01000040,
                        0x00000000,
                        0x41010000,
                        0x40000040,
                        0x41000000,
                        0x40010040,
                        0x00000040,
                        0x01010000,},
                {
                        0x00100402,
                        0x04000400,
                        0x00000002,
                        0x04100402,
                        0x00000000,
                        0x04100000,
                        0x04000402,
                        0x00100002,
                        0x04100400,
                        0x04000002,
                        0x04000000,
                        0x00000402,
                        0x04000002,
                        0x00100402,
                        0x00100000,
                        0x04000000,
                        0x04100002,
                        0x00100400,
                        0x00000400,
                        0x00000002,
                        0x00100400,
                        0x04000402,
                        0x04100000,
                        0x00000400,
                        0x00000402,
                        0x00000000,
                        0x00100002,
                        0x04100400,
                        0x04000400,
                        0x04100002,
                        0x04100402,
                        0x00100000,
                        0x04100002,
                        0x00000402,
                        0x00100000,
                        0x04000002,
                        0x00100400,
                        0x04000400,
                        0x00000002,
                        0x04100000,
                        0x04000402,
                        0x00000000,
                        0x00000400,
                        0x00100002,
                        0x00000000,
                        0x04100002,
                        0x04100400,
                        0x00000400,
                        0x04000000,
                        0x04100402,
                        0x00100402,
                        0x00100000,
                        0x04100402,
                        0x00000002,
                        0x04000400,
                        0x00100402,
                        0x00100002,
                        0x00100400,
                        0x04100000,
                        0x04000402,
                        0x00000402,
                        0x04000000,
                        0x04000002,
                        0x04100400,},
                {
                        0x02000000,
                        0x00004000,
                        0x00000100,
                        0x02004108,
                        0x02004008,
                        0x02000100,
                        0x00004108,
                        0x02004000,
                        0x00004000,
                        0x00000008,
                        0x02000008,
                        0x00004100,
                        0x02000108,
                        0x02004008,
                        0x02004100,
                        0x00000000,
                        0x00004100,
                        0x02000000,
                        0x00004008,
                        0x00000108,
                        0x02000100,
                        0x00004108,
                        0x00000000,
                        0x02000008,
                        0x00000008,
                        0x02000108,
                        0x02004108,
                        0x00004008,
                        0x02004000,
                        0x00000100,
                        0x00000108,
                        0x02004100,
                        0x02004100,
                        0x02000108,
                        0x00004008,
                        0x02004000,
                        0x00004000,
                        0x00000008,
                        0x02000008,
                        0x02000100,
                        0x02000000,
                        0x00004100,
                        0x02004108,
                        0x00000000,
                        0x00004108,
                        0x02000000,
                        0x00000100,
                        0x00004008,
                        0x02000108,
                        0x00000100,
                        0x00000000,
                        0x02004108,
                        0x02004008,
                        0x02004100,
                        0x00000108,
                        0x00004000,
                        0x00004100,
                        0x02004008,
                        0x02000100,
                        0x00000108,
                        0x00000008,
                        0x00004108,
                        0x02004000,
                        0x02000008,},
                {
                        0x20000010,
                        0x00080010,
                        0x00000000,
                        0x20080800,
                        0x00080010,
                        0x00000800,
                        0x20000810,
                        0x00080000,
                        0x00000810,
                        0x20080810,
                        0x00080800,
                        0x20000000,
                        0x20000800,
                        0x20000010,
                        0x20080000,
                        0x00080810,
                        0x00080000,
                        0x20000810,
                        0x20080010,
                        0x00000000,
                        0x00000800,
                        0x00000010,
                        0x20080800,
                        0x20080010,
                        0x20080810,
                        0x20080000,
                        0x20000000,
                        0x00000810,
                        0x00000010,
                        0x00080800,
                        0x00080810,
                        0x20000800,
                        0x00000810,
                        0x20000000,
                        0x20000800,
                        0x00080810,
                        0x20080800,
                        0x00080010,
                        0x00000000,
                        0x20000800,
                        0x20000000,
                        0x00000800,
                        0x20080010,
                        0x00080000,
                        0x00080010,
                        0x20080810,
                        0x00080800,
                        0x00000010,
                        0x20080810,
                        0x00080800,
                        0x00080000,
                        0x20000810,
                        0x20000010,
                        0x20080000,
                        0x00080810,
                        0x00000000,
                        0x00000800,
                        0x20000010,
                        0x20000810,
                        0x20080800,
                        0x20080000,
                        0x00000810,
                        0x00000010,
                        0x20080010,},
                {
                        0x00001000,
                        0x00000080,
                        0x00400080,
                        0x00400001,
                        0x00401081,
                        0x00001001,
                        0x00001080,
                        0x00000000,
                        0x00400000,
                        0x00400081,
                        0x00000081,
                        0x00401000,
                        0x00000001,
                        0x00401080,
                        0x00401000,
                        0x00000081,
                        0x00400081,
                        0x00001000,
                        0x00001001,
                        0x00401081,
                        0x00000000,
                        0x00400080,
                        0x00400001,
                        0x00001080,
                        0x00401001,
                        0x00001081,
                        0x00401080,
                        0x00000001,
                        0x00001081,
                        0x00401001,
                        0x00000080,
                        0x00400000,
                        0x00001081,
                        0x00401000,
                        0x00401001,
                        0x00000081,
                        0x00001000,
                        0x00000080,
                        0x00400000,
                        0x00401001,
                        0x00400081,
                        0x00001081,
                        0x00001080,
                        0x00000000,
                        0x00000080,
                        0x00400001,
                        0x00000001,
                        0x00400080,
                        0x00000000,
                        0x00400081,
                        0x00400080,
                        0x00001080,
                        0x00000081,
                        0x00001000,
                        0x00401081,
                        0x00400000,
                        0x00401080,
                        0x00000001,
                        0x00001001,
                        0x00401081,
                        0x00400001,
                        0x00401080,
                        0x00401000,
                        0x00001001,},
                {
                        0x08200020,
                        0x08208000,
                        0x00008020,
                        0x00000000,
                        0x08008000,
                        0x00200020,
                        0x08200000,
                        0x08208020,
                        0x00000020,
                        0x08000000,
                        0x00208000,
                        0x00008020,
                        0x00208020,
                        0x08008020,
                        0x08000020,
                        0x08200000,
                        0x00008000,
                        0x00208020,
                        0x00200020,
                        0x08008000,
                        0x08208020,
                        0x08000020,
                        0x00000000,
                        0x00208000,
                        0x08000000,
                        0x00200000,
                        0x08008020,
                        0x08200020,
                        0x00200000,
                        0x00008000,
                        0x08208000,
                        0x00000020,
                        0x00200000,
                        0x00008000,
                        0x08000020,
                        0x08208020,
                        0x00008020,
                        0x08000000,
                        0x00000000,
                        0x00208000,
                        0x08200020,
                        0x08008020,
                        0x08008000,
                        0x00200020,
                        0x08208000,
                        0x00000020,
                        0x00200020,
                        0x08008000,
                        0x08208020,
                        0x00200000,
                        0x08200000,
                        0x08000020,
                        0x00208000,
                        0x00008020,
                        0x08008020,
                        0x08200000,
                        0x00000020,
                        0x08208000,
                        0x00208020,
                        0x00000000,
                        0x08000000,
                        0x08200020,
                        0x00008000,
                        0x00208020}};

        private static final int cov_2char[] = {
                0x2E,
                0x2F,
                0x30,
                0x31,
                0x32,
                0x33,
                0x34,
                0x35,
                0x36,
                0x37,
                0x38,
                0x39,
                0x41,
                0x42,
                0x43,
                0x44,
                0x45,
                0x46,
                0x47,
                0x48,
                0x49,
                0x4A,
                0x4B,
                0x4C,
                0x4D,
                0x4E,
                0x4F,
                0x50,
                0x51,
                0x52,
                0x53,
                0x54,
                0x55,
                0x56,
                0x57,
                0x58,
                0x59,
                0x5A,
                0x61,
                0x62,
                0x63,
                0x64,
                0x65,
                0x66,
                0x67,
                0x68,
                0x69,
                0x6A,
                0x6B,
                0x6C,
                0x6D,
                0x6E,
                0x6F,
                0x70,
                0x71,
                0x72,
                0x73,
                0x74,
                0x75,
                0x76,
                0x77,
                0x78,
                0x79,
                0x7A};

        private static final int byteToUnsigned(byte b) {
            int value = (int)b;

            return (value >= 0? value: value + 256);
        }

        private static int fourBytesToInt(byte b[], int offset) {
            int value;

            value = byteToUnsigned(b[offset++]);
            value |= (byteToUnsigned(b[offset++]) << 8);
            value |= (byteToUnsigned(b[offset++]) << 16);
            value |= (byteToUnsigned(b[offset++]) << 24);

            return (value);
        }

        private static final void intToFourBytes(int iValue, byte b[], int offset) {
            b[offset++] = (byte)((iValue) & 0xff);
            b[offset++] = (byte)((iValue >>> 8) & 0xff);
            b[offset++] = (byte)((iValue >>> 16) & 0xff);
            b[offset++] = (byte)((iValue >>> 24) & 0xff);
        }

        private static final void PERM_OP(int a, int b, int n, int m, int results[]) {
            int t;

            t = ((a >>> n) ^ b) & m;
            a ^= t << n;
            b ^= t;

            results[0] = a;
            results[1] = b;
        }

        private static final int HPERM_OP(int a, int n, int m) {
            int t;

            t = ((a << (16 - n)) ^ a) & m;
            a = a ^ t ^ (t >>> (16 - n));

            return (a);
        }

        private static int[] des_set_key(byte key[]) {
            int schedule[] = new int[ITERATIONS * 2];

            int c = fourBytesToInt(key, 0);
            int d = fourBytesToInt(key, 4);

            int results[] = new int[2];

            PERM_OP(d, c, 4, 0x0f0f0f0f, results);
            d = results[0];
            c = results[1];

            c = HPERM_OP(c, -2, 0xcccc0000);
            d = HPERM_OP(d, -2, 0xcccc0000);

            PERM_OP(d, c, 1, 0x55555555, results);
            d = results[0];
            c = results[1];

            PERM_OP(c, d, 8, 0x00ff00ff, results);
            c = results[0];
            d = results[1];

            PERM_OP(d, c, 1, 0x55555555, results);
            d = results[0];
            c = results[1];

            d = (((d & 0x000000ff) << 16) | (d & 0x0000ff00) | ((d & 0x00ff0000) >>> 16) | ((c & 0xf0000000) >>> 4));
            c &= 0x0fffffff;

            int s, t;
            int j = 0;

            for (int i = 0; i < ITERATIONS; i++) {
                if (shifts2[i]) {
                    c = (c >>> 2) | (c << 26);
                    d = (d >>> 2) | (d << 26);
                } else {
                    c = (c >>> 1) | (c << 27);
                    d = (d >>> 1) | (d << 27);
                }

                c &= 0x0fffffff;
                d &= 0x0fffffff;

                s = skb[0][(c) & 0x3f] | skb[1][((c >>> 6) & 0x03) | ((c >>> 7) & 0x3c)]
                        | skb[2][((c >>> 13) & 0x0f) | ((c >>> 14) & 0x30)]
                        | skb[3][((c >>> 20) & 0x01) | ((c >>> 21) & 0x06) | ((c >>> 22) & 0x38)];

                t = skb[4][(d) & 0x3f] | skb[5][((d >>> 7) & 0x03) | ((d >>> 8) & 0x3c)]
                        | skb[6][(d >>> 15) & 0x3f]
                        | skb[7][((d >>> 21) & 0x0f) | ((d >>> 22) & 0x30)];

                schedule[j++] = ((t << 16) | (s & 0x0000ffff)) & 0xffffffff;
                s = ((s >>> 16) | (t & 0xffff0000));

                s = (s << 4) | (s >>> 28);
                schedule[j++] = s & 0xffffffff;
            }
            return (schedule);
        }

        private static final int D_ENCRYPT(int L, int R, int S, int E0, int E1, int s[]) {
            int t, u, v;

            v = R ^ (R >>> 16);
            u = v & E0;
            v = v & E1;
            u = (u ^ (u << 16)) ^ R ^ s[S];
            t = (v ^ (v << 16)) ^ R ^ s[S + 1];
            t = (t >>> 4) | (t << 28);

            L ^= SPtrans[1][(t) & 0x3f] | SPtrans[3][(t >>> 8) & 0x3f]
                    | SPtrans[5][(t >>> 16) & 0x3f] | SPtrans[7][(t >>> 24) & 0x3f]
                    | SPtrans[0][(u) & 0x3f] | SPtrans[2][(u >>> 8) & 0x3f]
                    | SPtrans[4][(u >>> 16) & 0x3f] | SPtrans[6][(u >>> 24) & 0x3f];

            return (L);
        }

        private static final int[] body(int schedule[], int Eswap0, int Eswap1) {
            int left = 0;
            int right = 0;
            int t = 0;

            for (int j = 0; j < 25; j++) {
                for (int i = 0; i < ITERATIONS * 2; i += 4) {
                    left = D_ENCRYPT(left, right, i, Eswap0, Eswap1, schedule);
                    right = D_ENCRYPT(right, left, i + 2, Eswap0, Eswap1, schedule);
                }
                t = left;
                left = right;
                right = t;
            }

            t = right;

            right = (left >>> 1) | (left << 31);
            left = (t >>> 1) | (t << 31);

            left &= 0xffffffff;
            right &= 0xffffffff;

            int results[] = new int[2];

            PERM_OP(right, left, 1, 0x55555555, results);
            right = results[0];
            left = results[1];

            PERM_OP(left, right, 8, 0x00ff00ff, results);
            left = results[0];
            right = results[1];

            PERM_OP(right, left, 2, 0x33333333, results);
            right = results[0];
            left = results[1];

            PERM_OP(left, right, 16, 0x0000ffff, results);
            left = results[0];
            right = results[1];

            PERM_OP(right, left, 4, 0x0f0f0f0f, results);
            right = results[0];
            left = results[1];

            int out[] = new int[2];

            out[0] = left;
            out[1] = right;

            return (out);
        }

        public static final String crypt(String salt, String original) {
            while (salt.length() < 2)
                salt += "A";

            StringBuffer buffer = new StringBuffer("             ");

            char charZero = salt.charAt(0);
            char charOne = salt.charAt(1);

            buffer.setCharAt(0, charZero);
            buffer.setCharAt(1, charOne);

            int Eswap0 = con_salt[(int)charZero];
            int Eswap1 = con_salt[(int)charOne] << 4;

            byte key[] = new byte[8];

            for (int i = 0; i < key.length; i++)
                key[i] = (byte)0;

            for (int i = 0; i < key.length && i < original.length(); i++) {
                int iChar = (int)original.charAt(i);

                key[i] = (byte)(iChar << 1);
            }

            int schedule[] = des_set_key(key);
            int out[] = body(schedule, Eswap0, Eswap1);

            byte b[] = new byte[9];

            intToFourBytes(out[0], b, 0);
            intToFourBytes(out[1], b, 4);
            b[8] = 0;

            for (int i = 2, y = 0, u = 0x80; i < 13; i++) {
                for (int j = 0, c = 0; j < 6; j++) {
                    c <<= 1;

                    if (((int)b[y] & u) != 0) c |= 1;

                    u >>>= 1;

                    if (u == 0) {
                        y++;
                        u = 0x80;
                    }
                    buffer.setCharAt(i, (char)cov_2char[c]);
                }
            }
            return (buffer.toString());
        }

        public static void main(String args[]) {
            if (args.length >= 2) {
                System.out.println("[" + args[0] + "] [" + args[1] + "] => ["
                        + jcrypt.crypt(args[0], args[1]) + "]");
            }
        }
    }

    // ------------------------------------------------------------------------
    public static class Hash {
        private Hash() {
        }

        private static final byte[] getBytes(String s) {
            try {
                return s.getBytes(UTF_8);
            } catch (UnsupportedEncodingException e) {
                return s.getBytes();
            }
        }

        public static final String crypt(String original, String salt) {
            return jcrypt.crypt(salt, original);
        }

        public static final byte[] md5_bytes(String original) {
            MessageDigest md;
            try {
                md = MessageDigest.getInstance("MD5");
            } catch (NoSuchAlgorithmException e) {
                throw new AssertionError("MD5 n'est pas disponible: " + getSummary(e));
            }
            return md.digest(getBytes(original));
        }

        public static final String md5(String original) {
            return Base64.encodeBytes(md5_bytes(original));
        }

        public static final byte[] smd5_bytes(String original, byte[] salt) {
            if (salt == null) salt = new byte[0];
            MessageDigest md;
            try {
                md = MessageDigest.getInstance("MD5");
            } catch (NoSuchAlgorithmException e) {
                throw new AssertionError("MD5 n'est pas disponible: " + getSummary(e));
            }
            md.update(getBytes(original));
            md.update(salt);
            byte[] tmpdigest = md.digest();
            byte[] digest = new byte[tmpdigest.length + salt.length];
            System.arraycopy(tmpdigest, 0, digest, 0, tmpdigest.length);
            System.arraycopy(salt, 0, digest, tmpdigest.length, salt.length);
            return digest;
        }

        public static final String smd5(String original, byte[] salt) {
            return Base64.encodeBytes(smd5_bytes(original, salt));
        }

        public static final byte[] sha_bytes(String original) {
            MessageDigest md;
            try {
                md = MessageDigest.getInstance("SHA-1");
            } catch (NoSuchAlgorithmException e) {
                throw new AssertionError("SHA-1 n'est pas disponible: " + getSummary(e));
            }
            return md.digest(getBytes(original));
        }

        public static final String sha(String original) {
            return Base64.encodeBytes(sha_bytes(original));
        }

        public static final byte[] ssha_bytes(String original, byte[] salt) {
            if (salt == null) salt = new byte[0];
            MessageDigest md;
            try {
                md = MessageDigest.getInstance("SHA-1");
            } catch (NoSuchAlgorithmException e) {
                throw new AssertionError("SHA-1 n'est pas disponible: " + getSummary(e));
            }
            md.update(getBytes(original));
            md.update(salt);
            byte[] tmpdigest = md.digest();
            byte[] digest = new byte[tmpdigest.length + salt.length];
            System.arraycopy(tmpdigest, 0, digest, 0, tmpdigest.length);
            System.arraycopy(salt, 0, digest, tmpdigest.length, salt.length);
            return digest;
        }

        public static final String ssha(String original, byte[] salt) {
            return Base64.encodeBytes(ssha_bytes(original, salt));
        }

        public static final byte[] ntHash_bytes(String password) {
            try {
                return new MD4().digest(password.getBytes("UTF-16LE"));
            } catch (UnsupportedEncodingException e) {
                throw new AssertionError("UTF-16LE n'est pas disponible: " + getSummary(e));
            }
        }

        public static final String ntlm(String password) {
            return toHex(ntHash_bytes(password));
        }

        public static final byte[] lmHash_bytes(String password) {
            try {
                byte[] oemPassword = password.toUpperCase().getBytes("US-ASCII");
                int length = Math.min(oemPassword.length, 14);
                byte[] keys = new byte[14];
                System.arraycopy(oemPassword, 0, keys, 0, length);
                byte[] magic = "KGS!@#$%".getBytes("US-ASCII");
                byte[] key1 = new byte[7];
                System.arraycopy(keys, 0, key1, 0, 7);
                byte[] hash1 = new DES(key1).encrypt(magic);
                byte[] key2 = new byte[7];
                System.arraycopy(keys, 7, key2, 0, 7);
                byte[] hash2 = new DES(key2).encrypt(magic);
                byte[] hash = new byte[16];
                System.arraycopy(hash1, 0, hash, 0, 8);
                System.arraycopy(hash2, 0, hash, 8, 8);
                return hash;
            } catch (UnsupportedEncodingException e) {
                throw new AssertionError("US-ASCII n'est pas disponible: " + getSummary(e));
            }
        }

        public static final String lm(String password) {
            return toHex(lmHash_bytes(password));
        }
    }

    // ------------------------------------------------------------------------
    public static class Salt {
        public static final String getCryptSalt(String pw) {
            if (pw == null) return null;
            if (strSubstr(pw, 0, 7).equalsIgnoreCase("{crypt}")) pw = strSubstr(pw, 7);
            return strSubstr(pw, 0, 2);
        }

        public static final byte[] getSmd5Salt(byte[] digest) {
            if (digest == null) return null;
            if (digest.length <= 16) return new byte[0];
            byte[] salt = new byte[digest.length - 16];
            System.arraycopy(digest, 16, salt, 0, salt.length);
            return salt;
        }

        public static final byte[] getSmd5Salt(String pw) {
            if (pw == null) return null;
            if (strSubstr(pw, 0, 6).equalsIgnoreCase("{smd5}")) pw = strSubstr(pw, 6);
            try {
                return getSmd5Salt(Base64.decode(pw));
            } catch (IllegalArgumentException e) {
                return null;
            }
        }

        public static final byte[] getSshaSalt(byte[] digest) {
            if (digest == null) return null;
            if (digest.length <= 20) return new byte[0];
            byte[] salt = new byte[digest.length - 20];
            System.arraycopy(digest, 20, salt, 0, salt.length);
            return salt;
        }

        public static final byte[] getSshaSalt(String pw) {
            if (pw == null) return null;
            if (strSubstr(pw, 0, 6).equalsIgnoreCase("{ssha}")) pw = strSubstr(pw, 6);
            try {
                return getSshaSalt(Base64.decode(pw));
            } catch (IllegalArgumentException e) {
                return null;
            }
        }

        public static final String[] DEFAULT_CRYPT_SALT_SECTIONS = new String[] {
                PasswordGenerator.UPPER,
                PasswordGenerator.LOWER,
                PasswordGenerator.NUMBERS};

        private static final SecureRandom newSecureRandom() {
            try {
                return SecureRandom.getInstance("SHA1PRNG");
            } catch (NoSuchAlgorithmException e) {
                return new SecureRandom();
            }
        }

        private static final SecureRandom DEFAULT_RAND = newSecureRandom();

        private static final int DEFAULT_BINARY_SALT_LENGTH = 20;

        private static final Salt instance = new Salt();

        public static final Salt getInstance() {
            return instance;
        }

        public Salt(Random rand, String[] cryptSaltSections, int binarySaltLength) {
            if (rand == null) rand = DEFAULT_RAND;
            if (cryptSaltSections == null) cryptSaltSections = DEFAULT_CRYPT_SALT_SECTIONS;
            if (binarySaltLength <= 0) binarySaltLength = DEFAULT_BINARY_SALT_LENGTH;
            this.rand = rand;
            this.cryptSaltSections = cryptSaltSections;
            this.binarySaltLength = binarySaltLength;
        }

        public Salt() {
            this(null, null, -1);
        }

        private Random rand;

        private String[] cryptSaltSections;

        public String newCryptSalt() {
            return PasswordGenerator.generate(2, cryptSaltSections, rand);
        }

        private int binarySaltLength;

        public byte[] newBinarySalt() {
            byte[] salt = new byte[binarySaltLength];
            rand.nextBytes(salt);
            return salt;
        }
    }

    // ------------------------------------------------------------------------
    public static class Password {
        public static final String CLEARTEXT = "", CRYPT = "CRYPT", MD5 = "MD5", SMD5 = "SMD5",
                SHA = "SHA", SSHA = "SSHA";

        public static final String DEFAULT_SCHEME = SSHA;

        public static final List<String> VALID_SCHEMES = Arrays.asList(new String[] {
                CLEARTEXT,
                CRYPT,
                MD5,
                SMD5,
                SHA,
                SSHA});

        private static final Pattern NORMALIZED_FORMAT = Pattern.compile("\\{.+\\}.+");

        public static final boolean isNormalizedFormat(String pw) {
            return NORMALIZED_FORMAT.matcher(pw).matches();
        }

        public static final String getNormalizedScheme(String pw) {
            if (pw == null) return null;
            if (isNormalizedFormat(pw)) {
                int p = pw.indexOf('}');
                return strSubstr(pw, 1, p).toUpperCase();
            } else {
                return CLEARTEXT;
            }
        }

        public static final String getNormalizedPassword(String pw) {
            if (pw == null) return null;
            if (isNormalizedFormat(pw)) {
                int p = pw.indexOf('}');
                return strSubstr(pw, p + 1);
            } else {
                return pw;
            }
        }

        public static final boolean isClearScheme(String pw) {
            String scheme = getNormalizedScheme(pw);
            return strIsempty(scheme) || CLEARTEXT.equals(scheme);
        }

        public static final boolean isCryptScheme(String pw) {
            return CRYPT.equals(getNormalizedScheme(pw));
        }

        public static final boolean isMd5Scheme(String pw) {
            return MD5.equals(getNormalizedScheme(pw));
        }

        public static final boolean isSmd5Scheme(String pw) {
            return SMD5.equals(getNormalizedScheme(pw));
        }

        public static final boolean isShaScheme(String pw) {
            return SHA.equals(getNormalizedScheme(pw));
        }

        public static final boolean isSshaScheme(String pw) {
            return SSHA.equals(getNormalizedScheme(pw));
        }

        public static final boolean validate(String clear, String normalized) {
            return new Password(normalized).validate(clear);
        }

        private Salt saltGenerator = Salt.getInstance();

        public void setSaltGenerator(Salt saltGenerator) {
            if (saltGenerator == null) saltGenerator = Salt.getInstance();
            this.saltGenerator = saltGenerator;
        }

        public Password(String clear, String scheme, String crypted, String lmhash, String ntlmhash) {
            reset(true);
            setClear(clear);
            setScheme(scheme);
            this.crypted = crypted;
            this.lmHash = lmhash;
            this.ntlmHash = ntlmhash;
        }

        public Password() {
            reset(true);
        }

        public Password(String normalized) {
            this();
            setNormalized(normalized);
        }

        public Password(String clear, String scheme) {
            this();
            setClear(clear);
            setScheme(scheme);
        }

        protected String scheme;

        public String getScheme() {
            return scheme;
        }

        public void setScheme(String scheme) {
            if (scheme == null) scheme = DEFAULT_SCHEME;
            if (!strEquals(this.scheme, scheme)) {
                this.scheme = scheme;
                crypted = normalized = ntlmHash = lmHash = null;
            }
        }

        public boolean isClearScheme() {
            return strIsempty(scheme) || CLEARTEXT.equals(scheme);
        }

        public boolean isCryptScheme() {
            return CRYPT.equals(scheme);
        }

        public boolean isMd5Scheme() {
            return MD5.equals(scheme);
        }

        public boolean isSmd5Scheme() {
            return SMD5.equals(scheme);
        }

        public boolean isShaScheme() {
            return SHA.equals(scheme);
        }

        public boolean isSshaScheme() {
            return SSHA.equals(scheme);
        }

        private final void reset(boolean resetScheme) {
            if (resetScheme || scheme == null) scheme = DEFAULT_SCHEME;
            clear = crypted = normalized = lmHash = ntlmHash = null;
        }

        protected String clear;

        public boolean hasClear() {
            return clear != null;
        }

        protected String randomCryptSalt() {
            return saltGenerator.newCryptSalt();
        }

        protected byte[] randomBinarySalt() {
            return saltGenerator.newBinarySalt();
        }

        public static class NotAvailableException extends Exception {
            private static final long serialVersionUID = 1L;

            public static final String DEFAULT_MESSAGE = "Mot de passe en clair non disponible";

            public NotAvailableException(String message) {
                super(message != null? message: DEFAULT_MESSAGE);
            }

            public NotAvailableException() {
                this(null);
            }
        }

        public String getClear() throws NotAvailableException {
            if (clear != null || crypted == null) return clear;
            throw new NotAvailableException();
        }

        public String getClearOrNull() {
            return clear;
        }

        public Password setClear(String clear) {
            reset(false);
            this.clear = clear;
            return this;
        }

        public Password setCryptPassword(String pw) {
            reset(true);
            scheme = CRYPT;
            crypted = pw;
            return this;
        }

        public Password setMd5Password(String pw) {
            reset(true);
            scheme = MD5;
            crypted = pw;
            return this;
        }

        public Password setSmd5Password(String pw) {
            reset(true);
            scheme = SMD5;
            crypted = pw;
            return this;
        }

        public Password setShaPassword(String pw) {
            reset(true);
            scheme = SHA;
            crypted = pw;
            return this;
        }

        public Password setSshaPassword(String pw) {
            reset(true);
            scheme = SSHA;
            crypted = pw;
            return this;
        }

        private static final String DISABLED = "*disabled*";

        public boolean isDisabled() {
            return !isClearScheme() && strEquals(crypted, DISABLED);
        }

        public Password setDisabled(boolean disabled) {
            if (disabled && !isClearScheme()) {
                crypted = DISABLED;
                ntlmHash = lmHash = normalized = null;
            }
            return this;
        }

        protected String crypted;

        public String getCrypted() {
            if (isClearScheme()) return clear;
            else if (crypted == null) {
                if (isCryptScheme()) crypted = Hash.crypt(clear, randomCryptSalt());
                else if (isMd5Scheme()) crypted = Hash.md5(clear);
                else if (isSmd5Scheme()) crypted = Hash.smd5(clear, randomBinarySalt());
                else if (isShaScheme()) crypted = Hash.sha(clear);
                else if (isSshaScheme()) crypted = Hash.ssha(clear, randomBinarySalt());
                else throw new IllegalStateException("Type de cryptage non reconnu: " + scheme);
            }
            return crypted;
        }

        public String getCryptCrypted() {
            if (isCryptScheme()) return getCrypted();
            else if (clear != null) return Hash.crypt(clear, randomCryptSalt());
            else return null;
        }

        public String getMd5Crypted() {
            if (isMd5Scheme()) return getCrypted();
            else if (clear != null) return Hash.md5(clear);
            else return null;
        }

        public String getSmd5Crypted() {
            if (isSmd5Scheme()) return getCrypted();
            else if (clear != null) return Hash.smd5(clear, randomBinarySalt());
            else return null;
        }

        public String getShaCrypted() {
            if (isShaScheme()) return getCrypted();
            else if (clear != null) return Hash.sha(clear);
            else return null;
        }

        public String getSshaCrypted() {
            if (isSshaScheme()) return getCrypted();
            else if (clear != null) return Hash.ssha(clear, randomBinarySalt());
            else return null;
        }

        protected String normalized;

        public String getNormalized() {
            if (normalized == null) {
                if (isClearScheme()) normalized = clear;
                else normalized = "{" + scheme + "}" + getCrypted();
            }
            return normalized;
        }

        public String toString() {
            return getNormalized();
        }

        public Password setNormalized(String pw) {
            reset(true);
            if (pw == null) {
                scheme = CLEARTEXT;
                return this;
            }
            if (isNormalizedFormat(pw)) {
                int p = pw.indexOf('}');
                scheme = strSubstr(pw, 1, p).toUpperCase();
                crypted = strSubstr(pw, p + 1);
            } else {
                scheme = CLEARTEXT;
                clear = pw;
            }
            return this;
        }

        private static final String NTLM_DISABLED = "XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX";

        protected String ntlmHash;

        public String getNtlmHash() throws NotAvailableException {
            if (ntlmHash == null) {
                if (isDisabled()) ntlmHash = NTLM_DISABLED;
                else if (hasClear()) ntlmHash = Hash.ntlm(clear);
                else throw new NotAvailableException();
            }
            return ntlmHash;
        }

        protected String lmHash;

        public String getLmHash() throws NotAvailableException {
            if (lmHash == null) {
                if (isDisabled()) lmHash = NTLM_DISABLED;
                else if (hasClear()) lmHash = Hash.lm(clear);
                else throw new NotAvailableException();
            }
            return lmHash;
        }

        public boolean validate(String userPassword) {
            getNormalized();
            if (isClearScheme()) {
                return strEquals(strNotnull(this.clear), strNotnull(userPassword));
            }
            if (userPassword == null) return false;
            if (isCryptScheme()) {
                return Hash.crypt(userPassword, Salt.getCryptSalt(crypted)).equals(crypted);
            } else if (isMd5Scheme()) {
                return Hash.md5(userPassword).equals(crypted);
            } else if (isSmd5Scheme()) {
                return Hash.smd5(userPassword, Salt.getSmd5Salt(crypted)).equals(crypted);
            } else if (isShaScheme()) {
                return Hash.sha(userPassword).equals(crypted);
            } else if (isSshaScheme()) {
                return Hash.ssha(userPassword, Salt.getSshaSalt(crypted)).equals(crypted);
            }
            return false;
        }

        public boolean equals(Object obj) {
            if (obj == null) return false;
            else if (obj instanceof String) {
                String pw = (String)obj;
                if (isNormalizedFormat(pw)) {
                    return strEquals(getNormalized(), pw);
                } else return validate((String)obj);
            } else if (obj instanceof Password) {
                return strEquals(getNormalized(), ((Password)obj).getNormalized());
            }
            throw new IllegalArgumentException("Ne peut comparer qu'avec une instance de "
                    + getClass() + " (obtenu une " + "instance de " + obj.getClass() + ")");
        }
    }

    // ------------------------------------------------------------------------
    public static class PasswordChecker {
        public PasswordChecker(int minLen, int minUpper, int minLower, int minAlpha, int minNumber,
                int minSymbol, int minSpecial, boolean allowMultibytes) {
            this.minLen = minLen;
            this.minUpper = minUpper;
            this.minLower = minLower;
            this.minAlpha = minAlpha;
            this.minNumber = minNumber;
            this.minSymbol = minSymbol;
            this.minSpecial = minSpecial;
            this.allowMultibytes = allowMultibytes;
        }

        public PasswordChecker() {
            this(MIN_LEN, MIN_UPPER, MIN_LOWER, MIN_ALPHA, MIN_NUMBER, MIN_SYMBOL, MIN_SPECIAL,
                    ALLOW_MULTIBYTES);
        }

        private int minLen;

        public int getMinLen() {
            return minLen;
        }

        public void setMinLen(int minLen) {
            this.minLen = minLen;
        }

        private int minUpper;

        public int getMinUpper() {
            return minUpper;
        }

        public void setMinUpper(int minUpper) {
            this.minUpper = minUpper;
        }

        private int minLower;

        public int getMinLower() {
            return minLower;
        }

        public void setMinLower(int minLower) {
            this.minLower = minLower;
        }

        private int minAlpha;

        public int getMinAlpha() {
            return minAlpha;
        }

        public void setMinAlpha(int minAlpha) {
            this.minAlpha = minAlpha;
        }

        private int minNumber;

        public int getMinNumber() {
            return minNumber;
        }

        public void setMinNumber(int minNumber) {
            this.minNumber = minNumber;
        }

        private int minSymbol;

        public int getMinSymbol() {
            return minSymbol;
        }

        public void setMinSymbol(int minSymbol) {
            this.minSymbol = minSymbol;
        }

        private int minSpecial;

        public int getMinSpecial() {
            return minSpecial;
        }

        public void setMinSpecial(int minSpecial) {
            this.minSpecial = minSpecial;
        }

        private boolean allowMultibytes;

        public boolean isAllowMultibytes() {
            return allowMultibytes;
        }

        public void setAllowMultibytes(boolean allowMultibytes) {
            this.allowMultibytes = allowMultibytes;
        }

        /**
         * Obtenir une description des caractéristiques du mot de passe. La chaine retournée est de
         * la forme '$prefix doit faire au moins N caractères, et contenir au moins 2 lettres, 2
         * chiffres ou caractères spéciaux'.
         * <p>
         * Les valeurs actualXxx sont les nombres effectifs de caractères de chaque classe, ou -1 si
         * leur valeur doit être ignorée. Si le nombre effectif est satisfaisant, cette
         * caractéristique n'est pas mentionnée. hasMultibytes==<code>true</code> si le mot de passe
         * contient des caractères qui doivent être encodés sur plusieurs octets en UTF-8.
         * </p>
         */
        public String getQualityDescription(String prefix, int actualLen, int actualUpper,
                int actualLower, int actualAlpha, int actualNumber, int actualSymbol,
                int actualSpecial, boolean hasMultibytes) {
            if (prefix == null) prefix = "Le mot de passe";
            if (minLen <= 0 && minUpper <= 0 && minLower <= 0 && minAlpha <= 0 && minNumber <= 0
                    && minSymbol <= 0 && minSpecial <= 0 && allowMultibytes) {
                return prefix + " n'as pas de restrictions particulières";
            }
            StringBuilder sb = new StringBuilder();
            sb.append(prefix);
            boolean mb = false;
            if (!allowMultibytes && hasMultibytes) {
                sb.append(" ne doit pas contenir de caractères accentués");
                mb = true;
            }
            if (mb) sb.append(", doit ");
            else sb.append(" doit ");
            String ccPrefix;
            if (minLen > 0 && (actualLen == -1 || actualLen < minLen)) {
                sb.append("faire au moins ");
                sb.append(minLen);
                sb.append(" caractères");
                ccPrefix = ", et contenir au moins";
            } else {
                sb.append("contenir au moins");
                ccPrefix = "";
            }
            boolean first = true;
            if (minUpper > 0 && (actualUpper == -1 || actualUpper < minUpper)) {
                if (first) sb.append(ccPrefix);
                else sb.append(",");
                sb.append(" ");
                sb.append(minUpper);
                if (minUpper > 1) sb.append(" lettres majuscules");
                else sb.append(" lettre majuscule");
                first = false;
            }
            if (minLower > 0 && (actualLower == -1 || actualLower < minLower)) {
                if (first) sb.append(ccPrefix);
                else sb.append(",");
                sb.append(" ");
                sb.append(minLower);
                if (minLower > 1) sb.append(" lettres minuscules");
                else sb.append(" lettre minuscule");
                first = false;
            }
            if (minAlpha > 0 && (actualAlpha == -1 || actualAlpha < minAlpha)) {
                if (first) sb.append(ccPrefix);
                else sb.append(",");
                sb.append(" ");
                sb.append(minAlpha);
                if (minAlpha > 1) sb.append(" lettres");
                else sb.append(" lettre");
                first = false;
            }
            if (minNumber > 0 && (actualNumber == -1 || actualNumber < minNumber)) {
                if (first) sb.append(ccPrefix);
                else sb.append(",");
                sb.append(" ");
                sb.append(minNumber);
                if (minNumber > 1) sb.append(" chiffres");
                else sb.append(" chiffre");
                first = false;
            }
            if (minSymbol > 0 && (actualSymbol == -1 || actualSymbol < minSymbol)) {
                if (first) sb.append(ccPrefix);
                else sb.append(",");
                sb.append(" ");
                sb.append(minSymbol);
                if (minSymbol > 1) sb.append(" caractères spéciaux");
                else sb.append(" caractère spécial");
                first = false;
            }
            if (minSpecial > 0 && (actualSpecial == -1 || actualSpecial < minSpecial)) {
                if (first) sb.append(ccPrefix);
                else sb.append(",");
                sb.append(" ");
                sb.append(minSpecial);
                if (minSpecial > 1) sb.append(" chiffres ou caractères spéciaux");
                else sb.append(" chiffre ou caractère spécial");
                first = false;
            }
            return sb.toString();
        }

        public String getQualityDescription(String prefix) {
            return getQualityDescription(prefix, -1, -1, -1, -1, -1, -1, -1, false);
        }

        private static final Pattern RE_NOT_UPPER = Pattern.compile("[^A-Z]");

        private static final Pattern RE_NOT_LOWER = Pattern.compile("[^a-z]");

        private static final Pattern RE_NOT_ALPHA = Pattern.compile("[^a-zA-Z]");

        private static final Pattern RE_NOT_NUMBER = Pattern.compile("[^0-9]");

        private static final Pattern RE_NOT_SYMBOL = Pattern.compile("[a-zA-Z0-9]");

        private static final Pattern RE_NOT_SPECIAL = Pattern.compile("[a-zA-Z]");

        /**
         * Vérifier que le mot de spécifié est de qualité.
         * 
         * @return null si le mot de passe est correct. Sinon, retourner un message qui indique ce
         *         qui ne va pas.
         */
        public String validateQuality(String password) {
            password = strNotnull(password);
            String upper = RE_NOT_UPPER.matcher(password).replaceAll("");
            String lower = RE_NOT_LOWER.matcher(password).replaceAll("");
            String alpha = RE_NOT_ALPHA.matcher(password).replaceAll("");
            String number = RE_NOT_NUMBER.matcher(password).replaceAll("");
            String symbol = RE_NOT_SYMBOL.matcher(password).replaceAll("");
            String special = RE_NOT_SPECIAL.matcher(password).replaceAll("");
            int nbChars = password.length();
            int nbBytes;
            try {
                byte[] passwordBytes = password.getBytes(UTF_8);
                nbBytes = passwordBytes.length;
            } catch (UnsupportedEncodingException e) {
                nbBytes = 0;
            }
            boolean hasMultibytes = nbChars != nbBytes;
            String qualityDescription = getQualityDescription(
                    "Il",
                    password.length(),
                    upper.length(),
                    lower.length(),
                    alpha.length(),
                    number.length(),
                    symbol.length(),
                    special.length(),
                    hasMultibytes);
            if (password.length() < minLen) {
                return "Votre mot de passe est trop court (" + qualityDescription + ")";
            }
            if (!allowMultibytes && hasMultibytes) {
                return "Votre mot de passe ne convient pas (" + qualityDescription + ")";
            }
            if (upper.length() < minUpper || lower.length() < minLower || alpha.length() < minAlpha
                    || number.length() < minNumber || symbol.length() < minSymbol
                    || special.length() < minSpecial) {
                return "Votre mot de passe est trop simple (" + qualityDescription + ")";
            }
            return null;
        }
    }

    // ------------------------------------------------------------------------
    public static class PasswordGenerator {
        public static final String UPPER = "AZERTYUIOPQSDFGHJKLMWXCVBN",
                LOWER = "azertyuiopqsdfghjklmwxcvbn", NUMBERS = "1234567890",
                SYMBOLS = "&\"'(-_)=^$*!:;,?./%+#{[|\\]}";

        public static final String[] DEFAULT_SECTIONS = new String[] {
                UPPER,
                LOWER,
                NUMBERS,
                SYMBOLS};

        public static final int[] DEFAULT_MIN_COUNTS = new int[] {2, 2, 2, 1};

        private static final Random DEFAULT_RAND = new Random();

        private static final PasswordGenerator instance = new PasswordGenerator();

        public static PasswordGenerator getInstance() {
            return instance;
        }

        public static final String generate(int minLen, String[] sections, int[] minCounts,
                Random rand) {
            if (sections == null) sections = DEFAULT_SECTIONS;
            if (minCounts == null) minCounts = new int[sections.length];
            if (sections.length != minCounts.length) {
                throw new IllegalArgumentException("Les tableaux sections et minCounts "
                        + "doivent faire la même taille");
            } else {
                int[] tmp = new int[minCounts.length];
                System.arraycopy(minCounts, 0, tmp, 0, minCounts.length);
                minCounts = tmp;
            }
            StringBuffer sb = new StringBuffer();
            int i = 0;
            while (true) {
                int index;
                do {
                    index = rand.nextInt(sections.length);
                } while (minCounts[index] <= 0);
                String section = sections[index];
                sb.append(section.charAt(rand.nextInt(section.length())));
                minCounts[index]--;
                i++;
                boolean done = true;
                for (int j = 0; j < minCounts.length; j++) {
                    if (minCounts[j] > 0) {
                        done = false;
                        break;
                    }
                }
                if (done) break;
            }
            while (sb.length() < minLen) {
                String section = sections[rand.nextInt(sections.length)];
                sb.append(section.charAt(rand.nextInt(section.length())));
            }
            return sb.toString();
        }

        public static final String generate(int minLen, String[] sections, int[] minCounts) {
            return generate(minLen, sections, minCounts, DEFAULT_RAND);
        }

        public static final String generate(int minLen, int maxLen, String[] sections,
                int[] minCounts, Random rand) {
            int len;
            if (minLen > maxLen) {
                len = minLen;
                minLen = maxLen;
                maxLen = len;
            }
            if (minLen == maxLen) len = minLen;
            else len = rand.nextInt(maxLen - minLen) + minLen;
            return generate(len, sections, minCounts);
        }

        public static final String generate(int minLen, int maxLen, String[] sections,
                int[] minCounts) {
            return generate(minLen, maxLen, sections, minCounts, DEFAULT_RAND);
        }

        public static final String generate(int len, String[] sections, Random rand) {
            StringBuffer sb = new StringBuffer();
            for (int i = 0; i < len; i++) {
                String section = sections[rand.nextInt(sections.length)];
                sb.append(section.charAt(rand.nextInt(section.length())));
            }
            return sb.toString();
        }

        public static final String generate(int len, String[] sections) {
            return generate(len, sections, DEFAULT_RAND);
        }

        public static final String generate(int minLen, int maxLen, String[] sections, Random rand) {
            int len;
            if (minLen > maxLen) {
                len = minLen;
                minLen = maxLen;
                maxLen = len;
            }
            if (minLen == maxLen) len = minLen;
            else len = rand.nextInt(maxLen - minLen) + minLen;
            return generate(len, sections);
        }

        public static final String generate(int minLen, int maxLen, String[] sections) {
            return generate(minLen, maxLen, sections, DEFAULT_RAND);
        }

        public PasswordGenerator(int minLen, String[] sections, int[] minCounts, Random rand) {
            setMinLen(minLen);
            setSections(sections);
            setMinCounts(minCounts);
            setRand(rand);
        }

        public PasswordGenerator() {
            this(-1, null, null, null);
        }

        private int minLen;

        public int getMinLen() {
            return minLen;
        }

        public void setMinLen(int minLen) {
            if (minLen < 0) minLen = MIN_LEN;
            this.minLen = minLen;
        }

        private String[] sections;

        public String[] getSections() {
            return sections;
        }

        public void setSections(String[] sections) {
            if (sections == null) sections = DEFAULT_SECTIONS;
            this.sections = sections;
        }

        private int[] minCounts;

        public int[] getMinCounts() {
            return minCounts;
        }

        public void setMinCounts(int[] minCounts) {
            if (minCounts == null) minCounts = DEFAULT_MIN_COUNTS;
            this.minCounts = minCounts;
        }

        private Random rand;

        public Random getRand() {
            return rand;
        }

        public void setRand(Random rand) {
            if (rand == null) rand = DEFAULT_RAND;
            this.rand = rand;
        }

        public String generate() {
            return generate(minLen, sections, minCounts, rand);
        }

        public String generate(int maxLen) {
            return generate(minLen, maxLen, sections, minCounts, rand);
        }
    }

    // ------------------------------------------------------------------------
    public static class AESEnc {
        private static final String AES = "AES";

        private static final String CIPHER = "AES/ECB/PKCS5Padding";

        public static final byte[] genkey() throws Exception {
            KeyGenerator kg = KeyGenerator.getInstance(AES);
            kg.init(new SecureRandom());
            SecretKey key = kg.generateKey();
            return key.getEncoded();
        }

        public static final String genskey() throws Exception {
            return Base64.encodeBytes(genkey());
        }

        public static final byte[] genkey(String password, byte[] salt, int iterations)
                throws Exception {
            if (salt == null) salt = Salt.getInstance().newBinarySalt();
            if (iterations <= 0) iterations = 10000;
            SecretKeyFactory factory = SecretKeyFactory.getInstance("PBKDF2WithHmacSHA1");
            SecretKey tmp = factory.generateSecret(new PBEKeySpec(password.toCharArray(), salt,
                    iterations, 128));
            SecretKeySpec key = new SecretKeySpec(tmp.getEncoded(), AES);
            return key.getEncoded();
        }

        public static final byte[] genkey(String password) throws Exception {
            return genkey(password, null, -1);
        }

        public static final String genskey(String password, byte[] salt, int iterations)
                throws Exception {
            return Base64.encodeBytes(genkey(password, salt, iterations));
        }

        public static final String genskey(String password) throws Exception {
            return Base64.encodeBytes(genkey(password));
        }

        public static final byte[] getKey(String skey) {
            return Base64.decode(skey);
        }

        public static final String getSkey(byte[] key) {
            return Base64.encodeBytes(key, Base64.DONT_BREAK_LINES);
        }

        public static final byte[] encrypt(byte[] clear, byte[] key) throws Exception {
            if (clear == null) return null;
            if (key == null) throw new NullPointerException("key is required");

            Cipher aes = Cipher.getInstance(CIPHER);
            aes.init(Cipher.ENCRYPT_MODE, new SecretKeySpec(key, AES));
            return aes.doFinal(clear);
        }

        public static final String sencrypt(String sclear, byte[] key) throws Exception {
            if (sclear == null) return null;
            byte[] input = sclear.getBytes(UTF_8);
            byte[] output = encrypt(input, key);
            return Base64.encodeBytes(output, Base64.DONT_BREAK_LINES);
        }

        public static final String sencrypt(String sclear, String skey) throws Exception {
            return sencrypt(sclear, getKey(skey));
        }

        public static final byte[] decrypt(byte[] crypted, byte[] key) throws Exception {
            if (crypted == null) return null;
            if (key == null) throw new NullPointerException("key is required");

            Cipher aes = Cipher.getInstance(CIPHER);
            aes.init(Cipher.DECRYPT_MODE, new SecretKeySpec(key, AES));
            return aes.doFinal(crypted);
        }

        public static final String sdecrypt(String scrypted, byte[] key) throws Exception {
            if (scrypted == null) return null;
            byte[] input = Base64.decode(scrypted);
            byte[] output = decrypt(input, key);
            return new String(output, UTF_8);
        }

        public static final String sdecrypt(String scrypted, String skey) throws Exception {
            return sdecrypt(scrypted, getKey(skey));
        }
    }

    // ------------------------------------------------------------------------

    private Password getPasswordAnySalt(String clear, String scheme, final String anySalt) {
        return new Password(clear, scheme) {
            @Override
            protected String randomCryptSalt() {
                String salt = Salt.getCryptSalt(anySalt);
                if (salt == null) salt = super.randomCryptSalt();
                return salt;
            }

            @Override
            protected byte[] randomBinarySalt() {
                byte[] salt = null;
                if (isSshaScheme()) salt = Salt.getSshaSalt(anySalt);
                else if (isSmd5Scheme()) salt = Salt.getSmd5Salt(anySalt);
                if (salt == null) salt = super.randomBinarySalt();
                return salt;
            }
        };
    }

    private Password getPasswordCryptSalt(String clear, String scheme, final String cryptSalt) {
        return new Password(clear, scheme) {
            @Override
            protected String randomCryptSalt() {
                String salt = cryptSalt;
                if (salt == null) salt = super.randomCryptSalt();
                return salt;
            }
        };
    }

    private Password getPasswordBinarySalt(String clear, String scheme, final byte[] binarySalt) {
        return new Password(clear, scheme) {
            @Override
            protected byte[] randomBinarySalt() {
                byte[] salt = binarySalt;
                if (salt == null) salt = super.randomBinarySalt();
                return salt;
            }
        };
    }

    private static final void die(String msg, Exception e) {
        if (msg != null) System.err.println(msg);
        if (e != null) e.printStackTrace(System.err);
        System.err.flush();
        System.exit(1);
    }

    // actions
    private static enum EAction {
        HASH_PASSWORD, GEN_AESKEY, SHOW_AESKEY, AES_ENCRYPT, AES_DECRYPT;
    }

    private static final void checkExisting(String aeskeyfile) {
        if (aeskeyfile == null) die("Vous devez spécifier l'option -f", null);
        if (!new File(aeskeyfile).exists()) die(aeskeyfile + ": Fichier introuvable", null);
    }

    private static final byte[] readAeskeyfile(String aeskeyfile) {
        if (aeskeyfile == null) return null;
        byte[] aeskey = null;
        try {
            FileInputStream fis = new FileInputStream(aeskeyfile);
            try {
                aeskey = new byte[16];
                fis.read(aeskey);
            } finally {
                fis.close();
            }
        } catch (Exception e) {
            die(null, e);
        }
        return aeskey;
    }

    private static final void writeAeskeyfile(String aeskeyfile, byte[] key) {
        if (aeskeyfile == null) return;
        try {
            FileOutputStream fis = new FileOutputStream(aeskeyfile);
            try {
                fis.write(key);
            } finally {
                fis.close();
            }
        } catch (Exception e) {
            die(null, e);
        }
    }

    private void run(String[] args) {
        if (args.length == 1 && strEquals(args[0], "--help")) {
            println("USAGE:" //
                    + "\n    upassword -p [-f aeskeyfile] [clear [salts...]]"
                    + "\n    upassword -f aeskeyfile -G [password [salt]]"
                    + "\n    upassword -f aeskeyfile -s"
                    + "\n    upassword -f aeskeyfile -e clear"
                    + "\n    upassword -f aeskeyfile -d crypted"
                    + "\n\nOPTIONS"
                    + "\n    -p, --hash-password"
                    + "\n        Crypter un mot de passe (option par défaut). Si le mot de passe en clair"
                    + "\n        et/ou le salt ne sont pas spécifiés, ils sont choisis au hasard. "
                    + "\n    -G, --aes-genkey"
                    + "\n        Générer une clé AES pour utilisation avec les options -s, -e, -d"
                    + "\n    -s, --aes-showkey"
                    + "\n        Afficher encodée en base64 la clé AES contenue dans le fichier spécifié"
                    + "\n    -e, --aes-encrypt"
                    + "\n        Crypter un mot de passe avec la clé AES spécifiée"
                    + "\n    -d, --aes-decrypt"
                    + "\n        Décrypter un mot de passe avec la clé AES spécifiée"
                    + "\n    -f, --aes-keyfile"
                    + "\n        Spécifier le fichier contenant la clé AES. Cette option est obligatoire"
                    + "\n        avec les options -G, -s, -e et -d");
            System.exit(0);
        }

        EAction action = EAction.HASH_PASSWORD;
        String aeskeyfile = null;
        int i = 0, max = args.length;
        while (i < args.length) {
            String arg = args[i];
            if (arg.equals("-p") || arg.equals("--hash-password")) {
                action = EAction.HASH_PASSWORD;
                i++;
            } else if (arg.equals("-G") || arg.equals("--aes-genkey")) {
                action = EAction.GEN_AESKEY;
                i++;
            } else if (arg.equals("-s") || arg.equals("--aes-showkey")) {
                action = EAction.SHOW_AESKEY;
                i++;
            } else if (arg.equals("-e") || arg.equals("--aes-encrypt")) {
                action = EAction.AES_ENCRYPT;
                i++;
            } else if (arg.equals("-d") || arg.equals("--aes-decrypt")) {
                action = EAction.AES_DECRYPT;
                i++;
            } else if (arg.substring(0, 2).equals("-f") || arg.equals("--aes-keyfile")) {
                int shift = 1;
                if (arg.equals("-f") || arg.equals("--aes-keyfile")) {
                    if (args.length > i + 1) {
                        aeskeyfile = args[i + 1];
                        shift = 2;
                    }
                } else {
                    aeskeyfile = arg.substring(2);
                }
                i += shift;
            } else {
                if (arg.equals("--")) i++;
                String[] newargs = new String[args.length - i];
                System.arraycopy(args, i, newargs, 0, newargs.length);
                args = newargs;
                break;
            }
        }

        switch (action) {
        case HASH_PASSWORD: {
            byte[] aeskey = readAeskeyfile(aeskeyfile);

            String clear = null;
            if (args.length > 0) clear = args[0];

            if (clear == null) {
                PasswordGenerator pg = new PasswordGenerator();
                clear = pg.generate();
            }

            if (args.length <= 1) {
                // Pas de salt, afficher simplement les versions cryptées des mots
                // de passe pour tous les schemes. Le salt est choisi au hasard pour
                // chaque mot de passe. Il est donc différent à chaque fois.
                Password p = getPasswordAnySalt(clear, null, null);
                String lm = null;
                String ntlm = null;
                try {
                    lm = p.getLmHash();
                    ntlm = p.getNtlmHash();
                } catch (Password.NotAvailableException e) {
                }
                String crypt = getPasswordAnySalt(clear, Password.CRYPT, null).getNormalized();
                String sha = getPasswordAnySalt(clear, Password.SHA, null).getNormalized();
                String ssha = getPasswordAnySalt(clear, Password.SSHA, null).getNormalized();
                String md5 = getPasswordAnySalt(clear, Password.MD5, null).getNormalized();
                String smd5 = getPasswordAnySalt(clear, Password.SMD5, null).getNormalized();
                String aes = null;
                if (aeskey != null) {
                    try {
                        aes = AESEnc.sencrypt(clear, aeskey);
                    } catch (Exception e) {
                        die("Impossible de crypter avec AES", e);
                    }
                }

                println("clear: " + clear);
                println("lm:    " + lm);
                println("ntlm:  " + ntlm);
                println("crypt: " + crypt);
                println("sha:   " + sha);
                println("ssha:  " + ssha);
                println("md5:   " + md5);
                println("smd5:  " + smd5);
                if (aes != null) println("aes:   " + aes);
            } else {
                // Afficher uniquement les versions cryptées des mots de passe avec
                // les schemes correspondant aux salts spécifiés, pour chacun des
                // salts spécifiés.
                println("clear: " + clear);
                i = 1;
                max = args.length;
                while (i < max) {
                    String salt = args[i++];
                    if (Password.isNormalizedFormat(salt)) {
                        if (Password.isCryptScheme(salt)) {
                            String cryptSalt = Salt.getCryptSalt(salt);
                            String crypt = getPasswordCryptSalt(clear, Password.CRYPT, cryptSalt)
                                    .getNormalized();
                            println("salt:  " + cryptSalt);
                            println("crypt: " + crypt);
                            if (salt.equals(crypt)) println("match: true");
                        } else if (Password.isSshaScheme(salt)) {
                            byte[] sshaSalt = Salt.getSshaSalt(salt);
                            String ssha = getPasswordBinarySalt(clear, Password.SSHA, sshaSalt)
                                    .getNormalized();
                            println("salt:  " + toHex(sshaSalt));
                            println("ssha:  " + ssha);
                            if (salt.equals(ssha)) println("match: true");
                        } else if (Password.isSmd5Scheme(salt)) {
                            byte[] smd5Salt = Salt.getSmd5Salt(salt);
                            String smd5 = getPasswordBinarySalt(clear, Password.SMD5, smd5Salt)
                                    .getNormalized();
                            println("salt:  " + toHex(smd5Salt));
                            println("smd5:  " + smd5);
                            if (salt.equals(smd5)) println("match: true");
                        } else {
                            println("salt:  " + salt + " !not supported");
                        }
                    } else if (isHex(salt)) {
                        byte[] binarySalt = fromHex(salt);
                        String ssha = getPasswordBinarySalt(clear, Password.SSHA, binarySalt)
                                .getNormalized();
                        String smd5 = getPasswordBinarySalt(clear, Password.SMD5, binarySalt)
                                .getNormalized();
                        println("salt:  " + toHex(binarySalt));
                        println("ssha:  " + ssha);
                        println("smd5:  " + smd5);
                        if (salt.length() >= 2) {
                            String cryptSalt = Salt.getCryptSalt(salt);
                            String crypt = getPasswordCryptSalt(clear, Password.CRYPT, cryptSalt)
                                    .getNormalized();
                            if (!salt.equals(cryptSalt)) println("salt:  " + cryptSalt);
                            println("crypt: " + crypt);
                            if (salt.equals(crypt)) println("match: true");
                        }
                        if (salt.equals(ssha)) println("match: true");
                        if (salt.equals(smd5)) println("match: true");
                    } else {
                        String cryptSalt = Salt.getCryptSalt(salt);
                        String crypt = getPasswordCryptSalt(clear, Password.CRYPT, cryptSalt)
                                .getNormalized();
                        println("salt:  " + cryptSalt);
                        println("crypt: " + crypt);
                        if (salt.equals(crypt)) println("match: true");
                    }
                }
            }
            break;
        }

        case GEN_AESKEY: {
            if (aeskeyfile == null) die("Vous devez spécifier l'option -f", null);
            if (new File(aeskeyfile).exists()) {
                die(aeskeyfile + ": Refus d'écraser un fichier existant", null);
            }

            byte[] key = null;
            try {
                String password = null;
                if (args.length > 0) password = args[0];
                byte[] salt = null;
                if (args.length > 1) salt = args[1].getBytes(UTF_8);

                if (password != null && salt != null) {
                    key = AESEnc.genkey();
                } else {
                    key = AESEnc.genkey(password, salt, -1);
                }
            } catch (Exception e) {
                die(null, e);
            }
            writeAeskeyfile(aeskeyfile, key);
            println(AESEnc.getSkey(key));

            break;
        }

        case SHOW_AESKEY: {
            checkExisting(aeskeyfile);
            byte[] key = readAeskeyfile(aeskeyfile);
            println(AESEnc.getSkey(key));
            break;
        }

        case AES_ENCRYPT: {
            checkExisting(aeskeyfile);
            byte[] key = readAeskeyfile(aeskeyfile);
            String sclear = null;
            if (args.length > 0) sclear = args[0];
            if (sclear != null) {
                try {
                    println(AESEnc.sencrypt(sclear, key));
                } catch (Exception e) {
                    die(null, e);
                }
            }
            break;
        }

        case AES_DECRYPT: {
            checkExisting(aeskeyfile);
            byte[] key = readAeskeyfile(aeskeyfile);
            String scrypted = null;
            if (args.length > 0) scrypted = args[0];
            if (scrypted != null) {
                try {
                    println(AESEnc.sdecrypt(scrypted, key));
                } catch (Exception e) {
                    die(null, e);
                }
            }
            break;
        }
        }
    }

    public static void main(String[] args) {
        new upassword().run(args);
    }
}