From 2e1de3a6f801c7e400bb3b27634e0582d45e503c Mon Sep 17 00:00:00 2001 From: "S. Andrew Sheppard" Date: Fri, 9 Feb 2024 21:05:14 +0000 Subject: [PATCH] built-in vector tile server via ST_AsMVT() --- tests/test_config.py | 1 + tests/test_vector_tile.py | 27 ++++ wq/db/rest/maps.py | 257 ++++++++++++++++++++++++++++++++++++++ wq/db/rest/routers.py | 27 +++- 4 files changed, 310 insertions(+), 2 deletions(-) create mode 100644 tests/test_vector_tile.py create mode 100644 wq/db/rest/maps.py diff --git a/tests/test_config.py b/tests/test_config.py index f547180..2d25307 100644 --- a/tests/test_config.py +++ b/tests/test_config.py @@ -38,6 +38,7 @@ def test_rest_config_meta(self): "list": True, "cache": "none", "form": conf["form"], + "geometry_fields": [], "label_template": "{{name}}", "verbose_name": "item", "verbose_name_plural": "items", diff --git a/tests/test_vector_tile.py b/tests/test_vector_tile.py new file mode 100644 index 0000000..46fda3f --- /dev/null +++ b/tests/test_vector_tile.py @@ -0,0 +1,27 @@ +import unittest +from .base import APITestCase +from django.conf import settings + + +class VectorTileTestCase(APITestCase): + @unittest.skipIf(settings.VARIANT == "postgis", "postgis supports tiles") + def test_no_tiles(self): + tile = self.client.get("/tiles/0/0/0.pbf") + self.assertEqual( + tile.content.decode(), + "Tile server not supported with this database engine", + ) + + @unittest.skipUnless(settings.VARIANT == "postgis", "requires postgis") + def test_tiles(self): + from .gis_app.models import PointModel + + empty_tile = self.client.get("/tiles/0/0/0.pbf").content + self.assertEqual(b"", empty_tile) + + PointModel.objects.create(pk=1, geometry="POINT(34 -84)") + single_point = self.client.get("/tiles/0/0/0.pbf").content + self.assertEqual( + single_point, + b'\x1a\x1e\n\npointmodel\x12\x0b\x08\x01\x18\x01"\x05\t\x86&\x84>(\x80 x\x02', + ) diff --git a/wq/db/rest/maps.py b/wq/db/rest/maps.py new file mode 100644 index 0000000..6822d11 --- /dev/null +++ b/wq/db/rest/maps.py @@ -0,0 +1,257 @@ +from django.db.models import Func, Value +from django.http import HttpResponse +from django.contrib.gis.db.models import GeometryField +from django.contrib.gis.db.models.functions import Transform +from django.db import connection +from django.conf import settings + + +def supports_vector_tiles(): + backend = settings.DATABASES.get("default", {}).get("ENGINE") + + if backend == "django.contrib.gis.db.backends.postgis": + return True + else: + return False + + +def tiles(router, request, z, x, y): + if not supports_vector_tiles(): + return HttpResponse( + "Tile server not supported with this database engine.", + content_type="text/plain", + status=404, + ) + tiledata = [] + envelope = TileEnvelope(z, x, y) + cursor = connection.cursor() + for name, conf in router.config["pages"].items(): + if not conf.get("list") or not conf.get("geometry_fields"): + continue + + fields = ["id"] + if lookup := conf.get("lookup"): + fields.append(lookup) + fields += list(conf.get("vector_tile_fields") or []) + + queryset = router.get_queryset_for_model(name, request) + for field in conf["geometry_fields"]: + model_info = (name, queryset, field["name"], fields) + tiledata.append(get_tile_data(model_info, envelope, cursor)) + + return HttpResponse( + b"".join(tiledata), + content_type="application/vnd.mapbox-vector-tile", + ) + + +def get_tile_data(model_info, envelope, cursor): + name, queryset, geometry_field, fields = model_info + geometry_mvt = f"{geometry_field}_mvt" + fields = fields + [geometry_mvt] + tile = ( + queryset.filter(geometry__bboverlaps=envelope) + .annotate(**{geometry_mvt: TileGeom(geometry_field, envelope)}) + .values(*fields) + ) + sql, params = tile.query.sql_with_params() + sql = sql.replace(f'::bytea AS "{geometry_mvt}"', f' AS "{geometry_mvt}"') + cursor.execute( + f"SELECT ST_AsMVT(tile.*, '{name}', 4096, '{geometry_mvt}', 'id') FROM ({sql}) AS tile", + params, + ) + row = cursor.fetchone() + return row[0] + + +class TileEnvelope(Func): + arity = 3 + function = "ST_TileEnvelope" + output_field = GeometryField(srid=3857) + + def __init__(self, z, x, y): + super().__init__(Value(z), Value(x), Value(y)) + + +class TileGeom(Func): + arity = 2 + function = "ST_AsMVTGeom" + + def __init__(self, geometry, envelope): + geometry = Transform(geometry, srid=3857) + super().__init__(geometry, envelope) + + +def update_tiles_url(plugin_conf, base_url): + if not plugin_conf: + return + if "tiles" in plugin_conf: + return + if not supports_vector_tiles(): + return + plugin_conf["tiles"] = base_url + "/tiles/{z}/{x}/{y}.pbf" + + +def update_geometry_fields(conf): + if conf.get("form") and "geometry_fields" not in conf: + conf["geometry_fields"] = get_geometry_fields(conf["form"]) + + +def update_map_config(conf, pages): + if not conf.get("map"): + return + + if conf.get("map") is True: + if conf.get("list"): + conf["map"] = [ + { + "mode": "list", + "auto_layers": True, + }, + {"mode": "detail", "auto_layers": True}, + {"mode": "edit", "auto_layers": True}, + ] + else: + conf["map"] = { + "mapId": "map", + "auto_layers": True, + } + + maps = conf.get("map") + if isinstance(maps, dict): + maps = [maps] + for map_conf in maps: + if map_conf.get("auto_layers") or map_conf.get("autoLayers"): + mode = map_conf.get("mode") + map_conf.pop("auto_layers", None) + map_conf.pop("autoLayers", None) + layers = map_conf.get("layers") or [] + if mode in ("list", "detail") and not layers: + layers += get_context_layers(conf, mode) + if supports_vector_tiles(): + layers += get_tile_layers(pages, mode) + map_conf["layers"] = layers + + return conf + + +def get_geometry_fields(form, prefix=""): + geometry_fields = [] + for field in form or []: + if field["type"].startswith("geo"): + geometry_fields.append( + { + "name": prefix + field["name"], + "label": field["label"] or field["name"], + "type": field["type"], + } + ) + elif field["type"] == "group": + geometry_fields += get_geometry_fields( + field["children"], + prefix=field["name"] + "." if field["name"] else "", + ) + elif field["type"] == "repeat": + geometry_fields += get_geometry_fields( + field["children"], prefix=field["name"] + "[]." + ) + + return geometry_fields + + +def get_page_label(conf, mode): + if mode == "list": + page_label = conf.get("verbose_name_plural") or conf["name"] + else: + page_label = conf.get("verbose_name") or conf.get["name"] + if page_label == page_label.lower(): + page_label = page_label.title() + return page_label + + +def get_geometry_label(conf, field, mode): + page_label = get_page_label(conf, mode) + if len(conf.get("geometry_fields") or []) > 1: + field_label = field["label"] or field["name"] + else: + field_label = None + + if field_label: + return f"{page_label} - {field_label}" + else: + return page_label + + +def get_context_layers(conf, mode): + if mode == "list": + lookup_fn = "context_feature_collection" + else: + lookup_fn = "context_feature" + + layers = [] + for field in conf.get("geometry_fields") or []: + layer_conf = { + "name": get_geometry_label(conf, field, mode), + "type": "geojson", + "data": [lookup_fn, field["name"]], + "popup": conf["name"], + } + if mode == "list": + layer_conf["cluster"] = True # TODO: implement in @wq/map-gl + if conf.get("map_color"): + layer_conf["color"] = conf["map_color"] + # TODO: layer_conf["legend"] = ... + layers.append(layer_conf) + return layers + + +def get_tile_layers(pages, mode): + layers = [] + for conf in sorted(pages.values(), key=layer_sort_key): + page_name = conf["name"] + fields = conf.get("geometry_fields") + if not fields: + continue + for index, field in enumerate(fields): + field_name = field["name"] + layer_name = ( + page_name if index == 0 else f"{page_name}_{field_name}" + ) + geometry_label = get_geometry_label(conf, field, "list") + if mode: + geometry_label = f"All {geometry_label}" + layer_conf = { + "name": geometry_label, + "type": "vector-tile", + "layer": layer_name, + "popup": page_name, + } + if conf.get("map_icon"): + layer_conf["icon"] = conf["map_icon"] + if mode != "edit": + layer_conf["identifyLayers"] = [layer_name] + else: + if conf.get("map_color"): + layer_conf["color"] = conf["map_color"] + # TODO: layer_conf["legend"] = ... + if mode != "edit": + if field.get("type") == "geopoint": + layer_conf["identifyLayers"] = [f"{layer_name}-circle"] + elif field.get("type") == "geotrace": + layer_conf["identifyLayers"] = [f"{layer_name}-line"] + elif field.get("type") == "geoshape": + layer_conf["identifyLayers"] = [f"{layer_name}-fill"] + + if "map_active" in conf: + layer_conf["active"] = conf["map_active"] + elif mode: + layer_conf["active"] = False + + layers.append(layer_conf) + return layers + + +def layer_sort_key(conf): + order = conf.get("order") or 0 + label = get_page_label(conf, "list") + return order, label diff --git a/wq/db/rest/routers.py b/wq/db/rest/routers.py index e7be682..e053bba 100644 --- a/wq/db/rest/routers.py +++ b/wq/db/rest/routers.py @@ -1,5 +1,5 @@ from django.utils.encoding import force_str -from django.urls import re_path +from django.urls import re_path, path from django.conf import settings from rest_framework.routers import DefaultRouter, Route @@ -11,6 +11,12 @@ from .views import SimpleViewSet, ModelViewSet from .renderers import JSONRenderer, ESMRenderer from .exceptions import ImproperlyConfigured +from .maps import ( + tiles, + update_tiles_url, + update_geometry_fields, + update_map_config, +) class ModelRouter(DefaultRouter): @@ -32,6 +38,7 @@ class ModelRouter(DefaultRouter): include_root_view = False include_config_view = True include_multi_view = True + include_tiles_view = True default_serializer_class = None @@ -321,7 +328,7 @@ def base_config(self): pages = {} for page in self._extra_pages: conf, view = self.get_page(page) - pages[page] = conf + pages[page] = conf.copy() for model in self._models: if not has_perm(user, model, "view"): continue @@ -353,9 +360,16 @@ def base_config(self): "name": "index", "show_in_index": False, "verbose_name": site_title, + "map": True, }, ) + for conf in pages.values(): + update_geometry_fields(conf) + + for conf in pages.values(): + update_map_config(conf, pages) + self._base_config = { "pages": pages, "site_title": site_title, @@ -367,6 +381,9 @@ def base_config(self): } if getattr(settings, "WQ_CONFIG", None): self._base_config.update(settings.WQ_CONFIG) + + update_tiles_url(self._base_config.get("map"), self.get_base_url()) + if settings.DEBUG: self._base_config["debug"] = True self._base_config.update(self._extra_config) @@ -484,6 +501,9 @@ def get_multi(self, request, urls): result[listurl] = self.paginate(model, 1, request) return Response(result) + def tiles(self, request, z, x, y): + return tiles(self, request, z, x, y) + def get_urls(self): # Register viewsets with DefaultRouter just before returning urls @@ -526,6 +546,9 @@ def register(config, viewset): # / - Skip registration and directly generate custom URLs urls.extend(self.get_root_urls(root["view"], root["name"])) + if self.include_tiles_view: + urls.append(path("tiles///.pbf", self.tiles)) + return urls def get_root_urls(self, viewset, name):