node.js:当传递过多数据时,gm会引发生成E2BIG错误

时间:2018-11-05 13:20:58

标签: node.js imagemagick gm

以下代码引发错误:

const COUNT = 2528; // 2527 works, 2528 errors

const gm = require('gm').subClass({ imageMagick: true });

const brokenData = [];

for (let i = 0; i < COUNT; i++) {
  brokenData.push([
    Math.random() * 500, Math.random() * 500
  ]);
}

const tile = gm('./blank-tile.png')
  .resize(500, 500)
  .fill("red");

brokenData.forEach((point) => {
  tile.drawCircle(point[0], point[1], point[0] + 4, point[1]);
});

tile.write(__dirname + '/test.png', (err) => {
  if (err) {
    throw err;
  }

  console.log('success');
});

根据评论,绘制2527个圆是可以的,但会在2528个圆上引发错误。每次至少在我的机器上都是一样的。

这是错误:

Error: spawn E2BIG
    at ChildProcess.spawn (internal/child_process.js:358:11)
    at Object.spawn (child_process.js:533:9)
    at spawn (/Users/callumacrae/Sites/testing-gm/node_modules/cross-spawn/index.js:17:18)
    at gm._spawn (/Users/callumacrae/Sites/testing-gm/node_modules/gm/lib/command.js:224:14)
    at /Users/callumacrae/Sites/testing-gm/node_modules/gm/lib/command.js:101:12
    at series (/Users/callumacrae/Sites/testing-gm/node_modules/array-series/index.js:11:36)
    at gm._preprocess (/Users/callumacrae/Sites/testing-gm/node_modules/gm/lib/command.js:177:5)
    at gm.write (/Users/callumacrae/Sites/testing-gm/node_modules/gm/lib/command.js:99:10)
    at Object.<anonymous> (/Users/callumacrae/Sites/testing-gm/test.js:21:6)
    at Module._compile (internal/modules/cjs/loader.js:688:30)

我假设它来自gm中的某个地方,因为我没有提供任何长参数列表!

无论我使用imagemagick还是graphicsmagick,都会发生相同的事情。节点版本10.13.0。

有什么想法吗?

1 个答案:

答案 0 :(得分:1)

我对node-gm并不真正熟悉,但是我感觉.drawCircle(x1, y1, x2, y2)方法只是附加了一个命令行参数-draw "circle x1,y1 x2,y2"。因此,在2527年之后,绘制命令超出了参数缓冲区。

使用ImageMagick,如果您有大量的绘制命令,则可以写入文件并告诉绘制命令从文件中读取。

文件看起来像...

# circles.txt
circle x1,y1 x2,y2
circle x1,y1 x2,y2
circle x1,y1 x2,y2
circle x1,y1 x2,y2

并使用符号(@)前缀引用文件。

convert ... -draw @cicles.txt ...

因此,您也可以创建一个临时文件,编写绘图命令,然后调用..

const tile = gm('./blank-tile.png')
  .resize(500, 500)
  .fill("red")
  .draw("@circles.txt");

但是,我不确定node-gm是否支持此功能,和/或许多现代系统使用默认的安全协议禁用了MVGTXT。值得调查。