2014-09-29 19:48:58 +00:00
|
|
|
// for vasprintf() on Linux
|
2016-04-27 22:12:03 +00:00
|
|
|
#ifndef _GNU_SOURCE
|
2014-09-29 19:48:58 +00:00
|
|
|
#define _GNU_SOURCE
|
2016-04-27 22:12:03 +00:00
|
|
|
#endif
|
2014-09-29 19:48:58 +00:00
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <sqlite3.h>
|
2016-04-27 22:09:06 +00:00
|
|
|
#include <vector>
|
2016-04-28 18:56:30 +00:00
|
|
|
#include <string>
|
2016-04-28 21:43:04 +00:00
|
|
|
#include <set>
|
|
|
|
#include "main.hpp"
|
2016-04-27 21:00:14 +00:00
|
|
|
#include "pool.hpp"
|
|
|
|
#include "mbtiles.hpp"
|
2016-04-27 22:09:06 +00:00
|
|
|
#include "geometry.hpp"
|
2014-09-29 19:48:58 +00:00
|
|
|
|
2016-01-28 22:18:31 +00:00
|
|
|
sqlite3 *mbtiles_open(char *dbname, char **argv, int forcetable) {
|
2014-09-29 19:48:58 +00:00
|
|
|
sqlite3 *outdb;
|
|
|
|
|
|
|
|
if (sqlite3_open(dbname, &outdb) != SQLITE_OK) {
|
2015-06-03 18:21:40 +00:00
|
|
|
fprintf(stderr, "%s: %s: %s\n", argv[0], dbname, sqlite3_errmsg(outdb));
|
2014-09-29 19:48:58 +00:00
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
char *err = NULL;
|
|
|
|
if (sqlite3_exec(outdb, "PRAGMA synchronous=0", NULL, NULL, &err) != SQLITE_OK) {
|
|
|
|
fprintf(stderr, "%s: async: %s\n", argv[0], err);
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
if (sqlite3_exec(outdb, "PRAGMA locking_mode=EXCLUSIVE", NULL, NULL, &err) != SQLITE_OK) {
|
|
|
|
fprintf(stderr, "%s: async: %s\n", argv[0], err);
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
if (sqlite3_exec(outdb, "PRAGMA journal_mode=DELETE", NULL, NULL, &err) != SQLITE_OK) {
|
|
|
|
fprintf(stderr, "%s: async: %s\n", argv[0], err);
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
if (sqlite3_exec(outdb, "CREATE TABLE metadata (name text, value text);", NULL, NULL, &err) != SQLITE_OK) {
|
|
|
|
fprintf(stderr, "%s: create metadata table: %s\n", argv[0], err);
|
2016-01-28 22:18:31 +00:00
|
|
|
if (!forcetable) {
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2014-09-29 19:48:58 +00:00
|
|
|
}
|
|
|
|
if (sqlite3_exec(outdb, "CREATE TABLE tiles (zoom_level integer, tile_column integer, tile_row integer, tile_data blob);", NULL, NULL, &err) != SQLITE_OK) {
|
|
|
|
fprintf(stderr, "%s: create tiles table: %s\n", argv[0], err);
|
2016-01-28 22:18:31 +00:00
|
|
|
if (!forcetable) {
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2014-09-29 19:48:58 +00:00
|
|
|
}
|
|
|
|
if (sqlite3_exec(outdb, "create unique index name on metadata (name);", NULL, NULL, &err) != SQLITE_OK) {
|
|
|
|
fprintf(stderr, "%s: index metadata: %s\n", argv[0], err);
|
2016-01-28 22:18:31 +00:00
|
|
|
if (!forcetable) {
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2014-09-29 19:48:58 +00:00
|
|
|
}
|
|
|
|
if (sqlite3_exec(outdb, "create unique index tile_index on tiles (zoom_level, tile_column, tile_row);", NULL, NULL, &err) != SQLITE_OK) {
|
|
|
|
fprintf(stderr, "%s: index tiles: %s\n", argv[0], err);
|
2016-01-28 22:18:31 +00:00
|
|
|
if (!forcetable) {
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2014-09-29 19:48:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return outdb;
|
|
|
|
}
|
|
|
|
|
|
|
|
void mbtiles_write_tile(sqlite3 *outdb, int z, int tx, int ty, const char *data, int size) {
|
|
|
|
sqlite3_stmt *stmt;
|
|
|
|
const char *query = "insert into tiles (zoom_level, tile_column, tile_row, tile_data) values (?, ?, ?, ?)";
|
|
|
|
if (sqlite3_prepare_v2(outdb, query, -1, &stmt, NULL) != SQLITE_OK) {
|
|
|
|
fprintf(stderr, "sqlite3 insert prep failed\n");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
sqlite3_bind_int(stmt, 1, z);
|
|
|
|
sqlite3_bind_int(stmt, 2, tx);
|
|
|
|
sqlite3_bind_int(stmt, 3, (1 << z) - 1 - ty);
|
|
|
|
sqlite3_bind_blob(stmt, 4, data, size, NULL);
|
|
|
|
|
|
|
|
if (sqlite3_step(stmt) != SQLITE_DONE) {
|
|
|
|
fprintf(stderr, "sqlite3 insert failed: %s\n", sqlite3_errmsg(outdb));
|
|
|
|
}
|
|
|
|
if (sqlite3_finalize(stmt) != SQLITE_OK) {
|
|
|
|
fprintf(stderr, "sqlite3 finalize failed: %s\n", sqlite3_errmsg(outdb));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-28 18:56:30 +00:00
|
|
|
static void quote(std::string *buf, const char *s) {
|
2014-09-29 19:48:58 +00:00
|
|
|
char tmp[strlen(s) * 8 + 1];
|
|
|
|
char *out = tmp;
|
|
|
|
|
|
|
|
for (; *s != '\0'; s++) {
|
2015-06-21 18:18:14 +00:00
|
|
|
unsigned char ch = (unsigned char) *s;
|
|
|
|
|
|
|
|
if (ch == '\\' || ch == '\"') {
|
2014-09-29 19:48:58 +00:00
|
|
|
*out++ = '\\';
|
2015-06-21 18:18:14 +00:00
|
|
|
*out++ = ch;
|
|
|
|
} else if (ch < ' ') {
|
|
|
|
sprintf(out, "\\u%04x", ch);
|
2014-09-29 19:48:58 +00:00
|
|
|
out = out + strlen(out);
|
|
|
|
} else {
|
2015-06-21 18:18:14 +00:00
|
|
|
*out++ = ch;
|
2014-09-29 19:48:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*out = '\0';
|
2016-04-28 19:46:40 +00:00
|
|
|
buf->append(tmp, strlen(tmp));
|
2014-09-29 19:48:58 +00:00
|
|
|
}
|
|
|
|
|
2016-05-03 18:14:09 +00:00
|
|
|
void aprintf(std::string *buf, const char *format, ...) {
|
2014-09-29 19:48:58 +00:00
|
|
|
va_list ap;
|
|
|
|
char *tmp;
|
|
|
|
|
|
|
|
va_start(ap, format);
|
|
|
|
if (vasprintf(&tmp, format, ap) < 0) {
|
|
|
|
fprintf(stderr, "memory allocation failure\n");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
va_end(ap);
|
|
|
|
|
2016-04-28 19:46:40 +00:00
|
|
|
buf->append(tmp, strlen(tmp));
|
2014-09-29 19:48:58 +00:00
|
|
|
free(tmp);
|
|
|
|
}
|
|
|
|
|
2016-04-28 21:43:04 +00:00
|
|
|
bool type_and_string::operator<(const type_and_string &o) const {
|
|
|
|
if (string < o.string) {
|
|
|
|
return true;
|
2016-01-28 23:46:58 +00:00
|
|
|
}
|
2016-04-28 21:43:04 +00:00
|
|
|
if (string == o.string && type < o.type) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
2016-01-28 23:46:58 +00:00
|
|
|
}
|
|
|
|
|
2016-05-03 17:52:49 +00:00
|
|
|
void mbtiles_write_metadata(sqlite3 *outdb, const char *fname, std::vector<std::string> &layername, int minzoom, int maxzoom, double minlat, double minlon, double maxlat, double maxlon, double midlat, double midlon, std::vector<std::set<type_and_string> > &file_keys, int nlayers, int forcetable, const char *attribution) {
|
2014-09-29 19:48:58 +00:00
|
|
|
char *sql, *err;
|
|
|
|
|
|
|
|
sql = sqlite3_mprintf("INSERT INTO metadata (name, value) VALUES ('name', %Q);", fname);
|
|
|
|
if (sqlite3_exec(outdb, sql, NULL, NULL, &err) != SQLITE_OK) {
|
|
|
|
fprintf(stderr, "set name in metadata: %s\n", err);
|
2016-01-28 22:18:31 +00:00
|
|
|
if (!forcetable) {
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2014-09-29 19:48:58 +00:00
|
|
|
}
|
|
|
|
sqlite3_free(sql);
|
|
|
|
|
|
|
|
sql = sqlite3_mprintf("INSERT INTO metadata (name, value) VALUES ('description', %Q);", fname);
|
|
|
|
if (sqlite3_exec(outdb, sql, NULL, NULL, &err) != SQLITE_OK) {
|
|
|
|
fprintf(stderr, "set description in metadata: %s\n", err);
|
2016-01-28 22:18:31 +00:00
|
|
|
if (!forcetable) {
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2014-09-29 19:48:58 +00:00
|
|
|
}
|
|
|
|
sqlite3_free(sql);
|
|
|
|
|
2016-02-11 19:09:05 +00:00
|
|
|
sql = sqlite3_mprintf("INSERT INTO metadata (name, value) VALUES ('version', %d);", 2);
|
2014-09-29 19:48:58 +00:00
|
|
|
if (sqlite3_exec(outdb, sql, NULL, NULL, &err) != SQLITE_OK) {
|
2016-01-28 22:18:31 +00:00
|
|
|
fprintf(stderr, "set version : %s\n", err);
|
|
|
|
if (!forcetable) {
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2014-09-29 19:48:58 +00:00
|
|
|
}
|
|
|
|
sqlite3_free(sql);
|
|
|
|
|
|
|
|
sql = sqlite3_mprintf("INSERT INTO metadata (name, value) VALUES ('minzoom', %d);", minzoom);
|
|
|
|
if (sqlite3_exec(outdb, sql, NULL, NULL, &err) != SQLITE_OK) {
|
2016-01-28 22:18:31 +00:00
|
|
|
fprintf(stderr, "set minzoom: %s\n", err);
|
|
|
|
if (!forcetable) {
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2014-09-29 19:48:58 +00:00
|
|
|
}
|
|
|
|
sqlite3_free(sql);
|
|
|
|
|
|
|
|
sql = sqlite3_mprintf("INSERT INTO metadata (name, value) VALUES ('maxzoom', %d);", maxzoom);
|
|
|
|
if (sqlite3_exec(outdb, sql, NULL, NULL, &err) != SQLITE_OK) {
|
2016-01-28 22:18:31 +00:00
|
|
|
fprintf(stderr, "set maxzoom: %s\n", err);
|
|
|
|
if (!forcetable) {
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2014-09-29 19:48:58 +00:00
|
|
|
}
|
|
|
|
sqlite3_free(sql);
|
|
|
|
|
|
|
|
sql = sqlite3_mprintf("INSERT INTO metadata (name, value) VALUES ('center', '%f,%f,%d');", midlon, midlat, maxzoom);
|
|
|
|
if (sqlite3_exec(outdb, sql, NULL, NULL, &err) != SQLITE_OK) {
|
2016-01-28 22:18:31 +00:00
|
|
|
fprintf(stderr, "set center: %s\n", err);
|
|
|
|
if (!forcetable) {
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2014-09-29 19:48:58 +00:00
|
|
|
}
|
|
|
|
sqlite3_free(sql);
|
|
|
|
|
|
|
|
sql = sqlite3_mprintf("INSERT INTO metadata (name, value) VALUES ('bounds', '%f,%f,%f,%f');", minlon, minlat, maxlon, maxlat);
|
|
|
|
if (sqlite3_exec(outdb, sql, NULL, NULL, &err) != SQLITE_OK) {
|
2016-01-28 22:18:31 +00:00
|
|
|
fprintf(stderr, "set bounds: %s\n", err);
|
|
|
|
if (!forcetable) {
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2014-09-29 19:48:58 +00:00
|
|
|
}
|
|
|
|
sqlite3_free(sql);
|
|
|
|
|
|
|
|
sql = sqlite3_mprintf("INSERT INTO metadata (name, value) VALUES ('type', %Q);", "overlay");
|
|
|
|
if (sqlite3_exec(outdb, sql, NULL, NULL, &err) != SQLITE_OK) {
|
2016-01-28 22:18:31 +00:00
|
|
|
fprintf(stderr, "set type: %s\n", err);
|
|
|
|
if (!forcetable) {
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2014-09-29 19:48:58 +00:00
|
|
|
}
|
|
|
|
sqlite3_free(sql);
|
|
|
|
|
2016-04-13 19:49:41 +00:00
|
|
|
if (attribution != NULL) {
|
|
|
|
sql = sqlite3_mprintf("INSERT INTO metadata (name, value) VALUES ('attribution', %Q);", attribution);
|
|
|
|
if (sqlite3_exec(outdb, sql, NULL, NULL, &err) != SQLITE_OK) {
|
|
|
|
fprintf(stderr, "set type: %s\n", err);
|
|
|
|
if (!forcetable) {
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
sqlite3_free(sql);
|
|
|
|
}
|
|
|
|
|
2014-09-29 19:48:58 +00:00
|
|
|
sql = sqlite3_mprintf("INSERT INTO metadata (name, value) VALUES ('format', %Q);", "pbf");
|
|
|
|
if (sqlite3_exec(outdb, sql, NULL, NULL, &err) != SQLITE_OK) {
|
2016-01-28 22:18:31 +00:00
|
|
|
fprintf(stderr, "set format: %s\n", err);
|
|
|
|
if (!forcetable) {
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2014-09-29 19:48:58 +00:00
|
|
|
}
|
|
|
|
sqlite3_free(sql);
|
|
|
|
|
2016-04-28 18:56:30 +00:00
|
|
|
std::string buf("{");
|
2015-03-23 23:12:12 +00:00
|
|
|
aprintf(&buf, "\"vector_layers\": [ ");
|
2014-09-29 19:48:58 +00:00
|
|
|
|
2015-03-23 23:12:12 +00:00
|
|
|
int i;
|
|
|
|
for (i = 0; i < nlayers; i++) {
|
|
|
|
if (i != 0) {
|
|
|
|
aprintf(&buf, ", ");
|
2014-09-29 19:48:58 +00:00
|
|
|
}
|
|
|
|
|
2015-03-23 23:12:12 +00:00
|
|
|
aprintf(&buf, "{ \"id\": \"");
|
2016-05-03 17:52:49 +00:00
|
|
|
quote(&buf, layername[i].c_str());
|
2015-03-23 23:12:12 +00:00
|
|
|
aprintf(&buf, "\", \"description\": \"\", \"minzoom\": %d, \"maxzoom\": %d, \"fields\": {", minzoom, maxzoom);
|
|
|
|
|
2016-04-28 21:43:04 +00:00
|
|
|
std::set<type_and_string>::iterator j;
|
|
|
|
bool first = true;
|
|
|
|
for (j = file_keys[i].begin(); j != file_keys[i].end(); ++j) {
|
|
|
|
if (first) {
|
|
|
|
first = false;
|
|
|
|
} else {
|
|
|
|
aprintf(&buf, ", ");
|
|
|
|
}
|
2016-01-28 23:46:58 +00:00
|
|
|
|
2015-03-23 23:12:12 +00:00
|
|
|
aprintf(&buf, "\"");
|
2016-04-28 21:43:04 +00:00
|
|
|
quote(&buf, j->string.c_str());
|
2015-03-23 23:12:12 +00:00
|
|
|
|
2016-04-28 21:43:04 +00:00
|
|
|
if (j->type == VT_NUMBER) {
|
2015-03-23 23:12:12 +00:00
|
|
|
aprintf(&buf, "\": \"Number\"");
|
2016-04-28 21:43:04 +00:00
|
|
|
} else if (j->type == VT_BOOLEAN) {
|
2015-04-10 20:03:11 +00:00
|
|
|
aprintf(&buf, "\": \"Boolean\"");
|
2015-03-23 23:12:12 +00:00
|
|
|
} else {
|
|
|
|
aprintf(&buf, "\": \"String\"");
|
|
|
|
}
|
2014-09-29 19:48:58 +00:00
|
|
|
}
|
2015-03-23 23:12:12 +00:00
|
|
|
|
|
|
|
aprintf(&buf, "} }");
|
2014-09-29 19:48:58 +00:00
|
|
|
}
|
|
|
|
|
2015-03-23 23:12:12 +00:00
|
|
|
aprintf(&buf, " ] }");
|
2014-09-29 19:48:58 +00:00
|
|
|
|
2016-04-28 18:56:30 +00:00
|
|
|
sql = sqlite3_mprintf("INSERT INTO metadata (name, value) VALUES ('json', %Q);", buf.c_str());
|
2014-09-29 19:48:58 +00:00
|
|
|
if (sqlite3_exec(outdb, sql, NULL, NULL, &err) != SQLITE_OK) {
|
2016-01-28 22:18:31 +00:00
|
|
|
fprintf(stderr, "set json: %s\n", err);
|
|
|
|
if (!forcetable) {
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2014-09-29 19:48:58 +00:00
|
|
|
}
|
|
|
|
sqlite3_free(sql);
|
|
|
|
}
|
|
|
|
|
|
|
|
void mbtiles_close(sqlite3 *outdb, char **argv) {
|
|
|
|
char *err;
|
|
|
|
|
|
|
|
if (sqlite3_exec(outdb, "ANALYZE;", NULL, NULL, &err) != SQLITE_OK) {
|
2015-07-08 18:52:22 +00:00
|
|
|
fprintf(stderr, "%s: ANALYZE failed: %s\n", argv[0], err);
|
2014-09-29 19:48:58 +00:00
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
if (sqlite3_close(outdb) != SQLITE_OK) {
|
|
|
|
fprintf(stderr, "%s: could not close database: %s\n", argv[0], sqlite3_errmsg(outdb));
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
}
|