Отправка образа base64 через http запрос в Java пропускает некоторые символы - PullRequest
1 голос
/ 28 апреля 2020

Я пытался отправить изображение base64 с помощью API java в NodeJS, после работы и поиска в течение нескольких часов я не мог знать, что может вызвать следующую проблему, такую ​​как:

После регистрации изображения base64 в nodejs я вижу все + символов, замененные на space

Вот часть исходного base64 в Java

f8A0NH2qH+/+hooouAfaof7/wCho+1Q/

и вот часть полученного изображения в NodeJS

f8A0NH2qH / hooouAfaof7/wCho 1Q/

Я пытался отправить изображение через POSTMAN и никаких проблем.

Все шаги, как указано ниже:

1- Я конвертирую изображение в base64, используя следующий фрагмент

public static String imgToBase64String(final RenderedImage img, final String formatName) {
        final ByteArrayOutputStream os = new ByteArrayOutputStream();
        try {
            ImageIO.write(img, formatName, Base64.getEncoder().wrap(os));
            return os.toString(StandardCharsets.ISO_8859_1.name());
        } catch (final IOException ioe) {
            throw new UncheckedIOException(ioe);
        }
    }

    public static BufferedImage base64StringToImg(final String base64String) {
        try {
            return ImageIO.read(new ByteArrayInputStream(Base64.getDecoder().decode(base64String)));
        } catch (final IOException ioe) {
            throw new UncheckedIOException(ioe);
        }
    }

И сделать снимок экрана

    final Robot robot = new Robot();
    final Rectangle r = new Rectangle(Toolkit.getDefaultToolkit().getScreenSize());
    final BufferedImage bi = robot.createScreenCapture(r);
    final String base64String = Base64Converter.imgToBase64String(bi, "jpg");

2 - я использую библиотеку Gson для строкового преобразования объекта

3 - я использую bodyParser в NodeJS

4- Отправка HTTP запрос как:

public static void sendPOST(String image) throws Exception {
        String POST_PARAMS = "screenShotData";
        URL obj = new URL(POST_URL);
        HttpURLConnection con = (HttpURLConnection) obj.openConnection();
        con.setRequestMethod("POST");
        con.setConnectTimeout(5000); // 5 seconds
        con.setReadTimeout(5000); // 5 seconds

        Gson gson = new Gson();
        Http.ScreenShot screenShot = new ScreenShot(); // This is just a class with a string property
        screenShot.setImage(image);
        POST_PARAMS += gson.toJsonTree(screenShot).getAsJsonObject();


        con.setDoOutput(true);
        OutputStream os = con.getOutputStream();
        byte[] outputBytesArray = POST_PARAMS.getBytes();
        os.write(outputBytesArray);
        os.flush();
        os.close();


        int responseCode = con.getResponseCode();
        System.out.println("POST Response Code :: " + responseCode);

        if (responseCode == HttpURLConnection.HTTP_OK) { //success
            BufferedReader in = new BufferedReader(new InputStreamReader(
                    con.getInputStream()));
            String inputLine;
            StringBuffer response = new StringBuffer();

            while ((inputLine = in.readLine()) != null) {
                response.append(inputLine);
            }
            in.close();

            Object responseObject = gson.fromJson(response.toString(), Object.class);
            System.out.println("Res: " + responseObject);
        } else {
            System.out.println(con.getResponseMessage());
        }
    }

Ответы [ 3 ]

1 голос
/ 28 апреля 2020

Вы забыли закрыть упакованный поток кодировщика Base64. Только если вы закроете его, конец кодированных данных base64 может быть записан:

public static String imgToBase64String(final RenderedImage img, final String formatName) {
    final ByteArrayOutputStream os = new ByteArrayOutputStream();
    try {
        try (OutputStream wrapped = Base64.getEncoder().wrap(os)) {
            ImageIO.write(img, formatName, wrapped);
        }
        return os.toString(StandardCharsets.ISO_8859_1.name());
    } catch (final IOException ioe) {
        throw new UncheckedIOException(ioe);
    }
}
1 голос
/ 28 апреля 2020

Я сталкивался с этой проблемой, но у меня не было полномочий меняться в классе или что-то еще Я просто заменил пробел на +, в большинстве случаев он работал.

1 голос
/ 28 апреля 2020

В URL-кодированном тексте символ + означает пробел. Например,

https://example.com/?s=nodejs+bodyparser

отправляет параметр s со значением

nodejs bodyparser 

(обратите внимание на пробел).

Когда вы делаете обычную форму сообщения ( добрые браузеры) вы используете тип данных application/x-www-form-urlencoded, то есть полезная нагрузка вашей операции POST выглядит как строка запроса. Я думаю, что вы передаете объект JSON в виде текстовой строки без url-кодирования.

Возможно, вы вместо этого захотите использовать тип данных application/json. Анализатор тела nodejs обнаруживает из заголовка Content-type, что это JSON, и анализирует его правильно.

Попробуйте это. (не отлажено, извините.)

    string payload = gson.toJsonTree(screenShot).getAsJsonObject();
    byte[] outputBytesArray = payload.getBytes();

    con.setRequestProperty("Content-Type", "application/json");
    con.setDoOutput(true);
    OutputStream os = con.getOutputStream();
    os.write(outputBytesArray);
    os.flush();
    os.close();
...