2016-10-13 11 views
0

大きい画像を送信しようとすると、Lambdaでこのコードに問題が発生します。 配列SIZESが示すように、たくさんのイメージを送信し、サイズを変更していくつかの異なるサイズを作成しました。Lambda大きな画像で空のバッファエラーが発生する

マイラムダ構成が高い、最大メモリ(1536メガバイト)最大タイムアウト5mimです。

大きな画像をぼかしたときに問題が発生します。私はすべてのインターネットを通して検索しましたが、私は解決策を見つけることができませんでした。

エラー

Unhandled rejection Error: Stream yields empty buffer 
at Socket.<anonymous> (/var/task/node_modules/gm/lib/command.js:57:17) 
at emitNone (events.js:72:20) 
at Socket.emit (events.js:166:7) 
at endReadableNT (_stream_readable.js:905:12) 
at nextTickCallbackWith2Args (node.js:437:9) 
at process._tickDomainCallback (node.js:392:17) 

**ここでは、コードがあります。*

var gm = require('gm').subClass({ 
     imageMagick: true 
    }), 
    aws = require('aws-sdk'), 
    async = require('async'), 
    fs = require('fs'), 
    promise = require('bluebird'), 
    request = require('request').defaults({ 
     encoding: null 
    }); 

var BUCKET = "myBucket", 
    FOLDER = 'photos/', 
    s3 = new aws.S3({ 
     accessKeyId: 'myKeyId', 
     secretAccessKey: 'mySecretKey' 
    }), 
    SIZES = [{ 
     type: 'original',size: 1080,mark: false,hide: true 
    }, { 
     type: 'thumb',size: 120, mark: false, hide: false 
    }, { 
     type: 'thumb',size: 240, mark: false, hide: false 
    }, { 
     type: 'thumb',size: 360, mark: false, hide: false 
    }, { 
     type: 'card',size: 350 ,mark: false, hide: false 
    }, { 
     type: 'full',size: 360, mark: true, hide: false 
    }, { 
     type: 'full',size: 540, mark: true, hide: false 
    }, { 
     type: 'full',size: 720, mark: true, hide: false 
    }, { 
     type: 'full',size: 900, mark: true, ide: false 
    }, { 
     type: 'full',size: 1080, mark: true, hide: false 
    }]; 

var service = { 
    s3: { 
     download: function(bucket, key) { 
      return new promise(function(resolve, reject) { 
       s3.getObject({ 
        Bucket: bucket, 
        Key: key 
       }, function(err, data) { 
        if (err) { 
         reject(err); 
        } else { 
         resolve(data); 
        } 
       }); 

      }); 
     }, 
     upload: { 
      async: function(photo, bucket, key, content_type, private) { 
       return new promise(function(resolve, reject) { 
        s3.putObject({ 
         Bucket: bucket, 
         Key: key, 
         Body: photo, 
         ContentType: content_type, 
         ACL: private ? 'private' : 'public-read' 
        }, function(err, data) { 
         if (err) { 
          reject(err) 
         } else { 
          resolve(key); 
         } 
        }); 
       }); 
      } 
     } 
    }, 
    watermark: { 
     resize: function(photo, name) { 
      var temp = '/tmp/w_' + name + '.png'; 
      return new promise(function(resolve, reject) { 
       gm(photo).size(function(err, size) { 
        var that = this; 
        if (err) { 
         reject(err); 
        } 
        var smaller = size.width > size.height ? size.height : size.width; 
        var _size = smaller * 0.8; 

        request.get('https://s3.amazonaws.com/FOLDER/photos/watermark.png', function(e, r, body) { 
         gm(body) 
          .resize(_size) 
          .write(temp, function(error) { 
           if (error) reject(error); 

           that.composite(temp) 
            .gravity('Center') 
            .toBuffer(function(e, buffer) { 
             if (e) reject(e); 

             fs.access(temp, fs.F_OK, function(err) { 
              if (!err) { 
               fs.unlinkSync(temp); 
              } 
             }); 

             resolve(buffer); 
            }); 
          }); 
        }); 
       }); 
      }); 
     } 
    }, 
    resize: function(photo, width, height) { 
     return new promise(function(resolve, reject) { 
      gm(photo).resize(width, height).toBuffer(function(err, buffer) { 
       if (err) { 
        reject(err); 
       } 

       resolve(buffer); 
      }); 
     }); 
    }, 
    crop: function(photo, width, height) { 
     return new promise(function(resolve, reject) { 
      gm(photo).size(function(err, size) { 
       if (err) { 
        reject(err); 
       } 

       var x = 0, 
        y = 0; 
       if (size.width > size.height) { 
        this.resize(null, height); 
        x = (((height * size.width)/size.height) - width)/2; 
       } else { 
        this.resize(width); 
        y = (((width * size.height)/size.width) - height)/2; 
       } 

       this.crop(width, height, x, y).toBuffer(function(err, buffer) { 
        resolve(buffer); 
       }); 
      }); 
     }); 
    }, 
    process: function(type, photo, width, height) { 
     if (type == 'thumb') { 
      return service.crop(photo, width, height); 
     } else { 
      return service.resize(photo, width, height); 
     } 
    } 
}; 


exports.handler = function(event, context, callback) { 
    var $filename = event.filename, 
     $session = event.session, 
     $mimetype = null, 
     $original = null, 
     $watermarked = null; 

    async.waterfall([ 
     function(_next) { 
      service.s3.download(BUCKET, FOLDER + $session + '/original/' + $filename).then(function(_original) { 
       $mimetype = _original.ContentType; 
       $original = _original.Body; 
       _next(null); 
      }, function(_error) { 
       _next(_error); 
      }); 
     }, 
     function(_next) { 
      service.watermark.resize($original, (Math.random().toString(36).slice(-6) + new Date().getTime())).then(function(_watermarked) { 
       $watermarked = _watermarked; 
       _next(null); 
      }); 
     }, 
     function(_next) { 
      async.each(SIZES, function(_image, _callback) { 
       service.process(_image.type, _image.mark ? $watermarked : $original, _image.size, _image.size).then(function(_generated) { 
        service.s3.upload.async(_generated, BUCKET, FOLDER + $session + '/' + _image.type + '/' + (_image.type != 'card' ? _image.size + '_' : '') + $filename, $mimetype, _image.hide).then(function() { 
         _callback(null); 
        }); 
       }); 
      }, function() { 
       _next(null); 
      }); 
     } 
    ], function() { 
     callback(null, 'Imagens processadas. Session: ' + $session); 
    }); 
}; 

答えて

1

最後の処理では時間がかかり、ラムダはメモリ不足になりました。

この問題は、2つのサイズで配列を分割して解決しました。次に、新しい配列にasync.eachを使って関数を複製します。

この新しい方法は完全に実行され、すべての画像が作成されます。

0

私は前にこのエラーが発生しましたし、最終的にはメモリの制限を打つにそれをトレースしてきました。幸いにも、私は自分の記憶のサイズを増やすことができ、それが問題を解決しました。

残念ながら、あなたはすでに最大許容メモリに達していて、まだ制限に達しています。アップロードされている画像はどれだけ大きいですか?彼らが本当に大きく、出力を1つのイメージとして必要とする場合は、巨大なイメージファイルをアップロードし、それを切り抜いてx4以上の小さなイメージを作成し、それらのサイズを変更し、それぞれを再び一緒に戻してください。

+0

イメージは約12MBです。しかし、私は5メガバイト、7メガバイトのような小さな画像で問題を抱えていました。 –

関連する問題