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

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

package com.nimbusds.jose.util;


import java.util.Arrays;


/**
 * Base 64 and base 64 URL-safe encoder and decoder.
 *
 * 

Based on Mikael Grev's MiG base 64 encoder / decoder, with modifications * to support URL-safe encoding and decoding. * *

Original licence: * *

 * Licence (BSD):
 *
 * Copyright (c) 2004, Mikael Grev, MiG InfoCom AB. (base64 @ miginfocom . com)
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 * Redistributions of source code must retain the above copyright notice, this
 * list of conditions and the following disclaimer. Redistributions in binary
 * form must reproduce the above copyright notice, this list of conditions and
 * the following disclaimer in the documentation and/or other materials provided
 * with the distribution. Neither the name of the MiG InfoCom AB nor the names
 * of its contributors may be used to endorse or promote products derived from
 * this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGE.
 * 
* * @author Mikael Grev * @author Jaap Beetstra * @author Vladimir Dzhuvinov */ final class Base64Codec { /** * The base 64 characters. */ private static final char[] CA = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/".toCharArray(); /** * The base 64 URL-safe characters. */ private static final char[] CA_URL_SAFE = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-_".toCharArray(); /** * Maps base 64 characters to their respective byte values. */ private static final int[] IA = new int[256]; /** * Maps base 64 URL-safe characters to their respective byte values. */ private static final int[] IA_URL_SAFE = new int[256]; static { // Regular map Arrays.fill(IA, -1); for (int i = 0, iS = CA.length; i < iS; i++) { IA[CA[i]] = i; } IA['='] = 0; // URL-safe map Arrays.fill(IA_URL_SAFE, -1); for (int i = 0, iS = CA_URL_SAFE.length; i < iS; i++) { IA_URL_SAFE[CA_URL_SAFE[i]] = i; } IA_URL_SAFE['='] = 0; } /** * 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. */ public 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; } } /** * Normalises a base 64 encoded string by ensuring any URL-safe * characters are replaced with their regular base64 representation and * any truncated '=' padding is restored. * * @param b64String The base 64 or base 64 URL-safe encoded string. * Must not be {@code null}. * * @return The normalised base 64 encoded string. */ public static String normalizeEncodedString(final String b64String) { final int inputLen = b64String.length(); // Compute missing padding, taking illegal chars into account final int legalLen = inputLen - countIllegalChars(b64String); final int padLength = legalLen % 4 == 0 ? 0 : 4 - (legalLen % 4); // Create output array char[] chars = new char[inputLen + padLength]; // Copy chars into output array b64String.getChars(0, inputLen, chars, 0); // Append padding chars if required for (int i = 0; i < padLength; i++) { chars[inputLen + i] = '='; } // Replace URL-safe chars for (int i = 0; i < inputLen; i++) { if (chars[i] == '_') { chars[i] = '/'; } else if (chars[i] == '-') { chars[i] = '+'; } } return new String(chars); } /** * Counts the illegal / separator characters in the specified base 64 * or base 64 URL-safe encoded string. * * @param b64String The base 64 or base 64 URL-safe encoded string. * Must not be {@code null}. * * @return The illegal character count, zero if none. */ public static int countIllegalChars(final String b64String) { // Number of separator and illegal characters int illegalCharCount = 0; for (int i = 0; i < b64String.length(); i++) { final char c = b64String.charAt(i); if (IA[c] == -1 && IA_URL_SAFE[c] == -1) { illegalCharCount++; } } return illegalCharCount; } /** * Encodes a byte array into a base 64 encoded character array. * * @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 character array. Never {@code null}. */ public static char[] encodeToChar(final byte[] byteArray, final boolean urlSafe) { // Check special case int sLen = byteArray != null ? byteArray.length : 0; if (sLen == 0) { return new char[0]; } int eLen = (sLen / 3) * 3; // Length of even 24-bits. int dLen = computeEncodedLength(sLen, urlSafe); // Returned character count char[] out = new char[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 int i = (byteArray[s++] & 0xff) << 16 | (byteArray[s++] & 0xff) << 8 | (byteArray[s++] & 0xff); // Encode the int into four chars if (urlSafe) { out[d++] = CA_URL_SAFE[(i >>> 18) & 0x3f]; out[d++] = CA_URL_SAFE[(i >>> 12) & 0x3f]; out[d++] = CA_URL_SAFE[(i >>> 6) & 0x3f]; out[d++] = CA_URL_SAFE[i & 0x3f]; } else { out[d++] = CA[(i >>> 18) & 0x3f]; out[d++] = CA[(i >>> 12) & 0x3f]; out[d++] = CA[(i >>> 6) & 0x3f]; out[d++] = CA[i & 0x3f]; } } // Pad and encode last bits if source isn't even 24 bits // according to URL-safe switch int left = sLen - eLen; // 0 - 2. if (left > 0) { // Prepare the int 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] = CA_URL_SAFE[i >> 12]; out[dLen - 2] = CA_URL_SAFE[(i >>> 6) & 0x3f]; out[dLen - 1] = CA_URL_SAFE[i & 0x3f]; } else { out[dLen - 2] = CA_URL_SAFE[i >> 12]; out[dLen - 1] = CA_URL_SAFE[(i >>> 6) & 0x3f]; } } else { // Original Mig code with padding out[dLen - 4] = CA[i >> 12]; out[dLen - 3] = CA[(i >>> 6) & 0x3f]; out[dLen - 2] = left == 2 ? CA[i & 0x3f] : '='; out[dLen - 1] = '='; } } return out; } /** * 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) { // Reuse char[] since we can't create a String incrementally // and StringBuffer/Builder would be slower return new String(encodeToChar(byteArray, urlSafe)); } /** * 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 String nStr = normalizeEncodedString(b64String); final int sLen = nStr.length(); // Count illegal characters (including '\r', '\n') to determine // the size of the byte array to return final int sepCnt = countIllegalChars(nStr); // Ensure the legal chars (including '=' padding) divide by 4 (RFC 2045) if ((sLen - sepCnt) % 4 != 0) { // The string is corrupted return new byte[0]; } // Count '=' at end int pad = 0; for (int i = sLen; i > 1 && IA[nStr.charAt(--i)] <= 0; ) { if (nStr.charAt(i) == '=') { pad++; } } int len = ((sLen - sepCnt) * 6 >> 3) - pad; // Preallocate byte[] of final length byte[] dArr = new byte[len]; for (int s = 0, d = 0; d < len; ) { // Assemble three bytes into an int from four base 64 // characters int i = 0; for (int j = 0; j < 4; j++) { // j only increased if a valid char was found int c = IA[nStr.charAt(s++)]; if (c >= 0) { i |= c << (18 - j * 6); } else { j--; } } // Add the bytes dArr[d++] = (byte) (i >> 16); if (d < len) { dArr[d++] = (byte) (i >> 8); if (d < len) { dArr[d++] = (byte) i; } } } return dArr; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy