Storing a buffer in RethinkDB

205 Views Asked by At

I'm uploading a file (multipart/form-data) to a Koa and want to store it into RethinkDB.

I parse it with co-busboy, which results in a stream.

Then I convert the stream into a buffer by attaching data/end listeners to it, gathering all buffers and concatenating them.

After I'm done with this, I get my DB record, put the buffer into the right field and save it.

But I always get a range error, so the update is never saved.

Does the buffer I created need some additional information for RethinkDB to store it?

1

There are 1 best solutions below

2
On

You don't need any additional information to store it. Taking direcly example from koajs https://github.com/koajs/examples/blob/master/multipart/app.js, use what you described, here is the code that works for me.

var os = require('os');
var path = require('path');
var koa = require('koa');
var fs = require('co-fs');
var parse = require('co-busboy');
var saveTo = require('save-to');
var app = module.exports = koa();
var r = require('rethinkdb')


r.connect()
  .then(function(conn) {
    app.use(function *(){
      // parse the multipart body
      var parts = parse(this, {
        autoFields: true // saves the fields to parts.field(s)
      });

      // create a temporary folder to store files
      var tmpdir = path.join(os.tmpdir(), uid());

      // make the temporary directory
      yield fs.mkdir(tmpdir);

      // list of all the files
      var files = [];
      var file;

      // yield each part as a stream
      var bufs = [];
      var part;
      while (part = yield parts) {
        // filename for this part
        files.push(file = path.join(tmpdir, part.filename));
        // save the file
        // yield saveTo(part, file);
        // We are saving into RethinkDB
        part.on('data', function(d){ bufs.push(d); });
        part.on('end', function(){
          var buf = Buffer.concat(bufs);

          r.table('user').insert({
            file: buf
          })
          .run(conn)
          .then(function(cursor) {
            console.log(cursor)
          })

        })
      }


      // return all the filenames as an array
      // after all the files have finished downloading
      this.body = files;

    })

})


if (!module.parent) app.listen(3000);

function uid() {
  return Math.random().toString(36).slice(2);
}

Then sending a request:

curl -F "file=@./a.png" 127.0.0.1:3000

A new document should be inserted with the binary data