Support for serving raw vector tiles
This commit is contained in:
parent
75e5fd1018
commit
7bef26baed
4 changed files with 122 additions and 18 deletions
|
@ -37,13 +37,9 @@ module.exports = function(maps, options, prefix) {
|
||||||
var lock = new asyncLock();
|
var lock = new asyncLock();
|
||||||
|
|
||||||
var app = express().disable('x-powered-by'),
|
var app = express().disable('x-powered-by'),
|
||||||
domains = [],
|
domains = options.domains,
|
||||||
tilePath = '/{z}/{x}/{y}.{format}';
|
tilePath = '/{z}/{x}/{y}.{format}';
|
||||||
|
|
||||||
if (options.domains && options.domains.length > 0) {
|
|
||||||
domains = options.domains.split(',');
|
|
||||||
}
|
|
||||||
|
|
||||||
var rootPath = path.join(process.cwd(), options.root);
|
var rootPath = path.join(process.cwd(), options.root);
|
||||||
|
|
||||||
var styleUrl = options.style;
|
var styleUrl = options.style;
|
101
src/serve_vector.js
Normal file
101
src/serve_vector.js
Normal file
|
@ -0,0 +1,101 @@
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
var async = require('async'),
|
||||||
|
asyncLock = require('async-lock'),
|
||||||
|
crypto = require('crypto'),
|
||||||
|
fs = require('fs'),
|
||||||
|
path = require('path'),
|
||||||
|
util = require('util'),
|
||||||
|
zlib = require('zlib');
|
||||||
|
|
||||||
|
var abaculus = require('abaculus'),
|
||||||
|
clone = require('clone'),
|
||||||
|
concat = require('concat-stream'),
|
||||||
|
express = require('express'),
|
||||||
|
mercator = new (require('sphericalmercator'))(),
|
||||||
|
mbgl = require('mapbox-gl-native'),
|
||||||
|
mbtiles = require('mbtiles'),
|
||||||
|
request = require('request');
|
||||||
|
|
||||||
|
var utils = require('./utils');
|
||||||
|
|
||||||
|
module.exports = function(maps, options, prefix) {
|
||||||
|
var lock = new asyncLock();
|
||||||
|
|
||||||
|
var app = express().disable('x-powered-by'),
|
||||||
|
domains = options.domains,
|
||||||
|
tilePath = '/{z}/{x}/{y}.pbf';
|
||||||
|
|
||||||
|
var rootPath = path.join(process.cwd(), options.root);
|
||||||
|
|
||||||
|
var mbtilesPath = options.mbtiles;
|
||||||
|
var map = {
|
||||||
|
tileJSON: {}
|
||||||
|
};
|
||||||
|
maps[prefix] = map;
|
||||||
|
|
||||||
|
var source = new mbtiles(path.join(rootPath, mbtilesPath), function(err) {
|
||||||
|
source.getInfo(function(err, info) {
|
||||||
|
map.tileJSON['name'] = prefix.substr(1);
|
||||||
|
|
||||||
|
Object.assign(map.tileJSON, info);
|
||||||
|
|
||||||
|
map.tileJSON['tilejson'] = '2.0.0';
|
||||||
|
map.tileJSON['basename'] = prefix.substr(1);
|
||||||
|
map.tileJSON['format'] = 'pbf';
|
||||||
|
|
||||||
|
Object.assign(map.tileJSON, options.options || {});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
var tilePattern = tilePath
|
||||||
|
.replace('{z}', ':z(\\d+)')
|
||||||
|
.replace('{x}', ':x(\\d+)')
|
||||||
|
.replace('{y}', ':y(\\d+)');
|
||||||
|
|
||||||
|
var getTile = function(z, x, y, callback) {
|
||||||
|
source.getTile(z, x, y, function(err, data, headers) {
|
||||||
|
if (err) {
|
||||||
|
callback(err);
|
||||||
|
} else {
|
||||||
|
var md5 = crypto.createHash('md5').update(data).digest('base64');
|
||||||
|
headers['content-md5'] = md5;
|
||||||
|
headers['content-type'] = 'application/x-protobuf';
|
||||||
|
headers['content-encoding'] = 'gzip';
|
||||||
|
|
||||||
|
callback(null, data, headers);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
app.get(tilePattern, function(req, res, next) {
|
||||||
|
var z = req.params.z | 0,
|
||||||
|
x = req.params.x | 0,
|
||||||
|
y = req.params.y | 0;
|
||||||
|
return getTile(z, x, y, function(err, data, headers) {
|
||||||
|
if (err) {
|
||||||
|
return next(err);
|
||||||
|
}
|
||||||
|
if (headers) {
|
||||||
|
res.set(headers);
|
||||||
|
}
|
||||||
|
if (data == null) {
|
||||||
|
return res.status(404).send('Not found');
|
||||||
|
} else {
|
||||||
|
return res.status(200).send(data);
|
||||||
|
}
|
||||||
|
}, res, next);
|
||||||
|
});
|
||||||
|
|
||||||
|
app.get('/index.json', function(req, res, next) {
|
||||||
|
var info = clone(map.tileJSON);
|
||||||
|
|
||||||
|
info.tiles = utils.getTileUrls(req.protocol, domains, req.headers.host,
|
||||||
|
prefix, tilePath, info.format,
|
||||||
|
req.query.key);
|
||||||
|
|
||||||
|
return res.send(info);
|
||||||
|
});
|
||||||
|
|
||||||
|
return app;
|
||||||
|
};
|
|
@ -13,7 +13,8 @@ var async = require('async'),
|
||||||
express = require('express'),
|
express = require('express'),
|
||||||
morgan = require('morgan');
|
morgan = require('morgan');
|
||||||
|
|
||||||
var serve = require('./app'),
|
var serve_raster = require('./serve_raster'),
|
||||||
|
serve_vector = require('./serve_vector'),
|
||||||
utils = require('./utils');
|
utils = require('./utils');
|
||||||
|
|
||||||
module.exports = function(opts, callback) {
|
module.exports = function(opts, callback) {
|
||||||
|
@ -36,7 +37,11 @@ module.exports = function(opts, callback) {
|
||||||
app.use(prefix, cors());
|
app.use(prefix, cors());
|
||||||
}
|
}
|
||||||
|
|
||||||
app.use(prefix, serve(maps, config[prefix], prefix));
|
if (config[prefix].style) {
|
||||||
|
app.use(prefix, serve_raster(maps, config[prefix], prefix));
|
||||||
|
} else {
|
||||||
|
app.use(prefix, serve_vector(maps, config[prefix], prefix));
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// serve index.html on the root
|
// serve index.html on the root
|
||||||
|
@ -50,16 +55,9 @@ module.exports = function(opts, callback) {
|
||||||
queue.push(function(callback) {
|
queue.push(function(callback) {
|
||||||
var info = clone(map.tileJSON);
|
var info = clone(map.tileJSON);
|
||||||
|
|
||||||
var domains = [],
|
info.tiles = utils.getTileUrls(
|
||||||
tilePath = '/{z}/{x}/{y}.{format}';
|
req.protocol, config[prefix].domains, req.headers.host,
|
||||||
|
prefix, '/{z}/{x}/{y}.{format}', info.format, req.query.key);
|
||||||
if (config[prefix].domains && config[prefix].domains.length > 0) {
|
|
||||||
domains = config[prefix].domains.split(',');
|
|
||||||
}
|
|
||||||
|
|
||||||
info.tiles = utils.getTileUrls(req.protocol, domains, req.headers.host,
|
|
||||||
prefix, tilePath, info.format,
|
|
||||||
req.query.key);
|
|
||||||
|
|
||||||
callback(null, info);
|
callback(null, info);
|
||||||
});
|
});
|
||||||
|
|
11
src/utils.js
11
src/utils.js
|
@ -2,7 +2,16 @@
|
||||||
|
|
||||||
module.exports.getTileUrls = function(
|
module.exports.getTileUrls = function(
|
||||||
protocol, domains, host, path, tilePath, format, key) {
|
protocol, domains, host, path, tilePath, format, key) {
|
||||||
domains = domains && domains.length > 0 ? domains : [host];
|
|
||||||
|
if (domains) {
|
||||||
|
if (domains.constructor === String && domains.length > 0) {
|
||||||
|
domains = domains.split(',');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!domains || domains.length == 0) {
|
||||||
|
domains = [host];
|
||||||
|
}
|
||||||
|
|
||||||
var query = (key && key.length > 0) ? ('?key=' + key) : '';
|
var query = (key && key.length > 0) ? ('?key=' + key) : '';
|
||||||
if (path == '/') {
|
if (path == '/') {
|
||||||
path = '';
|
path = '';
|
||||||
|
|
Loading…
Reference in a new issue