SaslUtils.java   [plain text]


package CyrusSasl;

public abstract class SaslUtils
{

	/**
	 * a static array that maps 6 bit integers to a specific char
	 */
	private final static char enc_table[] = 
	{ 
	//   0   1   2   3   4   5   6   7 
		'A','B','C','D','E','F','G','H', // 0 
		'I','J','K','L','M','N','O','P', // 1 
		'Q','R','S','T','U','V','W','X', // 2 
		'Y','Z','a','b','c','d','e','f', // 3 
		'g','h','i','j','k','l','m','n', // 4 
		'o','p','q','r','s','t','u','v', // 5 
		'w','x','y','z','0','1','2','3', // 6 
		'4','5','6','7','8','9','+','/'  // 7 
	}; 
	private final static byte dec_table[] = 
	{ 
		-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
		-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
		-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 62, -1, -1, -1, 63,
		52, 53, 54, 55, 56, 57, 58, 59, 60, 61, -1, -1, -1, -1, -1, -1,
		-1, 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, -1, -1, -1, -1, -1,
		-1, 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, -1, -1, -1, -1, -1,
		-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
		-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
		-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
		-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
		-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
		-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
		-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
		-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
	}; 

    /**
     * Base 64 Encodes a String in byte[] form
     *
     * @param buf String to be encoded
     *
     * @return the encoded value in string form
     */

 
	public static String 
	encode64( byte buf[] ) 
	{
		int i = 0; 
		StringBuffer buffer = new StringBuffer(); 
		int len = buf.length; 
		int delta =	len % 3; 
		byte a, b, c;
		for (int count = len / 3; count > 0; count--) 
		{
			a = buf[i++];
			b = buf[i++];
			c = buf[i++];
			buffer.append(enc_table[(a >>> 2) & 0x3F]); 
			buffer.append(enc_table[((a << 4) & 0x30) + ((b >>> 4) & 0xf)]); 
			buffer.append(enc_table[((b << 2) & 0x3c) + ((c >>> 6) & 0x3)]); 
			buffer.append(enc_table[c & 0x3F]); 

			/*			if (i != 0 && i%57 == 0)
						buffer.append("\r\n");*/
		}

		if (delta == 1) 
		{
			a = buf[i++];
			buffer.append(enc_table[(a >>> 2) & 0x3F]); 
			buffer.append(enc_table[((a << 4) & 0x30)]); 
			buffer.append('='); 
			buffer.append('='); 
		}
		if (delta == 2) 
		{
			a = buf[i++];
			b = buf[i++];
			buffer.append(enc_table[(a >>> 2) & 0x3F]); 
			buffer.append(enc_table[((a << 4) & 0x30) + ((b >>> 4) & 0xf)]); 
			buffer.append(enc_table[((b << 2) & 0x3c)]); 
			buffer.append('='); 
		}

		/*buffer.append("\r\n");*/
		return buffer.toString(); 
	}

    /**
     * Base 64 Encodes a String in String form
     *
     * @param s String to be encoded
     *
     * @return the encoded value in string form
     */

    public static String encode64( String s ) 
    {
      return encode64( s.getBytes() );	
    }


    /**
     * Base 64 Decodes a byte[] string
     * Ignores trailing whitespace and newlines
     *
     * @param buf buffer to be decoded
     *
     * @return the decoded value in byte[] form
     */


	public static byte[]
	decode64( byte buf[] ) 
	{
                int padCount = 0;
                int i, len = buf.length;
		int real_len = 0;

		for (i=len-1; i >= 0; --i)
                {
			if (buf[i] > ' ')
				real_len++;

                        if (buf[i] == 0x3D)
                                padCount++;
                }

		// Hmm - should this be a "bad format MIME" exception instead?
		if (real_len%4 != 0)
			throw new IllegalArgumentException("Length not a multiple of 4");

                int ret_len = (real_len/4)*3 - padCount;
		byte ret[] = new byte[ret_len];

		i = 0;
		byte[] t = new byte[4];
		int output_index = 0;
		int j = 0;
		t[0] = t[1] = t[2] = t[3] = 61;	// Ascii =
		while (i < len)
		{
			byte c =  buf[i++];
			if (c > ' ')
				t[j++] = c;

			if (j == 4)
			{
				output_index += decode64(ret, output_index, t[0], t[1], t[2], t[3]);
				j = 0;
				t[0] = t[1] = t[2] = t[3] = 61;	// Ascii =
			}
		}
		if (j > 0)
			decode64(ret, output_index, t[0], t[1], t[2], t[3]);

		return ret;
	}

    /**
     * Base 64 Decodes a String
     * Ignores trailing whitespace and newlines
     *
     * @param msg String to be decoded
     *
     * @return the decoded value in byte[] form
     */

    public static byte[] decode64( String msg ) 
    {
      return decode64( msg.getBytes());
    }

	// Returns the number of bytes converted
        private static int
	decode64( byte ret[], int ret_off, byte a, byte b, byte c, byte d )
	{
		byte da = dec_table[a];
		byte db = dec_table[b];
		byte dc = dec_table[c];
		byte dd = dec_table[d];

		if (da == -1 || db == -1 || (dc == -1 && c != 0x3d) || (dd == -1 && d != 0x3d))
			throw new IllegalArgumentException("Invalid character ["+a+", "+b+", "+c+", "+d+"]");

		ret[ret_off++] = (byte)(da << 2 | db >>> 4);
		if (c == 0x3d)	// Ascii =
			return 1;
		ret[ret_off++] = (byte)(db << 4 | dc >>> 2);
		if (d == 0x3d)	// Ascii =
			return 2;
		ret[ret_off++] = (byte)(dc << 6 | dd);
                        return 3;
	}


}