All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.nimbusds.jose.util.Base64Codec Maven / Gradle / Ivy

/*
 * nimbus-jose-jwt
 *
 * Copyright 2012-2018, Connect2id Ltd.
 *
 * Licensed under the Apache License, Version 2.0 (the "License"); you may not use
 * this file except in compliance with the License. You may obtain a copy of the
 * License at
 *
 *    http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software distributed
 * under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
 * CONDITIONS OF ANY KIND, either express or implied. See the License for the
 * specific language governing permissions and limitations under the License.
 */

package com.nimbusds.jose.util;


import java.util.Arrays;


/**
 * @author Tim McLean
 * @author others
 * @version 2018-07-23
 */
final class Base64Codec {


	/**
	 * Computes the base 64 encoded character length for the specified
	 * input byte length.
	 *
	 * @param inputLength The input byte length.
	 * @param urlSafe     {@code true} for URL-safe encoding.
	 *
	 * @return The base 64 encoded character length.
	 */
	static int computeEncodedLength(final int inputLength, final boolean urlSafe) {

		if (inputLength == 0) {
			return 0;
		}

		if (urlSafe) {

			// Compute the number of complete quads (4-char blocks)
			int fullQuadLength = (inputLength / 3) << 2;

			// Compute the remaining bytes at the end
			int remainder = inputLength % 3;

			// Compute the total
			return remainder == 0 ? fullQuadLength : fullQuadLength + remainder + 1;
		} else {
			// Original Mig code
			return ((inputLength - 1) / 3 + 1) << 2;
		}
	}


	// *** Timing-protected (tp) utility methods ***
	// This class is used to encode/decode private keys, so we make an effort
	// to prevent side channel leaks. Here we define a number of timing leak
	// resistant utility methods. Boolean values are stored in ints in order to
	// prevent optimizations that would reintroduce timing leaks.
	//
	// Some background information on preventing side channel leaks:
	// - https://www.chosenplaintext.ca/articles/beginners-guide-constant-time-cryptography.html

	/**
	 * Select one of two values based on {@code bool} without leaking
	 * information about which one was selected.
	 *
	 * @param bool Must be 1 or 0
	 * @param when_true The value to return if {@code bool} is 1
	 * @param when_false The value to return if {@code bool} is 0
	 * @return Either when_true or when_false
	 */
	static int tpSelect(int bool, int when_true, int when_false) {

		// Will be 0x00000000 when bool == 1, or 0xFFFFFFFF when bool == 0
		final int mask = bool - 1;

		return when_true ^ (mask & (when_true ^ when_false));
	}


	/**
	 * Checks if a < b without leaking information about either a or b
	 *
	 * @param a Any int
	 * @param b Any int
	 * @return 1 if yes, 0 if no
	 */
	static int tpLT(int a, int b) {

		return (int) (((long) a - (long) b) >>> 63);
	}


	/**
	 * Checks if a > b without leaking information about either a or b
	 *
	 * @param a Any int
	 * @param b Any int
	 * @return 1 if yes, 0 if no
	 */
	static int tpGT(int a, int b) {

		return (int) (((long) b - (long) a) >>> 63);
	}


	/**
	 * Checks if a == b without leaking information about either a or b
	 *
	 * @param a Any int
	 * @param b Any int
	 * @return 1 if yes, 0 if no
	 */
	static int tpEq(int a, int b) {

		// This is magic but it will make sense
		// if you think about it for 30 minutes

		final int bit_diff = a ^ b;
		final int msb_iff_zero_diff = (bit_diff - 1) & (~bit_diff);
		return msb_iff_zero_diff >>> 63;
	}


	/**
	 * Convert a digit index to the appropriate base 64 ASCII byte.
	 *
	 * Uses '+' and '/' for 62 and 63, as required for standard base 64.
	 *
	 * @param digit_idx Must be at least 0 and at most 63. Output is undefined
	 *                  if digit_idx is not on this range.
	 * @return An ASCII character
	 */
	static byte encodeDigitBase64(int digit_idx) {

		assert digit_idx >= 0 && digit_idx <= 63;

		// Figure out which type of digit this should be
		final int is_uppercase = tpLT(digit_idx, 26);
		final int is_lowercase = tpGT(digit_idx, 25) & tpLT(digit_idx, 52);
		final int is_decimal   = tpGT(digit_idx, 51) & tpLT(digit_idx, 62);
		final int is_62        = tpEq(digit_idx, 62);
		final int is_63        = tpEq(digit_idx, 63);

		// Translate from digit index to ASCII for each hypothetical scenario
		final int as_uppercase = digit_idx -  0 + 65;
		final int as_lowercase = digit_idx - 26 + 97;
		final int as_decimal   = digit_idx - 52 + 48;
		final int as_62        = '+';
		final int as_63        = '/';

		// Zero out all scenarios except for the right one, and combine
		final int ascii =
			tpSelect(is_uppercase, as_uppercase, 0) |
			tpSelect(is_lowercase, as_lowercase, 0) |
			tpSelect(is_decimal  , as_decimal  , 0) |
			tpSelect(is_62       , as_62       , 0) |
			tpSelect(is_63       , as_63       , 0);

		return (byte) ascii;
	}


