web-dev-qa-db-ja.com

NodeJS:base64でエンコードされたイメージをディスクに保存する

Expressアプリケーションは、ブラウザーからbase64エンコードされたPNG(toDataURL()を使用してcanvasから生成された)を受け取り、それをファイルに書き込みます。しかし、ファイルは有効な画像ファイルではなく、「ファイル」ユーティリティは単に「データ」として識別します。

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
});
126
mahemoff

必要以上にデータを変換していると思います。適切なエンコーディングでバッファを作成したら、ファイルにバッファを書き込むだけです。

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

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

new Buffer(...、 'base64')は、入力をbase64エンコード文字列として解釈することにより、入力文字列を単なるバイトの配列であるBufferに変換します。次に、そのバイト配列をファイルに書き込むだけです。

更新

コメントで述べたように、req.rawBodyはもはや問題ではありません。 express/connectを使用している場合は、bodyParser()ミドルウェアを使用し、req.bodyを使用する必要があります。標準Nodeを使用してこれを行う場合は、着信dataイベントBufferオブジェクトを集約し、この画像データ解析を行う必要がありますendコールバック内。

268
loganfsmyth

これは、base64イメージ形式を読み取り、データベースに適切な形式で保存する完全なソリューションです。

    // 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);
    }
19
Placeholder

更新

私はこれを見つけました PHPで問題を解決する興味深いリンク リンクに示されているspace+asに置き換えるのを忘れたと思います。

私はこの円を http://images-mediawiki-sites.thefullwiki.org/04/1/7/5/6204600836255205.png のようなサンプルとして取りました:

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

次に、私はそれを http://www.greywyvern.com/code/php/binary2base64 に通しました。



この文字列をbase64に保存し、コードから読み取りました。

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
});

私は円を取り戻しますが、面白いのはファイルサイズが変わったことです:)...

終わり

画像を読み返すとき、ヘッダーを設定する必要があると思います

たとえば、 imagepng PHPページから:

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

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

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

2行目のheader('Content-Type: image/png');は重要だと思います。そうしないと、画像はブラウザに表示されず、ブラウザに表示されるのはバイナリデータの束だけです。

Express では、単に次のようなものを使用します。 http://www.gravatar.com/avatar/cabf735ce7b8b4471ef46ea54f71832d?s=32&d=identicon&r=PG にあるグラバターを表示し、curl --head http://www.gravatar.com/avatar/cabf735ce7b8b4471ef46ea54f71832d?s=32&d=identicon&r=PGのときにjpegファイルになります。ヘッダーのみをリクエストするのは、curlが大量のバイナリアイテム(Google Chromeがすぐにダウンロードする)をコンソールに表示するためです。

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
15
Alfred

また、データURLの一部であるBase64でエンコードされた画像を保存する必要があったため、私(または他の誰か)が将来再び行う必要がある場合に備えて、小さなnpmモジュールを作成することになりました。 ba64 と呼ばれます。

簡単に言えば、Base64でエンコードされたイメージを含むデータURLを受け取り、イメージをファイルシステムに保存します。同期的または非同期的に保存できます。また、2つのヘルパー関数があります。1つはイメージのファイル拡張子を取得し、もう1つはBase64エンコードをdata:スキームプレフィックスから分離します。

以下に例を示します。

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
});

インストールしてください:npm i ba64 -S。レポはGitHubにあります: https://github.com/HarryStevens/ba64

追伸ba64はおそらくBase64エンコーディングとデコーディングを行うと仮定するかもしれないので、ba64はおそらく悪い名前だと後で思いつきました(既にそうしているモジュールはたくさんあります)。しかたがない。

5
Harry Stevens

base64画像をファイルに変換し、ランダムなIDまたは名前として保存する簡単な方法。

// 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
Carlos

Base64文字列を持つファイルからpng画像への変換。

動作する4つのバリアント。

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
Vladimir Buskin

ファイルを保存する関数の下に、base64ファイルを渡すだけで、ファイル名を返し、DBに保存します。

import fs from 'fs';
 const uuid = require('uuid/v1');

/*Download the base64 image in the server and returns the filename and path of image.*/
function saveImage(baseImage) {
    /*path of the folder where your project is saved. (In my case i got it from config file, root path of project).*/
    const uploadPath = "/home/documents/project";
    //path of folder where you want to save the image.
    const localPath = `${uploadPath}/uploads/images/`;
    //Find extension of file
    const ext = baseImage.substring(baseImage.indexOf("/")+1, baseImage.indexOf(";base64"));
    const fileType = baseImage.substring("data:".length,baseImage.indexOf("/"));
    //Forming regex to extract base64 data of file.
    const regex = new RegExp(`^data:${fileType}\/${ext};base64,`, 'gi');
    //Extract base64 data.
    const base64Data = baseImage.replace(regex, "");
    const filename = `${uuid()}.${ext}`;

    //Check that if directory is present or not.
    if(!fs.existsSync(`${uploadPath}/uploads/`)) {
        fs.mkdirSync(`${uploadPath}/uploads/`);
    }
    if (!fs.existsSync(localPath)) {
        fs.mkdirSync(localPath);
    }
    fs.writeFileSync(localPath+filename, base64Data, 'base64');
    return filename;
}

ブロッククォート

0
Shaik Matheen