CartoDB-SQL-API/lib/models/formats/ogr/shp.js

134 lines
4.2 KiB
JavaScript
Raw Normal View History

2018-10-24 21:42:33 +08:00
'use strict';
2015-05-13 19:00:01 +08:00
var step = require('step');
var fs = require('fs');
var spawn = require('child_process').spawn;
2019-01-16 22:58:11 +08:00
var Ogr = require('./../ogr');
2013-05-16 17:24:52 +08:00
2019-12-24 01:19:08 +08:00
function ShpFormat () {
2013-05-16 17:24:52 +08:00
}
2019-01-16 22:58:11 +08:00
ShpFormat.prototype = new Ogr('shp');
2013-05-16 17:24:52 +08:00
2019-12-24 01:19:08 +08:00
ShpFormat.prototype._contentType = 'application/zip; charset=utf-8';
ShpFormat.prototype._fileExtension = 'zip';
// As of GDAL 1.10 SRID detection is bogus, so we use
// our own method. See:
// http://trac.osgeo.org/gdal/ticket/5131
// http://trac.osgeo.org/gdal/ticket/5287
// http://github.com/CartoDB/CartoDB-SQL-API/issues/110
// http://github.com/CartoDB/CartoDB-SQL-API/issues/116
2014-08-03 02:27:05 +08:00
ShpFormat.prototype._needSRS = true;
2013-05-16 17:24:52 +08:00
2019-12-24 01:19:08 +08:00
ShpFormat.prototype.generate = function (options, callback) {
this.toSHP(options, callback);
2013-05-16 17:24:52 +08:00
};
2014-08-03 02:27:05 +08:00
ShpFormat.prototype.toSHP = function (options, callback) {
2019-12-24 01:19:08 +08:00
var dbname = options.database;
2019-12-27 01:01:38 +08:00
var userId = options.user_id;
2019-12-24 01:19:08 +08:00
var gcol = options.gn;
var sql = options.sql;
var skipfields = options.skipfields;
var filename = options.filename;
var fmtObj = this;
var zip = global.settings.zipCommand || 'zip';
var zipOptions = '-qrj';
var tmpdir = global.settings.tmpDir || '/tmp';
var reqKey = this.limitPathname(['shp', dbname, userId, gcol, this.generateMD5(sql)].concat(skipfields).join(':'));
2019-12-24 01:19:08 +08:00
var outdirpath = tmpdir + '/sqlapi-' + process.pid + '-' + reqKey;
var zipfile = outdirpath + '.zip';
var shapefile = outdirpath + '/' + filename + '.shp';
// TODO: following tests:
// - fetch query with no "the_geom" column
step(
function createOutDir () {
fs.mkdir(outdirpath, 0o777, this);
},
function spawnDumper (err) {
if (err) {
throw err;
}
fmtObj.toOGR(options, 'ESRI Shapefile', shapefile, this);
},
function doZip (err) {
if (err) {
throw err;
}
var next = this;
var child = spawn(zip, [zipOptions, zipfile, outdirpath]);
child.on('error', function (err) {
next(new Error('Error executing zip command, ' + err));
});
var stderrData = [];
child.stderr.setEncoding('utf8');
child.stderr.on('data', function (data) {
stderrData.push(data);
});
2013-05-16 17:24:52 +08:00
2019-12-24 01:19:08 +08:00
child.on('exit', function (code) {
if (code !== 0) {
var errMessage = 'Zip command return code ' + code;
if (stderrData.length) {
errMessage += ', Error: ' + stderrData.join('\n');
}
return next(new Error(errMessage));
}
return next();
2013-05-16 17:24:52 +08:00
});
2019-12-24 01:19:08 +08:00
},
function cleanupDir (topError) {
var next = this;
// Unlink the dir content
var unlinkall = function (dir, files, finish) {
var f = files.shift();
if (!f) { finish(null); return; }
var fn = dir + '/' + f;
fs.unlink(fn, function (err) {
if (err) {
console.log('Unlinking ' + fn + ': ' + err);
finish(err);
} else {
unlinkall(dir, files, finish);
}
});
};
fs.readdir(outdirpath, function (err, files) {
if (err) {
if (err.code !== 'ENOENT') {
next(new Error([topError, err].join('\n')));
} else {
next(topError);
}
} else {
unlinkall(outdirpath, files, function (/* err */) {
fs.rmdir(outdirpath, function (err) {
if (err) {
console.log('Removing dir ' + outdirpath + ': ' + err);
}
next(topError, zipfile);
});
});
}
});
},
function finalStep (err, zipfile) {
callback(err, zipfile);
2013-05-16 17:24:52 +08:00
}
2019-12-24 01:19:08 +08:00
);
};
2013-05-16 17:24:52 +08:00
2014-08-03 02:27:05 +08:00
module.exports = ShpFormat;