Base64: java.lang.IllegalArgumentException: Недопустимый символ
Я пытаюсь отправить электронное письмо с подтверждением после регистрации пользователя. Я использую библиотеку JavaMail для этой цели и класс Java 8 Base64.
Я кодирую электронные письма пользователей следующим образом:
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);
где confirmationURL
:
private final static String confirmationURL = "http://localhost:8080/project/controller?command=confirmRegistration&ID=";
И затем декодируем это в 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;
}
И когда я пытаюсь декодировать:
http://localhost:8080/project/controller?command=confirmRegistration&ID=[[email protected]
Я получаю java.lang.IllegalArgumentException: Illegal base64 character 5b
.
Я попытался использовать базовый кодировщик/декодер (а не URL-адреса) без успеха.
РЕШИТЬ:
Проблема следующая: в строке:
String confirmLink = "Complete your registration by clicking on following"+ "\n<a href='" + confirmationURL + encodedEmail + "'>link</a>";
Я вызываю toString в массиве байтов, поэтому я должен сделать следующее:
String encodedEmail = new String(Base64.getEncoder().encode(
user.getEmail().getBytes(StandardCharsets.UTF_8)));
Благодаря Jon Skeet и ByteHamster.
Ответы
Ответ 1
Ваш зашифрованный текст [[email protected]
. Это не Base64, что-то пошло не так при кодировании. Этот код декодирования выглядит хорошо.
Используйте этот код для преобразования байта [] в строку перед добавлением его в 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
Я столкнулся с этой ошибкой, так как мое кодированное изображение начиналось с data:image/png;base64,iVBORw0...
.
Этот ответ привел меня к решению:
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
Просто используйте приведенный ниже код, чтобы решить эту проблему:
JsonObject obj = Json.createReader(new ByteArrayInputStream(Base64.getDecoder().decode(accessToken.split("\\.")[1].
replace('-', '+').replace('_', '/')))).readObject();
В приведенном выше коде replace('-', '+').replace('_', '/')
выполнялась работа. Подробнее см. 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
Метод Base64.Encoder.encodeToString автоматически использует набор символов ISO-8859-1.
Для утилиты шифрования, которую я пишу, я взял строку ввода шифрованного текста, а Base64 закодировал ее для передачи, а затем отменил процесс. Соответствующие части показаны ниже. ПРИМЕЧАНИЕ. Свойство file.encoding имеет значение ISO-8859-1 при вызове JVM, так что может также иметь подшипник.
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();
}
}
Результат выглядит следующим образом:
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...
Ответ 5
Я получил эту же ошибку, и проблема была в том, что строка начиналась с data:image/png;base64,...
Решение было:
byte[] imgBytes = Base64.getMimeDecoder().decode(imgBase64.split(",")