Вчера я провел сеанс глубокого ночного кодирования и создал небольшое приложение node.js / JS (ну, на самом деле CoffeeScript, но CoffeeScript - это просто JavaScript, так что, скажем, JS).
какова цель:
- клиент отправляет данные холста (png) на сервер (через socket.io)
- сервер загружает изображение на amazon s3
шаг 1 выполнен.
на сервере теперь есть строка а-ля
data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAMgAAADICAYAAACt...
у меня вопрос: каковы мои следующие шаги для «потоковой передачи» / загрузки этих данных в Amazon S3 и создания там реального изображения?
knox https://github.com/LearnBoost/knox кажется отличной библиотекой для ПОМЕЩЕНИЯ чего-то в S3, но чего мне не хватает, так это связующего звена между строкой изображения в кодировке base64 и фактическим действием загрузки ?
Любые идеи, указатели и отзывы приветствуются.
javascript
node.js
amazon-s3
coffeescript
Франц Энценхофер
источник
источник
Ответы:
Для людей, которые все еще борются с этой проблемой. Вот подход, который я использовал с собственным aws-sdk:
var AWS = require('aws-sdk'); AWS.config.loadFromPath('./s3_config.json'); var s3Bucket = new AWS.S3( { params: {Bucket: 'myBucket'} } );
Внутри вашего метода маршрутизатора (ContentType должен быть установлен на тип содержимого файла изображения):
buf = Buffer.from(req.body.imageBinary.replace(/^data:image\/\w+;base64,/, ""),'base64') var data = { Key: req.body.userId, Body: buf, ContentEncoding: 'base64', ContentType: 'image/jpeg' }; s3Bucket.putObject(data, function(err, data){ if (err) { console.log(err); console.log('Error uploading data: ', data); } else { console.log('successfully uploaded the image!'); } });
файл s3_config.json:
{ "accessKeyId":"xxxxxxxxxxxxxxxx", "secretAccessKey":"xxxxxxxxxxxxxx", "region":"us-east-1" }
источник
How to make S3 generates a unique KEY to prevent from overriding files?
аIf I don't set the ContentType, when I download the files I won't be able to get the correct file?
я имею ввиду, я получу такой испорченный файл? Заранее спасибо!ContentEncoding: 'base64'
это неверно, потому чтоnew Buffer(..., 'base64')
декодирует строку в кодировке base64 в ее двоичное представление.хорошо, это ответ, как сохранить данные холста в файл
в основном это выглядит так в моем коде
buf = new Buffer(data.dataurl.replace(/^data:image\/\w+;base64,/, ""),'base64') req = knoxClient.put('/images/'+filename, { 'Content-Length': buf.length, 'Content-Type':'image/png' }) req.on('response', (res) -> if res.statusCode is 200 console.log('saved to %s', req.url) socket.emit('upload success', imgurl: req.url) else console.log('error %d', req.statusCode) ) req.end(buf)
источник
Вот код из одной статьи, которую я нашел, размещенной ниже:
const imageUpload = async (base64) => { const AWS = require('aws-sdk'); const { ACCESS_KEY_ID, SECRET_ACCESS_KEY, AWS_REGION, S3_BUCKET } = process.env; AWS.config.setPromisesDependency(require('bluebird')); AWS.config.update({ accessKeyId: ACCESS_KEY_ID, secretAccessKey: SECRET_ACCESS_KEY, region: AWS_REGION }); const s3 = new AWS.S3(); const base64Data = new Buffer.from(base64.replace(/^data:image\/\w+;base64,/, ""), 'base64'); const type = base64.split(';')[0].split('/')[1]; const userId = 1; const params = { Bucket: S3_BUCKET, Key: `${userId}.${type}`, // type is not required Body: base64Data, ACL: 'public-read', ContentEncoding: 'base64', // required ContentType: `image/${type}` // required. Notice the back ticks } let location = ''; let key = ''; try { const { Location, Key } = await s3.upload(params).promise(); location = Location; key = Key; } catch (error) { } console.log(location, key); return location; } module.exports = imageUpload;
Прочитайте больше: http://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/S3.html#upload-property
Кредиты: https://medium.com/@mayneweb/upload-a-base64-image-data-from-nodejs-to-aws-s3-bucket-6c1bd945420f
источник
Принятый ответ отлично работает, но если кому-то нужно принять любой файл вместо изображений, это регулярное выражение отлично работает:
/^data:.+;base64,/
источник