Base64:. lang。IllegalArgumentException:非法字符

时间:2021-04-10 18:09:59

I'm trying to send a confirmation email after user registration. I'm using the JavaMail library for this purpose and the Java 8 Base64 util class.

我想在用户注册后发送确认邮件。为此,我使用了JavaMail库和Java 8 Base64 util类。

I'm encoding user emails in the following way:

我用以下方式编码用户电子邮件:

byte[] encodedEmail = Base64.getUrlEncoder().encode(user.getEmail().getBytes(StandardCharsets.UTF_8));
Multipart multipart = new MimeMultipart();
InternetHeaders headers = new InternetHeaders();
headers.addHeader("Content-type", "text/html; charset=UTF-8");
String confirmLink = "Complete your registration by clicking on following"+ "\n<a href='" + confirmationURL + encodedEmail + "'>link</a>";
MimeBodyPart link = new MimeBodyPart(headers,
confirmLink.getBytes("UTF-8"));
multipart.addBodyPart(link);

where confirmationURL is:

confirmationURL在哪里:

private final static String confirmationURL = "http://localhost:8080/project/controller?command=confirmRegistration&ID=";

And then decoding this in ConfirmRegistrationCommand in such way:

然后在ConfirmRegistrationCommand中解码如下:

    String encryptedEmail = request.getParameter("ID");

    String decodedEmail = new String(Base64.getUrlDecoder().decode(encryptedEmail), StandardCharsets.UTF_8);

    RepositoryFactory repositoryFactory = RepositoryFactory
            .getFactoryByName(FactoryType.MYSQL_REPOSITORY_FACTORY);
    UserRepository userRepository = repositoryFactory.getUserRepository();
    User user = userRepository.find(decodedEmail);

    if (user.getEmail().equals(decodedEmail)) {
        user.setActiveStatus(true);
        return Path.WELCOME_PAGE;
    } else {
        return Path.ERROR_PAGE;
    }

And when I'm trying to decode:

当我试图解码时

http://localhost:8080/project/controller?command=confirmRegistration&ID=[B@6499375d

I'm getting java.lang.IllegalArgumentException: Illegal base64 character 5b.

我得到. lang。非法争论例外:非法的基64字符5b。

I tried to use basic Encode/Decoder (not URL ones) with no success.

我尝试使用基本的编码/解码器(不是URL),但没有成功。

SOLVED:

解决:

The problem was the next - in the line:

问题是下一个——排队的人:

 String confirmLink = "Complete your registration by clicking on following"+ "\n<a href='" + confirmationURL + encodedEmail + "'>link</a>";

I'm calling toString on an array of bytes, so I should do the following:

我在一个字节数组中调用toString,所以我应该执行以下操作:

String encodedEmail = new String(Base64.getEncoder().encode(
                user.getEmail().getBytes(StandardCharsets.UTF_8)));

Thanks to Jon Skeet and ByteHamster.

感谢Jon Skeet和ByteHamster。

4 个解决方案

#1


10  

