NodeJS: Guardar una imagen codificada en base64 en el disco


Mi aplicación Express recibe un PNG codificado en base64 desde el navegador (generado desde canvas con toDataURL() ) y lo escribe en un archivo. Pero el archivo no es un archivo de imagen válido, y la utilidad "archivo" simplemente lo identifica como "datos".

var body = req.rawBody,
  base64Data = body.replace(/^data:image\/png;base64,/,""),
  binaryData = new Buffer(base64Data, 'base64').toString('binary');

require("fs").writeFile("out.png", binaryData, "binary", function(err) {
  console.log(err); // writes out file without error, but it's not a valid image
});
Author: mahemoff, 2011-08-03

6 answers

Creo que está convirtiendo los datos un poco más de lo necesario. Una vez que cree el búfer con la codificación adecuada, solo necesita escribir el búfer en el archivo.

var base64Data = req.rawBody.replace(/^data:image\/png;base64,/, "");

require("fs").writeFile("out.png", base64Data, 'base64', function(err) {
  console.log(err);
});

Nuevo búfer (..., 'base64') convertirá la cadena de entrada en un búfer, que es solo una matriz de bytes, interpretando la entrada como una cadena codificada en base64. Entonces solo puede escribir esa matriz de bytes en el archivo.

Actualizar

Como se menciona en los comentarios, req.rawBody ya no es una cosa. Si están usando express/connect entonces debe usar el middleware bodyParser() y usar req.body, y si está haciendo esto usando el nodo estándar, entonces necesita agregar los objetos entrantes data event Buffer y hacer este análisis de datos de imagen en la devolución de llamada end.

 223
Author: loganfsmyth,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/ajaxhispano.com/template/agent.layouts/content.php on line 61
2014-06-17 05:24:10

Esta es mi solución completa que leería cualquier formato de imagen base64 y lo guardaría en el formato adecuado en la base de datos:

    // Save base64 image to disk
    try
    {
        // Decoding base-64 image
        // Source: http://stackoverflow.com/questions/20267939/nodejs-write-base64-image-file
        function decodeBase64Image(dataString) 
        {
          var matches = dataString.match(/^data:([A-Za-z-+\/]+);base64,(.+)$/);
          var response = {};

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

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

          return response;
        }

        // Regular expression for image type:
        // This regular image extracts the "jpeg" from "image/jpeg"
        var imageTypeRegularExpression      = /\/(.*?)$/;      

        // Generate random string
        var crypto                          = require('crypto');
        var seed                            = crypto.randomBytes(20);
        var uniqueSHA1String                = crypto
                                               .createHash('sha1')
                                                .update(seed)
                                                 .digest('hex');

        var base64Data = '...';

        var imageBuffer                      = decodeBase64Image(base64Data);
        var userUploadedFeedMessagesLocation = '../img/upload/feed/';

        var uniqueRandomImageName            = 'image-' + uniqueSHA1String;
        // This variable is actually an array which has 5 values,
        // The [1] value is the real image extension
        var imageTypeDetected                = imageBuffer
                                                .type
                                                 .match(imageTypeRegularExpression);

        var userUploadedImagePath            = userUploadedFeedMessagesLocation + 
                                               uniqueRandomImageName +
                                               '.' + 
                                               imageTypeDetected[1];

        // Save decoded binary image to disk
        try
        {
        require('fs').writeFile(userUploadedImagePath, imageBuffer.data,  
                                function() 
                                {
                                  console.log('DEBUG - feed:message: Saved to disk image attached by user:', userUploadedImagePath);
                                });
        }
        catch(error)
        {
            console.log('ERROR:', error);
        }

    }
    catch(error)
    {
        console.log('ERROR:', error);
    }
 18
Author: Placeholder,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/ajaxhispano.com/template/agent.layouts/content.php on line 61
2014-10-16 06:26:17

ACTUALIZAR

He encontrado este enlace interesante cómo resolver su problema en PHP. Creo que olvidaste reemplazar space por + como se muestra en el enlace.

Tomé este círculo de http://images-mediawiki-sites.thefullwiki.org/04/1/7/5/6204600836255205.png como muestra que se parece a:

http://images-mediawiki-sites.thefullwiki.org/04/1/7/5/6204600836255205.png

Luego lo paso http://www.greywyvern.com/code/php/binary2base64 que me devolvió:



Guardó esta cadena en base64 que leo en mi código.

var fs      = require('fs'),
data        = fs.readFileSync('base64', 'utf8'),
base64Data,
binaryData;

base64Data  =   data.replace(/^data:image\/png;base64,/, "");
base64Data  +=  base64Data.replace('+', ' ');
binaryData  =   new Buffer(base64Data, 'base64').toString('binary');

fs.writeFile("out.png", binaryData, "binary", function (err) {
    console.log(err); // writes out file without error, but it's not a valid image
});

Obtengo un círculo de vuelta, pero lo curioso es que el tamaño del archivo ha cambiado :)...

FIN

Cuando lees la imagen, creo que necesitas configurar los encabezados

Tomemos por ejemplo imagepng de la página PHP:

<?php
$im = imagecreatefrompng("test.png");

header('Content-Type: image/png');

imagepng($im);
imagedestroy($im);
?>

Creo que la segunda línea header('Content-Type: image/png');, es importante de lo contrario su imagen no se mostrará en el navegador, pero solo un montón de datos binarios se muestra al navegador.