	/**
	 * Convert a digit index to the appropriate base64url ASCII byte.
	 *
	 * Uses '-' and '_' for 62 and 63, as required for the base64url encoding.
	 *
	 * @param digit_idx Must be at least 0 and at most 63. Output is undefined
	 *                  if digit_idx is not on this range.
	 * @return An ASCII character
	 */
	static byte encodeDigitBase64URL(int digit_idx) {

		assert digit_idx >= 0 && digit_idx <= 63;

		// Figure out which type of digit this should be
		final int is_uppercase = tpLT(digit_idx, 26);
		final int is_lowercase = tpGT(digit_idx, 25) & tpLT(digit_idx, 52);
		final int is_decimal   = tpGT(digit_idx, 51) & tpLT(digit_idx, 62);
		final int is_62        = tpEq(digit_idx, 62);
		final int is_63        = tpEq(digit_idx, 63);

		// Translate from digit index to ASCII for each hypothetical scenario
		final int as_uppercase = digit_idx -  0 + 65;
		final int as_lowercase = digit_idx - 26 + 97;
		final int as_decimal   = digit_idx - 52 + 48;
		final int as_62        = '-';
		final int as_63        = '_';

		// Zero out all scenarios except for the right one, and combine
		final int ascii =
			tpSelect(is_uppercase, as_uppercase, 0) |
			tpSelect(is_lowercase, as_lowercase, 0) |
			tpSelect(is_decimal  , as_decimal  , 0) |
			tpSelect(is_62       , as_62       , 0) |
			tpSelect(is_63       , as_63       , 0);

		return (byte) ascii;
	}


	/**
	 * Decode an ASCII byte to a base 64 digit index (0 to 63), or -1 if the
	 * input is not a valid base 64 digit.
	 *
	 * Supports '+' and '/' for standard base 64, but also '-' and '_' for
	 * base64url.
	 *
	 * @param ascii An ASCII character.
	 * @return A digit index i such that 0 <= i <= 63, or -1 if the input was not a digit.
	 */
	static int decodeDigit(byte ascii) {

		// Figure out which type of digit this is
		final int is_uppercase = tpGT(ascii, 64) & tpLT(ascii, 91);
		final int is_lowercase = tpGT(ascii, 96) & tpLT(ascii, 123);
		final int is_decimal   = tpGT(ascii, 47) & tpLT(ascii, 58);
		final int is_62        = tpEq(ascii, '-') | tpEq(ascii, '+');
		final int is_63        = tpEq(ascii, '_') | tpEq(ascii, '/');

		// It should be one of the five categories
		final int is_valid = is_uppercase | is_lowercase | is_decimal | is_62 | is_63;

		// Translate from ASCII to digit index for each hypothetical scenario
		final int from_uppercase = ascii - 65 +  0;
		final int from_lowercase = ascii - 97 + 26;
		final int from_decimal   = ascii - 48 + 52;
		final int from_62        = 62;
		final int from_63        = 63;

		// Zero out all scenarios except for the right one, and combine
		final int digit_idx =
			tpSelect(is_uppercase, from_uppercase, 0) |
			tpSelect(is_lowercase, from_lowercase, 0) |
			tpSelect(is_decimal  , from_decimal  , 0) |
			tpSelect(is_62       , from_62       , 0) |
			tpSelect(is_63       , from_63       , 0) |
			tpSelect(is_valid    , 0             , -1);

		assert digit_idx >= -1 && digit_idx <= 63;

		return digit_idx;
	}


