loadImage() with base64 encoded jpg

I am writing a program that gets an image encoded in base64 through the Serial port. I want to then load that image in my Processing program to analyze it.

It seems that loadImage() should work with base64 encoded data, provided you use the right header (reference), but even the gist used in the pull request throws a NullPointerException.

I have checked out the discussions i will post on a comment (this forum only lets new users post 2 links per post, for some reason) but nothing seems to work. I get either NullPointerException or IllegalArgumentException. Any help would be appreciated!

Here’s my code:

import processing.serial.*;

import java.awt.Image;
import java.awt.Toolkit;
import java.util.Base64;
import java.io.*;
import java.awt.image.BufferedImage;
import javax.imageio.ImageIO;

processing.serial.Serial myPort;
String val;
PImage feed;
PrintWriter debug;

void setup(){
  size(800, 800);
  //print(Serial.list());
  String portName = "COM3";
  
  debug = createWriter("debug.txt");
  
  myPort = new processing.serial.Serial(this, portName, 2000000);

}

void draw(){
  background(0);
  if(myPort.available() > 0){
    val = myPort.readStringUntil('\n');
    if(val != null){
      println("_______________________");
      debug.println(val.substring(873)); //Get rid of a bunch of characters prior to the base64 image
      val = val.substring(val.indexOf("/"));
      println(val.length());

      if(val.startsWith("/")){
        
        //This throws NullPointerException on image(feed, 0, 0)
          //String data = "data:image/jpg;base64,"+val;
          //feed = loadImage(data);
        
        try {
          feed = DecodePImageFromBase64(val);
        } 
        catch (IOException e) {
          println(e);
        }
        
        image(feed, 0, 0);
      }
    }
  }
}

public PImage DecodePImageFromBase64(String i_Image64) throws IOException
{
  PImage result = null;
  byte[] decodedBytes = Base64.getDecoder().decode(i_Image64);
  println("Base64");

  ByteArrayInputStream in = new ByteArrayInputStream(decodedBytes);
  BufferedImage bImageFromConvert = ImageIO.read(in);
  BufferedImage convertedImg = new BufferedImage(bImageFromConvert.getWidth(), bImageFromConvert.getHeight(), BufferedImage.TYPE_INT_ARGB);
  convertedImg.getGraphics().drawImage(bImageFromConvert, 0, 0, null);
  result = new PImage(convertedImg);

  return result;
}

void keyPressed() {
  debug.flush(); // Writes the remaining data to the file
  debug.close(); // Finishes the file
  exit(); // Stops the program
}

I know I am getting the right serial data because I get it printed to a .txt file, and converting it with this tool gives me a valid image.

Using the code above, I get one of two errors:

  1. IllegalArgumentException: Input byte array has incorrect ending byte at 145536
  2. IllegalArgumentException: Input byte array has wrong 4-byte ending unit

I will also post an example of an encoded image.

1 Like

Here are the discussions I checked out (1/3)
this discussion, this discussion

Hello @papertree,

Welcome to the forum.

Assuming the system silenced you automatically for enthusiasm and you will be back soon!

I simplified your code without the serial and this works on my end:

import java.awt.Image;
import java.awt.Toolkit;
import java.util.Base64;
import java.io.*;
import java.awt.image.BufferedImage;
import javax.imageio.ImageIO;

PImage feed;

String val = "";

void setup() 
  {
  size(200, 200);
  noLoop();
  }

void draw() 
  {
  background(0);

  // Remove the "data:image/jpeg;base64," prefix (if exists)
  if (val.startsWith("data:image/jpeg;base64,"))
    {
    val = val.substring("data:image/jpeg;base64,".length());
    //println(base64Str);
    //println();
    }

  try 
    {
    feed = DecodePImageFromBase64(val);
    }
  catch (IOException e) 
    {
    println(e);
    }

  image(feed, 0, 0);
  }

