#include #include #include #include #include #include #include #include "jsonpull/jsonpull.h" #include "geometry.hpp" #include "projection.hpp" #include "read_json.hpp" #include "text.hpp" #include "mvt.hpp" #include "milo/dtoa_milo.h" const char *geometry_names[GEOM_TYPES] = { "Point", "MultiPoint", "LineString", "MultiLineString", "Polygon", "MultiPolygon", }; int geometry_within[GEOM_TYPES] = { -1, /* point */ GEOM_POINT, /* multipoint */ GEOM_POINT, /* linestring */ GEOM_LINESTRING, /* multilinestring */ GEOM_LINESTRING, /* polygon */ GEOM_POLYGON, /* multipolygon */ }; int mb_geometry[GEOM_TYPES] = { VT_POINT, VT_POINT, VT_LINE, VT_LINE, VT_POLYGON, VT_POLYGON, }; void json_context(json_object *j) { char *s = json_stringify(j); if (strlen(s) >= 500) { sprintf(s + 497, "..."); } fprintf(stderr, "In JSON object %s\n", s); free(s); // stringify } void parse_geometry(int t, json_object *j, drawvec &out, int op, const char *fname, int line, json_object *feature) { if (j == NULL || j->type != JSON_ARRAY) { fprintf(stderr, "%s:%d: expected array for type %d\n", fname, line, t); json_context(feature); return; } int within = geometry_within[t]; if (within >= 0) { size_t i; for (i = 0; i < j->length; i++) { if (within == GEOM_POINT) { if (i == 0 || mb_geometry[t] == GEOM_MULTIPOINT) { op = VT_MOVETO; } else { op = VT_LINETO; } } parse_geometry(within, j->array[i], out, op, fname, line, feature); } } else { if (j->length >= 2 && j->array[0]->type == JSON_NUMBER && j->array[1]->type == JSON_NUMBER) { long long x, y; double lon = j->array[0]->number; double lat = j->array[1]->number; projection->project(lon, lat, 32, &x, &y); if (j->length > 2) { static int warned = 0; if (!warned) { fprintf(stderr, "%s:%d: ignoring dimensions beyond two\n", fname, line); json_context(j); json_context(feature); warned = 1; } } draw d(op, x, y); out.push_back(draw(op, x, y)); } else { fprintf(stderr, "%s:%d: malformed point\n", fname, line); json_context(j); json_context(feature); } } if (t == GEOM_POLYGON) { // Note that this is not using the correct meaning of closepath. // // We are using it here to close an entire Polygon, to distinguish // the Polygons within a MultiPolygon from each other. // // This will be undone in fix_polygon(), which needs to know which // rings come from which Polygons so that it can make the winding order // of the outer ring be the opposite of the order of the inner rings. out.push_back(draw(VT_CLOSEPATH, 0, 0)); } } void canonicalize(json_object *o) { if (o->type == JSON_NUMBER) { std::string s; long long v; unsigned long long uv; if (is_integer(o->string, &v)) { s = std::to_string(v); } else if (is_unsigned_integer(o->string, &uv)) { s = std::to_string(uv); } else { s = milo::dtoa_milo(o->number); } free(o->string); o->string = strdup(s.c_str()); } else if (o->type == JSON_HASH) { for (size_t i = 0; i < o->length; i++) { canonicalize(o->values[i]); } } else if (o->type == JSON_ARRAY) { for (size_t i = 0; i < o->length; i++) { canonicalize(o->array[i]); } } } void stringify_value(json_object *value, int &type, std::string &stringified, const char *reading, int line, json_object *feature) { if (value != NULL) { int vt = value->type; std::string val; if (vt == JSON_STRING || vt == JSON_NUMBER) { val = value->string; } else if (vt == JSON_TRUE) { val = "true"; } else if (vt == JSON_FALSE) { val = "false"; } else if (vt == JSON_NULL) { val = "null"; } else { canonicalize(value); const char *v = json_stringify(value); val = std::string(v); free((void *) v); // stringify } if (vt == JSON_STRING) { type = mvt_string; stringified = val; std::string err = check_utf8(val); if (err != "") { fprintf(stderr, "%s:%d: %s\n", reading, line, err.c_str()); json_context(feature); exit(EXIT_FAILURE); } } else if (vt == JSON_NUMBER) { type = mvt_double; long long v; unsigned long long uv; if (is_integer(value->string, &v)) { stringified = std::to_string(v); } else if (is_unsigned_integer(value->string, &uv)) { stringified = std::to_string(uv); } else { stringified = milo::dtoa_milo(value->number); } } else if (vt == JSON_TRUE || vt == JSON_FALSE) { type = mvt_bool; stringified = val; } else if (vt == JSON_NULL) { type = mvt_null; stringified = "null"; } else { type = mvt_string; stringified = val; } } }