#include #include #include #include #include #include #include #include #include #include #include #include #include "jsonpull/jsonpull.h" #include "dirtiles.hpp" std::string dir_read_tile(std::string base, struct zxy tile) { std::ifstream pbfFile(base + "/" + tile.path(), std::ios::in | std::ios::binary); std::ostringstream contents; contents << pbfFile.rdbuf(); pbfFile.close(); return (contents.str()); } void dir_write_tile(const char *outdir, int z, int tx, int ty, std::string const &pbf) { mkdir(outdir, S_IRWXU | S_IRWXG | S_IRWXO); std::string curdir(outdir); std::string slash("/"); std::string newdir = curdir + slash + std::to_string(z); mkdir(newdir.c_str(), S_IRWXU | S_IRWXG | S_IRWXO); newdir = newdir + "/" + std::to_string(tx); mkdir(newdir.c_str(), S_IRWXU | S_IRWXG | S_IRWXO); newdir = newdir + "/" + std::to_string(ty) + ".pbf"; struct stat st; if (stat(newdir.c_str(), &st) == 0) { fprintf(stderr, "Can't write tile to already existing %s\n", newdir.c_str()); exit(EXIT_FAILURE); } std::ofstream pbfFile(newdir, std::ios::out | std::ios::binary); pbfFile.write(pbf.data(), pbf.size()); pbfFile.close(); } static bool numeric(const char *s) { if (*s == '\0') { return false; } for (; *s != 0; s++) { if (*s < '0' || *s > '9') { return false; } } return true; } static bool pbfname(const char *s) { while (*s >= '0' && *s <= '9') { s++; } return strcmp(s, ".pbf") == 0 || strcmp(s, ".mvt") == 0; } void check_dir(const char *dir, char **argv, bool force, bool forcetable) { struct stat st; mkdir(dir, S_IRWXU | S_IRWXG | S_IRWXO); std::string meta = std::string(dir) + "/" + "metadata.json"; if (force) { unlink(meta.c_str()); // error OK since it may not exist; } else { if (stat(meta.c_str(), &st) == 0) { fprintf(stderr, "%s: Tileset \"%s\" already exists. You can use --force if you want to delete the old tileset.\n", argv[0], dir); fprintf(stderr, "%s: %s: file exists\n", argv[0], meta.c_str()); if (!forcetable) { exit(EXIT_FAILURE); } } } if (forcetable) { // Don't clear existing tiles return; } std::vector tiles = enumerate_dirtiles(dir, INT_MIN, INT_MAX); for (size_t i = 0; i < tiles.size(); i++) { std::string fn = std::string(dir) + "/" + tiles[i].path(); if (force) { if (unlink(fn.c_str()) != 0) { perror(fn.c_str()); exit(EXIT_FAILURE); } } else { fprintf(stderr, "%s: file exists\n", fn.c_str()); exit(EXIT_FAILURE); } } } std::vector enumerate_dirtiles(const char *fname, int minzoom, int maxzoom) { std::vector tiles; DIR *d1 = opendir(fname); if (d1 != NULL) { struct dirent *dp; while ((dp = readdir(d1)) != NULL) { if (numeric(dp->d_name) && atoi(dp->d_name) >= minzoom && atoi(dp->d_name) <= maxzoom) { std::string z = std::string(fname) + "/" + dp->d_name; int tz = atoi(dp->d_name); DIR *d2 = opendir(z.c_str()); if (d2 == NULL) { perror(z.c_str()); exit(EXIT_FAILURE); } struct dirent *dp2; while ((dp2 = readdir(d2)) != NULL) { if (numeric(dp2->d_name)) { std::string x = z + "/" + dp2->d_name; int tx = atoi(dp2->d_name); DIR *d3 = opendir(x.c_str()); if (d3 == NULL) { perror(x.c_str()); exit(EXIT_FAILURE); } struct dirent *dp3; while ((dp3 = readdir(d3)) != NULL) { if (pbfname(dp3->d_name)) { int ty = atoi(dp3->d_name); zxy tile(tz, tx, ty); if (strstr(dp3->d_name, ".mvt") != NULL) { tile.extension = ".mvt"; } tiles.push_back(tile); } } closedir(d3); } } closedir(d2); } } closedir(d1); } std::sort(tiles.begin(), tiles.end()); return tiles; } sqlite3 *dirmeta2tmp(const char *fname) { sqlite3 *db; char *err = NULL; if (sqlite3_open("", &db) != SQLITE_OK) { fprintf(stderr, "Temporary db: %s\n", sqlite3_errmsg(db)); exit(EXIT_FAILURE); } if (sqlite3_exec(db, "CREATE TABLE metadata (name text, value text);", NULL, NULL, &err) != SQLITE_OK) { fprintf(stderr, "Create metadata table: %s\n", err); exit(EXIT_FAILURE); } std::string name = fname; name += "/metadata.json"; FILE *f = fopen(name.c_str(), "r"); if (f == NULL) { perror(name.c_str()); } else { json_pull *jp = json_begin_file(f); json_object *o = json_read_tree(jp); if (o == NULL) { fprintf(stderr, "%s: metadata parsing error: %s\n", name.c_str(), jp->error); exit(EXIT_FAILURE); } if (o->type != JSON_HASH) { fprintf(stderr, "%s: bad metadata format\n", name.c_str()); exit(EXIT_FAILURE); } for (size_t i = 0; i < o->length; i++) { if (o->keys[i]->type != JSON_STRING || o->values[i]->type != JSON_STRING) { fprintf(stderr, "%s: non-string in metadata\n", name.c_str()); } char *sql = sqlite3_mprintf("INSERT INTO metadata (name, value) VALUES (%Q, %Q);", o->keys[i]->string, o->values[i]->string); if (sqlite3_exec(db, sql, NULL, NULL, &err) != SQLITE_OK) { fprintf(stderr, "set %s in metadata: %s\n", o->keys[i]->string, err); } sqlite3_free(sql); } json_end(jp); fclose(f); } return db; }