public PImage DecodePImageFromBase64(String i_Image64) throws IOException
  {
  PImage result = null;
  byte[] decodedBytes = Base64.getDecoder().decode(i_Image64);
  println("Base64");

  ByteArrayInputStream in = new ByteArrayInputStream(decodedBytes);
  BufferedImage bImageFromConvert = ImageIO.read(in);
  BufferedImage convertedImg = new BufferedImage(bImageFromConvert.getWidth(), bImageFromConvert.getHeight(), BufferedImage.TYPE_INT_ARGB);
  convertedImg.getGraphics().drawImage(bImageFromConvert, 0, 0, null);
  result = new PImage(convertedImg); //Depracated but works

  return result;
  }

test

I used this to create the encoded string:

:)

2 Likes

Hi,
Since the Forum wouldn’t give me access again, and I am really hitting a wall with this, I had to create another account..

Your sketch works for me too, yay! I since discovered that, since I changed the code that took and encoded the images, the base64 strings I was receiving through the Serial port were missing some characters, which meant that they weren’t valid files anyway.

However, I am now getting 100% correct strings, and the encoder still doesn’t work. Here’s an example of a base64 encoded jpeg file, and here’s my current sketch:

import processing.serial.*;
import java.awt.Image;
import java.awt.Toolkit;
import java.util.Base64;
import java.io.*;
import java.awt.image.BufferedImage;
import javax.imageio.ImageIO;

processing.serial.Serial myPort;

PImage feed = null;

String val = "";

void setup() 
  {
  size(320, 240);
  String portName = "COM3";
  myPort = new processing.serial.Serial(this, portName, 115200);

  
  //noLoop();
  }

void draw() 
  {
  background(0);
  if(myPort.available() > 0){
  
    val = myPort.readStringUntil('\n');
    if(val != null){
      println(val); //THIS prints a correct base64-encoded jpeg string
      
      // Remove the "data:image/jpeg;base64," prefix (if exists)
      if (val.startsWith("data:image/jpeg;base64,"))
        {
        val = val.substring("data:image/jpeg;base64,".length());
        }
    
      try 
        {
        feed = DecodePImageFromBase64(val);
        }
      catch (IOException e) 
        {
        println(e);
        }
    
      image(feed, 0, 0);
    }
  }
}

public PImage DecodePImageFromBase64(String i_Image64) throws IOException{
  PImage result = null;
  println(i_Image64);//THIS prints a correct base64-encoded jpeg string

  byte[] decodedBytes = Base64.getDecoder().decode(i_Image64); //IllegalArgumentException: Input byte array has incorrect ending byte at 6372. String is correct JPG
  println(decodedBytes.length);

  ByteArrayInputStream in = new ByteArrayInputStream(decodedBytes);
  BufferedImage bImageFromConvert = ImageIO.read(in);
  BufferedImage convertedImg = new BufferedImage(bImageFromConvert.getWidth(), bImageFromConvert.getHeight(), BufferedImage.TYPE_INT_ARGB);
  convertedImg.getGraphics().drawImage(bImageFromConvert, 0, 0, null);
  result = new PImage(convertedImg); //Depracated but works

  return result;
}

As you can see, I just edited your code to add the Serial reading. The println(val) before sending it to the DecodePImageFromBase64() function prints a valid base64-encoded jpeg string, and the println(i_Image64) prints the exact same string, but Base64.getDecoder().decode(i_image64) throws Input array has incorrect ending byte OR Illegal base64 character d (always d, the character never changes).

Could this be a problem of the decoder? Should I look into using another library?

Hi @papertree / @papertree2,

It looks like your account was mistakenly flagged by Discourse’s automatic spam detection. I have removed the flag on your original account so you should be able to log back in. Sorry for the inconvenience.

1 Like

Hello @papertree,

If this sends the terminating character it needs to be trimmed on receiving end:
https://docs.arduino.cc/language-reference/en/functions/communication/serial/println/

Reference:
trim() / Reference / Processing.org

Example:

Try while (myPort.available() > 0) in your loop:
readStringUntil() / Libraries / Processing.org
Although… it may be blocking for a large frame of data ending in a \n.

This uses virtual COM ports on PC and works:

import processing.serial.*;

import java.awt.Image;
import java.awt.Toolkit;
import java.util.Base64;
import java.io.*;
import java.awt.image.BufferedImage;
import javax.imageio.ImageIO;

processing.serial.Serial portTx;
processing.serial.Serial portRx;

String imgTxt = "";
String val = "";

PImage feed = null;

void setup()
  {
  size(200, 200);
  println(imgTxt.length());
  //imgTxt = imgTxt + "\n";

  printArray(processing.serial.Serial.list());

  String comRx = "COM4";
  portRx = new processing.serial.Serial(this, comRx, 115200);
  portRx.clear();

  String comTx = "COM5";
  portTx = new processing.serial.Serial(this, comTx, 115200); // Fixed variable name

  //portRx.readStringUntil('\n');

  // Think of this as Arduino
  thread("txtImage");   // start thread to generate data;
  }

void draw()
  {
  //background(0);
  while (portRx.available() > 0)
    {
    //println(portRx.available());
    val = portRx.readStringUntil('\n');

    if (val != null)
      {
      val = val.trim();
      println("test");
      println(val); //THIS prints a correct base64-encoded jpeg string

      // Remove the "data:image/jpeg;base64," prefix (if exists)
      if (val.startsWith("data:image/jpeg;base64,"))
        {
        println("here", val.length());
        val = val.substring("data:image/jpeg;base64,".length());
        }
      try
        {
        feed = DecodePImageFromBase64(val);
        println("ok");
        }
      catch (IOException e)
        {
        println(e);
        }
      image(feed, 0, 0);  
      }
    }
  }

public PImage DecodePImageFromBase64(String i_Image64) throws IOException   {
  PImage result = null;
  println(i_Image64);//THIS prints a correct base64-encoded jpeg string

  byte[] decodedBytes = Base64.getDecoder().decode(i_Image64); //IllegalArgumentException: Input byte array has incorrect ending byte at 6372. String is correct JPG
  println(decodedBytes.length);

  ByteArrayInputStream in = new ByteArrayInputStream(decodedBytes);
  BufferedImage bImageFromConvert = ImageIO.read(in);
  BufferedImage convertedImg = new BufferedImage(bImageFromConvert.getWidth(), bImageFromConvert.getHeight(), BufferedImage.TYPE_INT_ARGB);
  convertedImg.getGraphics().drawImage(bImageFromConvert, 0, 0, null);
  result = new PImage(convertedImg); //Depracated but works

  return result;
  }

// Think of this as the Arduino
void txtImage()   {
  println("Thread 1");
  println(imgTxt.length());
  //while (true)
    {
    if (portTx != null)   {
      println("Thread 2");

      // Loop through every character in the Base64 string
      for (int i = 0; i < imgTxt.length(); i++)
        {
        portTx.write(imgTxt.charAt(i));
        //println(i, imgTxt.charAt(i));
        //delay(1); // 1ms delay per character to stabilize transmission
        }

      // Send the terminal newline so readStringUntil('\n') triggers
      portTx.write('\n');
      println("Finished sending full string.");
      }
    }
  }
``'
You will not be able to run above until you have virtual COM ports available.
The code works and displays the image.

Reference:
https://en.wikipedia.org/wiki/Virtual_COM_port

``:)``

@sableraph Thanks! Really appreciate it.

Hi @glv,

I had no idea the Arduino serial.println() method printed a \r at the end of the data.

Thanks for your trim() suggestion inside or Processing, everything works as expected now! I haven’t encountered the Illegal base64 character d again, either, so that was the issue there also.

Thanks a lot again!

2 Likes

You are very welcome!

:)

2 Likes