
Adds pmtiles support to TileServer-GL Signed-off-by: Andrew Calcutt <acalcutt@techidiots.net> Signed-off-by: Michael Nutt <michael@nuttnet.net> Co-authored-by: Michael Nutt <michael@nuttnet.net>
151 lines
3.2 KiB
JavaScript
151 lines
3.2 KiB
JavaScript
import fs from 'node:fs';
|
|
import PMTiles from 'pmtiles';
|
|
import { isValidHttpUrl } from './utils.js';
|
|
|
|
class PMTilesFileSource {
|
|
constructor(fd) {
|
|
this.fd = fd;
|
|
}
|
|
getKey() {
|
|
return this.fd;
|
|
}
|
|
async getBytes(offset, length) {
|
|
const buffer = Buffer.alloc(length);
|
|
await ReadFileBytes(this.fd, buffer, offset);
|
|
const ab = buffer.buffer.slice(
|
|
buffer.byteOffset,
|
|
buffer.byteOffset + buffer.byteLength,
|
|
);
|
|
return { data: ab };
|
|
}
|
|
}
|
|
|
|
/**
|
|
*
|
|
* @param fd
|
|
* @param buffer
|
|
* @param offset
|
|
*/
|
|
async function ReadFileBytes(fd, buffer, offset) {
|
|
return new Promise((resolve, reject) => {
|
|
fs.read(fd, buffer, 0, buffer.length, offset, (err) => {
|
|
if (err) {
|
|
return reject(err);
|
|
}
|
|
resolve();
|
|
});
|
|
});
|
|
}
|
|
|
|
/**
|
|
*
|
|
* @param FilePath
|
|
*/
|
|
export function PMtilesOpen(FilePath) {
|
|
let pmtiles = undefined;
|
|
|
|
if (isValidHttpUrl(FilePath)) {
|
|
const source = new PMTiles.FetchSource(FilePath);
|
|
pmtiles = new PMTiles.PMTiles(source);
|
|
} else {
|
|
const fd = fs.openSync(FilePath, 'r');
|
|
const source = new PMTilesFileSource(fd);
|
|
pmtiles = new PMTiles.PMTiles(source);
|
|
}
|
|
return pmtiles;
|
|
}
|
|
|
|
/**
|
|
*
|
|
* @param pmtiles
|
|
*/
|
|
export async function GetPMtilesInfo(pmtiles) {
|
|
const header = await pmtiles.getHeader();
|
|
const metadata = await pmtiles.getMetadata();
|
|
|
|
//Add missing metadata from header
|
|
metadata['format'] = GetPmtilesTileType(header.tileType).type;
|
|
metadata['minzoom'] = header.minZoom;
|
|
metadata['maxzoom'] = header.maxZoom;
|
|
|
|
if (header.minLon && header.minLat && header.maxLon && header.maxLat) {
|
|
metadata['bounds'] = [
|
|
header.minLon,
|
|
header.minLat,
|
|
header.maxLon,
|
|
header.maxLat,
|
|
];
|
|
} else {
|
|
metadata['bounds'] = [-180, -85.05112877980659, 180, 85.0511287798066];
|
|
}
|
|
|
|
if (header.centerZoom) {
|
|
metadata['center'] = [
|
|
header.centerLon,
|
|
header.centerLat,
|
|
header.centerZoom,
|
|
];
|
|
} else {
|
|
metadata['center'] = [
|
|
header.centerLon,
|
|
header.centerLat,
|
|
parseInt(metadata['maxzoom']) / 2,
|
|
];
|
|
}
|
|
|
|
return metadata;
|
|
}
|
|
|
|
/**
|
|
*
|
|
* @param pmtiles
|
|
* @param z
|
|
* @param x
|
|
* @param y
|
|
*/
|
|
export async function GetPMtilesTile(pmtiles, z, x, y) {
|
|
const header = await pmtiles.getHeader();
|
|
const TileType = GetPmtilesTileType(header.tileType);
|
|
let zxyTile = await pmtiles.getZxy(z, x, y);
|
|
if (zxyTile && zxyTile.data) {
|
|
zxyTile = Buffer.from(zxyTile.data);
|
|
} else {
|
|
zxyTile = undefined;
|
|
}
|
|
return { data: zxyTile, header: TileType.header };
|
|
}
|
|
|
|
/**
|
|
*
|
|
* @param typenum
|
|
*/
|
|
function GetPmtilesTileType(typenum) {
|
|
let head = {};
|
|
let tileType;
|
|
switch (typenum) {
|
|
case 0:
|
|
tileType = 'Unknown';
|
|
break;
|
|
case 1:
|
|
tileType = 'pbf';
|
|
head['Content-Type'] = 'application/x-protobuf';
|
|
break;
|
|
case 2:
|
|
tileType = 'png';
|
|
head['Content-Type'] = 'image/png';
|
|
break;
|
|
case 3:
|
|
tileType = 'jpeg';
|
|
head['Content-Type'] = 'image/jpeg';
|
|
break;
|
|
case 4:
|
|
tileType = 'webp';
|
|
head['Content-Type'] = 'image/webp';
|
|
break;
|
|
case 5:
|
|
tileType = 'avif';
|
|
head['Content-Type'] = 'image/avif';
|
|
break;
|
|
}
|
|
return { type: tileType, header: head };
|
|
}
|