Coverage for src/couchers/servicers/gis.py: 77%
31 statements
« prev ^ index » next coverage.py v7.6.10, created at 2025-01-22 06:42 +0000
« prev ^ index » next coverage.py v7.6.10, created at 2025-01-22 06:42 +0000
1import json
2import logging
4from sqlalchemy.dialects.postgresql import JSON
5from sqlalchemy.sql import func
7from couchers.materialized_views import clustered_users, lite_users
8from couchers.models import Node, Page, PageType, PageVersion
9from couchers.sql import couchers_select as select
10from proto import gis_pb2_grpc
11from proto.google.api import httpbody_pb2
13logger = logging.getLogger(__name__)
16def _build_geojson_select(statement):
17 """
18 See usages below.
19 """
20 # this is basically a translation of the postgis ST_AsGeoJSON example into sqlalchemy/geoalchemy2
21 return func.json_build_object(
22 "type",
23 "FeatureCollection",
24 "features",
25 func.json_agg(func.ST_AsGeoJSON(statement.subquery(), maxdecimaldigits=5).cast(JSON)),
26 )
29def _statement_to_geojson_response(session, statement):
30 json_dict = session.execute(select(_build_geojson_select(statement))).scalar_one_or_none()
31 return httpbody_pb2.HttpBody(
32 content_type="application/json",
33 # json.dumps escapes non-ascii characters
34 data=json.dumps(json_dict).encode("ascii"),
35 )
38class GIS(gis_pb2_grpc.GISServicer):
39 def GetUsers(self, request, context, session):
40 statement = (
41 select(lite_users.c.id, lite_users.c.geom, lite_users.c.has_completed_profile)
42 .where_users_visible(context, table=lite_users.c)
43 .where(lite_users.c.geom != None)
44 )
45 return _statement_to_geojson_response(session, statement)
47 def GetClusteredUsers(self, request, context, session):
48 return _statement_to_geojson_response(session, select(clustered_users.c.geom, clustered_users.c.count))
50 def GetCommunities(self, request, context, session):
51 return _statement_to_geojson_response(session, select(Node).where(Node.geom != None))
53 def GetPlaces(self, request, context, session):
54 # need to do a subquery here so we get pages without a geom, not just versions without geom
55 latest_pages = (
56 select(func.max(PageVersion.id).label("id"))
57 .join(Page, Page.id == PageVersion.page_id)
58 .where(Page.type == PageType.place)
59 .group_by(PageVersion.page_id)
60 .subquery()
61 )
63 statement = (
64 select(PageVersion.page_id.label("id"), PageVersion.slug.label("slug"), PageVersion.geom)
65 .join(latest_pages, latest_pages.c.id == PageVersion.id)
66 .where(PageVersion.geom != None)
67 )
69 return _statement_to_geojson_response(session, statement)
71 def GetGuides(self, request, context, session):
72 latest_pages = (
73 select(func.max(PageVersion.id).label("id"))
74 .join(Page, Page.id == PageVersion.page_id)
75 .where(Page.type == PageType.guide)
76 .group_by(PageVersion.page_id)
77 .subquery()
78 )
80 statement = (
81 select(PageVersion.page_id.label("id"), PageVersion.slug.label("slug"), PageVersion.geom)
82 .join(latest_pages, latest_pages.c.id == PageVersion.id)
83 .where(PageVersion.geom != None)
84 )
86 return _statement_to_geojson_response(session, statement)