In Express you simplemente usaría algo como a continuación. Voy a mostrar su gravatar que se encuentra en http://www.gravatar.com/avatar/cabf735ce7b8b4471ef46ea54f71832d?s=32&d=identicon&r=PG y es un archivo jpeg cuando curl --head http://www.gravatar.com/avatar/cabf735ce7b8b4471ef46ea54f71832d?s=32&d=identicon&r=PG. Solo solicito encabezados porque curl de lo contrario mostrará un montón de cosas binarias (Google Chrome inmediatamente va a descargar) a la consola:

curl --head "http://www.gravatar.com/avatar/cabf735ce7b8b4471ef46ea54f71832d?s=32&d=identicon&r=PG"
HTTP/1.1 200 OK
Server: nginx
Date: Wed, 03 Aug 2011 12:11:25 GMT
Content-Type: image/jpeg
Connection: keep-alive
Last-Modified: Mon, 04 Oct 2010 11:54:22 GMT
Content-Disposition: inline; filename="cabf735ce7b8b4471ef46ea54f71832d.jpeg"
Access-Control-Allow-Origin: *
Content-Length: 1258
X-Varnish: 2356636561 2352219240
Via: 1.1 varnish
Expires: Wed, 03 Aug 2011 12:16:25 GMT
Cache-Control: max-age=300
Source-Age: 1482

$ mkdir -p ~/tmp/6922728
$ cd ~/tmp/6922728/
$ touch app.js

App.js

var app = require('express').createServer();

app.get('/', function (req, res) {
    res.contentType('image/jpeg');
    res.sendfile('cabf735ce7b8b4471ef46ea54f71832d?s=32&d=identicon&r=PG');
});

app.get('/binary', function (req, res) {
    res.sendfile('cabf735ce7b8b4471ef46ea54f71832d?s=32&d=identicon&r=PG');
});

app.listen(3000);

$ wget "http://www.gravatar.com/avatar/cabf735ce7b8b4471ef46ea54f71832d?s=32&d=identicon&r=PG"
$ node app.js
 14
Author: Alfred,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/ajaxhispano.com/template/agent.layouts/content.php on line 61
2011-08-03 21:02:38

También tuve que guardar imágenes codificadas Base64 que son parte de URL de datos, así que terminé haciendo un pequeño módulo npm para hacerlo en caso de que yo (o alguien más) necesitara hacerlo de nuevo en el futuro. Se llama ba64 .

En pocas palabras, toma una URL de datos con una imagen codificada en Base64 y guarda la imagen en su sistema de archivos. Puede guardar de forma sincrónica o asincrónica. También tiene dos funciones auxiliares, una para obtener la extensión de archivo de la imagen, y la otra para separar la codificación Base64 desde el prefijo de scheme data:.

Aquí hay un ejemplo:

var ba64 = require("ba64"),
    data_url = "data:image/jpeg;base64,[Base64 encoded image goes here]";

// Save the image synchronously.
ba64.writeImageSync("myimage", data_url); // Saves myimage.jpeg.

// Or save the image asynchronously.
ba64.writeImage("myimage", data_url, function(err){
    if (err) throw err;

    console.log("Image saved successfully");

    // do stuff
});

Instálalo: npm i ba64 -S. El repositorio está en GitHub: https://github.com/HarryStevens/ba64 .

P.d. Se me ocurrió más tarde que ba64 es probablemente un mal nombre para el módulo, ya que la gente puede asumir que hace la codificación y decodificación Base64, que no lo hace (hay muchos módulos que ya lo hacen). Oh, bueno.

 3
Author: Harry Stevens,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/ajaxhispano.com/template/agent.layouts/content.php on line 61
2017-10-15 16:30:26

Una manera fácil de convertir la imagen base64 en un archivo y guardarla como un id o nombre aleatorio.

// to create some random id or name for your image name
const imgname = new Date().getTime().toString();

// to declare some path to store your converted image
const path = yourpath.png    

// image takes from body which you uploaded
const imgdata = req.body.image;    

// to convert base64 format into random filename
const base64Data = imgdata.replace(/^data:([A-Za-z-+/]+);base64,/, '');
fs.writeFile(path, base64Data, 'base64', (err) => {
    console.log(err);
});

// assigning converted image into your database
req.body.coverImage = imgname
 1
Author: Carlos,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/ajaxhispano.com/template/agent.layouts/content.php on line 61
2017-08-24 13:37:36

Convertir de archivo con cadena base64 a imagen png.

4 variantes que funciona.

var {promisify} = require('util');
var fs = require("fs");

var readFile = promisify(fs.readFile)
var writeFile = promisify(fs.writeFile)

async function run () {

  // variant 1
  var d = await readFile('./1.txt', 'utf8')
  await writeFile("./1.png", d, 'base64')

  // variant 2
  var d = await readFile('./2.txt', 'utf8')
  var dd = new Buffer(d, 'base64')
  await writeFile("./2.png", dd)

  // variant 3
  var d = await readFile('./3.txt')
  await writeFile("./3.png", d.toString('utf8'), 'base64')

  // variant 4
  var d = await readFile('./4.txt')
  var dd = new Buffer(d.toString('utf8'), 'base64')
  await writeFile("./4.png", dd)

}

run();
 1
Author: Vladimir Buskin,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/ajaxhispano.com/template/agent.layouts/content.php on line 61
2018-04-06 08:37:43