Your encoded text is [B@6499375d. That is not Base64, something went wrong while encoding. That decryption code looks good.

您的编码文本是[B@6499375d]。这不是Base64,编码时出了问题。解密代码看起来不错。

Use this code to convert the byte[] to a String before adding it to the URL:

使用此代码将字节[]转换为字符串,然后将其添加到URL:

String encodedEmailString = new String(encodedEmail, "UTF-8");
// ...
String confirmLink = "Complete your registration by clicking on following"
    + "\n<a href='" + confirmationURL + encodedEmailString + "'>link</a>";

#2


5  

I encountered this error since my encoded image started with data:image/png;base64,iVBORw0....

我遇到了这个错误,因为我的编码图像开始数据:图像/ png;base64,iVBORw0 ....

This answer led me to the solution:

这个答案让我想到了答案:

String partSeparator = ",";
if (data.contains(partSeparator) {
  String encodedImg = data.split(partSeparator)[1];
  byte[] decodedImg = Base64.getDecoder().decode(encodedImg.getBytes(StandardCharsets.UTF_8));
  Path destinationFile = Paths.get("/path/to/imageDir", "myImage.jpg");
  Files.write(destinationFile, decodedImg);
}

#3


2  

Just use the below code to resolve this:

使用下面的代码来解决这个问题:

JsonObject obj = Json.createReader(new ByteArrayInputStream(Base64.getDecoder().decode(accessToken.split("\\.")[1].
                        replace('-', '+').replace('_', '/')))).readObject();

In the above code replace('-', '+').replace('_', '/') did the job. For more details see the https://jwt.io/js/jwt.js. I understood the problem from the part of the code got from that link:

在上面的代码中替换('-','+')。替换('_','/')完成了工作。有关详细信息,请参见https://jwt.io/js/jwt.js。我从那个链接得到的代码中理解了这个问题:

function url_base64_decode(str) {
  var output = str.replace(/-/g, '+').replace(/_/g, '/');
  switch (output.length % 4) {
    case 0:
      break;
    case 2:
      output += '==';
      break;
    case 3:
      output += '=';
      break;
    default:
      throw 'Illegal base64url string!';
  }
  var result = window.atob(output); //polifyll https://github.com/davidchambers/Base64.js
  try{
    return decodeURIComponent(escape(result));
  } catch (err) {
    return result;
  }
}

#4


0  

The Base64.Encoder.encodeToString method automatically uses the ISO-8859-1 character set.

Base64.Encoder。encodeToString方法自动使用ISO-8859-1字符集。

For an encryption utility I am writing, I took the input string of cipher text and Base64 encoded it for transmission, then reversed the process. Relevant parts shown below. NOTE: My file.encoding property is set to ISO-8859-1 upon invocation of the JVM so that may also have a bearing.

对于我正在编写的加密实用程序,我取密码文本的输入字符串,并用Base64对其进行编码以进行传输,然后反转这个过程。相关部分所示。注意:我的文件。在调用JVM时,将编码属性设置为ISO-8859-1,这样也可以具有承载。

static String getBase64EncodedCipherText(String cipherText) {
    byte[] cText = cipherText.getBytes();
    // return an ISO-8859-1 encoded String
    return Base64.getEncoder().encodeToString(cText);
}

static String getBase64DecodedCipherText(String encodedCipherText) throws IOException {
    return new String((Base64.getDecoder().decode(encodedCipherText)));
}

public static void main(String[] args) {
    try {
        String cText = getRawCipherText(null, "Hello World of Encryption...");

        System.out.println("Text to encrypt/encode: Hello World of Encryption...");
        // This output is a simple sanity check to display that the text
        // has indeed been converted to a cipher text which 
        // is unreadable by all but the most intelligent of programmers.
        // It is absolutely inhuman of me to do such a thing, but I am a
        // rebel and cannot be trusted in any way.  Please look away.
        System.out.println("RAW CIPHER TEXT: " + cText);
        cText = getBase64EncodedCipherText(cText);
        System.out.println("BASE64 ENCODED: " + cText);
        // There he goes again!!
        System.out.println("BASE64 DECODED:  " + getBase64DecodedCipherText(cText));
        System.out.println("DECODED CIPHER TEXT: " + decodeRawCipherText(null, getBase64DecodedCipherText(cText)));
    } catch (Exception e) {
        e.printStackTrace();
    }

}

The output looks like:

输出:

Text to encrypt/encode: Hello World of Encryption...
RAW CIPHER TEXT: q$;�C�l��<8��U���X[7l
BASE64 ENCODED: HnEPJDuhQ+qDbInUCzw4gx0VDqtVwef+WFs3bA==
BASE64 DECODED:  q$;�C�l��<8��U���X[7l``
DECODED CIPHER TEXT: Hello World of Encryption...

#1


10  

Your encoded text is [B@6499375d. That is not Base64, something went wrong while encoding. That decryption code looks good.

您的编码文本是[B@6499375d]。这不是Base64,编码时出了问题。解密代码看起来不错。

Use this code to convert the byte[] to a String before adding it to the URL:

使用此代码将字节[]转换为字符串,然后将其添加到URL:

String encodedEmailString = new String(encodedEmail, "UTF-8");
// ...
String confirmLink = "Complete your registration by clicking on following"
    + "\n<a href='" + confirmationURL + encodedEmailString + "'>link</a>";

#2


5  

I encountered this error since my encoded image started with data:image/png;base64,iVBORw0....

我遇到了这个错误,因为我的编码图像开始数据:图像/ png;base64,iVBORw0 ....

This answer led me to the solution:

这个答案让我想到了答案:

String partSeparator = ",";
if (data.contains(partSeparator) {
  String encodedImg = data.split(partSeparator)[1];
  byte[] decodedImg = Base64.getDecoder().decode(encodedImg.getBytes(StandardCharsets.UTF_8));
  Path destinationFile = Paths.get("/path/to/imageDir", "myImage.jpg");
  Files.write(destinationFile, decodedImg);
}

#3


2  

Just use the below code to resolve this:

使用下面的代码来解决这个问题:

JsonObject obj = Json.createReader(new ByteArrayInputStream(Base64.getDecoder().decode(accessToken.split("\\.")[1].
                        replace('-', '+').replace('_', '/')))).readObject();

In the above code replace('-', '+').replace('_', '/') did the job. For more details see the https://jwt.io/js/jwt.js. I understood the problem from the part of the code got from that link:

在上面的代码中替换('-','+')。替换('_','/')完成了工作。有关详细信息,请参见https://jwt.io/js/jwt.js。我从那个链接得到的代码中理解了这个问题:

function url_base64_decode(str) {
  var output = str.replace(/-/g, '+').replace(/_/g, '/');
  switch (output.length % 4) {
    case 0:
      break;
    case 2:
      output += '==';
      break;
    case 3:
      output += '=';
      break;
    default:
      throw 'Illegal base64url string!';
  }
  var result = window.atob(output); //polifyll https://github.com/davidchambers/Base64.js
  try{
    return decodeURIComponent(escape(result));
  } catch (err) {
    return result;
  }
}

#4


0  

The Base64.Encoder.encodeToString method automatically uses the ISO-8859-1 character set.

Base64.Encoder。encodeToString方法自动使用ISO-8859-1字符集。

For an encryption utility I am writing, I took the input string of cipher text and Base64 encoded it for transmission, then reversed the process. Relevant parts shown below. NOTE: My file.encoding property is set to ISO-8859-1 upon invocation of the JVM so that may also have a bearing.

对于我正在编写的加密实用程序,我取密码文本的输入字符串,并用Base64对其进行编码以进行传输,然后反转这个过程。相关部分所示。注意:我的文件。在调用JVM时,将编码属性设置为ISO-8859-1,这样也可以具有承载。

static String getBase64EncodedCipherText(String cipherText) {
    byte[] cText = cipherText.getBytes();
    // return an ISO-8859-1 encoded String
    return Base64.getEncoder().encodeToString(cText);
}

static String getBase64DecodedCipherText(String encodedCipherText) throws IOException {
    return new String((Base64.getDecoder().decode(encodedCipherText)));
}

public static void main(String[] args) {
    try {
        String cText = getRawCipherText(null, "Hello World of Encryption...");

        System.out.println("Text to encrypt/encode: Hello World of Encryption...");
        // This output is a simple sanity check to display that the text
        // has indeed been converted to a cipher text which 
        // is unreadable by all but the most intelligent of programmers.
        // It is absolutely inhuman of me to do such a thing, but I am a
        // rebel and cannot be trusted in any way.  Please look away.
        System.out.println("RAW CIPHER TEXT: " + cText);
        cText = getBase64EncodedCipherText(cText);
        System.out.println("BASE64 ENCODED: " + cText);
        // There he goes again!!
        System.out.println("BASE64 DECODED:  " + getBase64DecodedCipherText(cText));
        System.out.println("DECODED CIPHER TEXT: " + decodeRawCipherText(null, getBase64DecodedCipherText(cText)));
    } catch (Exception e) {
        e.printStackTrace();
    }

}

The output looks like:

输出:

Text to encrypt/encode: Hello World of Encryption...
RAW CIPHER TEXT: q$;�C�l��<8��U���X[7l
BASE64 ENCODED: HnEPJDuhQ+qDbInUCzw4gx0VDqtVwef+WFs3bA==
BASE64 DECODED:  q$;�C�l��<8��U���X[7l``
DECODED CIPHER TEXT: Hello World of Encryption...