	/**
	 * Encodes a byte array into a base 64 encoded string.
	 *
	 * @param byteArray The bytes to convert. If {@code null} or length 0
	 *                  an empty array will be returned.
	 * @param urlSafe   If {@code true} to apply URL-safe encoding (padding
	 *                  still included and not to spec).
	 *
	 * @return The base 64 encoded string. Never {@code null}.
	 */
	public static String encodeToString(byte[] byteArray, final boolean urlSafe) {

		// Check special case
		final int sLen = byteArray != null ? byteArray.length : 0;

		if (sLen == 0) {
			return "";
		}

		final int eLen = (sLen / 3) * 3;                      // Length of even 24-bits.
		final int dLen = computeEncodedLength(sLen, urlSafe); // Returned byte count
		final byte[] out = new byte[dLen];

		// Encode even 24-bits
		for (int s = 0, d = 0; s < eLen; ) {

			// Copy next three bytes into lower 24 bits of int, paying attention to sign
			final int i = (byteArray[s++] & 0xff) << 16 | (byteArray[s++] & 0xff) << 8 | (byteArray[s++] & 0xff);

			// Encode the int into four chars
			if (urlSafe) {
				out[d++] = encodeDigitBase64URL((i >>> 18) & 0x3f);
				out[d++] = encodeDigitBase64URL((i >>> 12) & 0x3f);
				out[d++] = encodeDigitBase64URL((i >>> 6) & 0x3f);
				out[d++] = encodeDigitBase64URL(i & 0x3f);
			} else {
				out[d++] = encodeDigitBase64((i >>> 18) & 0x3f);
				out[d++] = encodeDigitBase64((i >>> 12) & 0x3f);
				out[d++] = encodeDigitBase64((i >>> 6) & 0x3f);
				out[d++] = encodeDigitBase64(i & 0x3f);
			}
		}

		// Pad and encode last bits if source isn't even 24 bits
		// according to URL-safe switch
		final int left = sLen - eLen; // 0 - 2.
		if (left > 0) {
			// Prepare the int
			final int i = ((byteArray[eLen] & 0xff) << 10) | (left == 2 ? ((byteArray[sLen - 1] & 0xff) << 2) : 0);

			// Set last four chars
			if (urlSafe) {

				if (left == 2) {
					out[dLen - 3] = encodeDigitBase64URL(i >> 12);
					out[dLen - 2] = encodeDigitBase64URL((i >>> 6) & 0x3f);
					out[dLen - 1] = encodeDigitBase64URL(i & 0x3f);
				} else {
					out[dLen - 2] = encodeDigitBase64URL(i >> 12);
					out[dLen - 1] = encodeDigitBase64URL((i >>> 6) & 0x3f);
				}
			} else {
				// Original Mig code with padding
				out[dLen - 4] = encodeDigitBase64(i >> 12);
				out[dLen - 3] = encodeDigitBase64((i >>> 6) & 0x3f);
				out[dLen - 2] = left == 2 ? encodeDigitBase64(i & 0x3f) : (byte) '=';
				out[dLen - 1] = (byte) '=';
			}
		}

		return new String(out, StandardCharset.UTF_8);
	}


	/**
	 * Decodes a base 64 or base 64 URL-safe encoded string. May contain
	 * line separators. Any illegal characters are ignored.
	 *
	 * @param b64String The base 64 or base 64 URL-safe encoded string. May
	 *                  be empty or {@code null}.
	 *
	 * @return The decoded byte array, empty if the input base 64 encoded
	 *         string is empty, {@code null} or corrupted.
	 */
	public static byte[] decode(final String b64String) {

		// Check special case
		if (b64String == null || b64String.isEmpty()) {
			return new byte[0];
		}

		final byte[] srcBytes = b64String.getBytes(StandardCharset.UTF_8);
		final int sLen = srcBytes.length;

		// Calculate output length assuming zero bytes are padding or separators
		final int maxOutputLen = checkedCast((long)sLen * 6 >> 3);

		// Allocate output array (may be too large)
		final byte[] dstBytes = new byte[maxOutputLen];

		// Process all input bytes
		int d = 0;
		for (int s = 0; s < srcBytes.length; ) {
			// Assemble three bytes into an int from four base 64
			// characters
			int i = 0;

			int j = 0;
			while (j < 4 && s < sLen) {
				// j only increased if a valid char was found
				final int c = decodeDigit(srcBytes[s++]);
				if (c >= 0) {
					i |= c << (18 - j * 6);
					j++;
				}
			}

			// j is now the number of valid digits decoded

			// Add output bytes
			if (j >= 2) {
				dstBytes[d++] = (byte) (i >> 16);
				if (j >= 3) {
					dstBytes[d++] = (byte) (i >> 8);
					if (j >= 4) {
						dstBytes[d++] = (byte) i;
					}
				}
			}
		}

		// d is now the number of output bytes written

		// Copy dstBytes to new array of proper size
		return Arrays.copyOf(dstBytes, d);
	}

	private static int checkedCast(long value) {
		int result = (int) value;
		if (result != value) {
			throw new IllegalArgumentException(value +
					" cannot be cast to int without changing its value.");
		}
		return result;
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy