1

I'm using MEAN.io and i'm trying to upload a base64 encoded image.

Client side, AngularJS:

          // Image we're going to send it out
          var base64Image = files[i];

          var file = {
            image: base64Image,
            type: type,
            filetype: extension,
            characterId: character._id
          };

          var newFile = new MediaSendBase64(file);

          newFile.$save(function(image) {
            if ( !image ) {
              console.log('ERROR IMAGE');
            }
            else {
              console.log('SUCCESS.');

              console.log(image);
            }
          });

Server side, NodeJS, controller:

app.use(bodyParser.json({limit: '50mb'}));
app.use(bodyParser.urlencoded({limit: '50mb', extended: true}));

exports.uploadBase64 = function(req, res, next) {

  var uploadPath  = path.normalize(process.cwd() + '/packages/characters/public/assets/uploads/'),
      data        = new Buffer(''),
      imgURL      = undefined,        // public URL to show the pic
      type        = undefined;


  // In case the '/uploads' directoy doesn't exist
  if( !fs.existsSync(uploadPath) ) {
    fs.mkdirSync(uploadPath, 0755);
  }


  // Decoding the base64 image
  var data = new Buffer(req.body.image, 'base64');

  // Creating the name for the file --> characterId + type + timestamp + extension
  var filename = req.body.characterId + '-' + req.body.type + '-' + new Date().getTime().toString() + '.' + req.body.filetype;

  // Writing the image to filesystem
  fs.writeFile(uploadPath + filename, data, function (err) {

    if ( err ) {
      console.log(err);
      return res.status(500).json({
        error: 'Cannot upload the image. Sorry.'
      });
    }

    console.log('SAVED ON HD');

    console.log('FINISHING');

    // Sending success response
    res.json({
      imgURL: imgURL,
      type: type
    });

  });

};

The thing is the file stored in /uploads isn't work. I can't see the image. The base64 image is sent and the file is written to hard disk, but it is not possible to open it.

What's wrong? Any suggestion?

Thanks!

Rubén Jiménez
  • 1,835
  • 5
  • 21
  • 29

3 Answers3

8

It maybe that you are trying to save the file with the metadata. hence making the decoded data from base64 invalid.

Here is what I mean:

....

The data:image/png;base64 is just meta and is not encoded using base64 so:

You need to strip that from the sting then decode then save to disk.

I use this handy function:

function decodeBase64Image(dataString) {
  var matches = dataString.match(/^data:([A-Za-z-+\/]+);base64,(.+)$/),
    response = {};

  if (matches.length !== 3) {
    return new Error('Invalid input string');
  }

  response.type = matches[1];
  response.data = new Buffer(matches[2], 'base64');

  return response;
}

source: NodeJS write base64 image-file

So a snip from how I use it:

 var decodedImg = decodeBase64Image(imgB64Data);
 var imageBuffer = decodedImg.data;
 var type = decodedImg.type;
 var extension = mime.extension(type);
 var fileName =  "image." + extension;
 try{
       fs.writeFileSync(".tmp/uploads/" + fileName, imageBuffer, 'utf8');
    }
 catch(err){
    console.error(err)
 }

Where mime is the great node-mime lib. npm install mime.

Community
  • 1
  • 1
Johan Buys
  • 176
  • 1
  • 5
  • Your code works but not completely. It is not url-safe. The uploaded image returns blank. – Grogu Jun 22 '21 at 17:48
  • @Grogu Hi I am not sure what you mean, the original question was to save a base64 encoded image file to server as an Image. I believed that the image was a base64 url string, this contains the extra meta data. So I proposed to strip that info then decode the base64 string to binary data then save it. The saved image will not be url-safe as it now contains binary data. If you simply want to save it to be url-safe you should not strip the metadata and simply save the base64 url as is. – Johan Buys Jun 23 '21 at 07:30
0

if your using express file upload you could do this

routerApi.patch('/upload_gambar', (req, res) => {
    var file_data= req.files.upload_file_name.data;
    var base64Image = file_data.toString('base64');
}

if your using multer file upload you could do this

routerApi.patch('/upload_gambar', upload.single('upload_file_name'),(req, res) => {
    var file_data= req.file.buffer
    var base64Image = file_data.toString('base64');
}
0

Node js base64 to image convert and upload on disk after read and write here code working with me.

var imgBase64 = '';

var base64Data = imgBase64.replace(/^data:image\/png;base64,/, "");
    fs.writeFile(uploadPath+"/out.png", base64Data, 'base64', function(err) {
    console.log(err);
    });
fs.readFileSync(base64Data, {encoding: 'base64'});
pawan sen
  • 716
  • 5
  • 14