Skip to content
This repository has been archived by the owner on Oct 5, 2024. It is now read-only.

Commit

Permalink
Merge branch 'master' into fix-deb
Browse files Browse the repository at this point in the history
  • Loading branch information
robsavoye authored Jun 19, 2024
2 parents 38d32c2 + b256496 commit d9e5a41
Show file tree
Hide file tree
Showing 86 changed files with 4,997 additions and 1,687 deletions.
2 changes: 1 addition & 1 deletion .github/workflows/tests.yml
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,6 @@ jobs:
compose_service: underpass
compose_command: '"make check -j $(nproc)"'
tag_override: ci
# TODO update postgis image to use github repo var ${{ vars.POSTGIS_TAG }}
# TODO update postgis image to use github repo var ${{ vars.UNDERPASSDB_TAG }}
cache_extra_imgs: |
"docker.io/postgis/postgis:15-3.3-alpine"
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -43,3 +43,4 @@ unconfig.h.in
**/__pycache__/
.vscode
data
data_osm
18 changes: 5 additions & 13 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -9,12 +9,12 @@ It **updates a local copy of the OSM database** in near real-time, and provides

## Demo

We've deployed a rudimentary demo that keeps a database up-to-date for (some country),
We've deployed a basic demo that keeps a database up-to-date for (some country),
rendering buildings and highlighting the ones identified as "un-squared":

[https://underpass.live](https://underpass.live)

<img width="993" alt="Screenshot 2023-11-22 at 10 32 56" src="https://github.com/hotosm/underpass/assets/1226194/73622131-b69f-4716-bf95-9e195efdbba9">
<img width="1160" alt="Screenshot 2024-06-05 at 15 51 57" src="https://github.com/hotosm/underpass/assets/1226194/8a95e518-a12f-45c3-b460-d74d81ca3cfe">

## Getting started

Expand All @@ -26,17 +26,9 @@ Check the tasks board and roadmap [here](https://github.com/orgs/hotosm/projects

### Get involved!

We invite software designers and developers to contribute to the project, there are several issues
where we need help, some of them are:

* Designs for data visualizations
* React UI components
* Data quality checks for the C++ core engine
* PostgreSQL queries for the Python `dbapi` module
* Endpoints for the Python `restapi` module
* Packages for Python, React and system binaries
* Data models for semantic validation
* Tests for everything
This is an exciting project, its core is made with C++ but it also includes a Python API and there are UI components for interacting and visualizing data.

You can learn a lot contributing to Underpass, while helping the FOSS and mapping communities, get involved!

### License

Expand Down
2 changes: 2 additions & 0 deletions config/default.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,8 @@
config:
- underpass_db_url:
- underpass:underpass@localhost:5432/underpass
- underpass_osm_db_url:
- underpass:underpass@localhost:5432/underpass
- planet_servers:
- planet.maps.mail.ru
- destdir_base:
Expand Down
65 changes: 36 additions & 29 deletions docker-compose.yml
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
#
# Copyright (c) 2020, 2021, 2022, 2023 Humanitarian OpenStreetMap Team
# Copyright (c) 2020, 2021, 2022, 2023, 2024 Humanitarian OpenStreetMap Team
#
# This file is part of Underpass.
#
Expand All @@ -20,10 +20,10 @@
version: "3"

services:
# Database
postgis:
image: postgis/postgis:${POSTGIS_TAG:-15-3.3-alpine}
container_name: "underpass_postgis"
# Database for Underpass
underpass_db:
image: postgis/postgis:${UNDERPASS_DB_TAG:-15-3.3-alpine}
container_name: "underpass_db"
ports:
- "${DB_PORT:-5439}:5432"
environment:
Expand All @@ -41,6 +41,28 @@ services:
networks:
internal:

# Un-comment for starting a second database
# Database for OSM Raw Data
# osm_db:
# image: postgis/postgis:${OSM_DB_TAG:-15-3.3-alpine}
# container_name: "osm_db"
# ports:
# - "${DB_PORT:-5440}:5432"
# environment:
# - POSTGRES_DB=osm
# - POSTGRES_USER=underpass
# - POSTGRES_PASSWORD=underpass
# volumes:
# - ./data_osm:/var/lib/postgresql/data
# restart: on-failure
# logging:
# driver: "json-file"
# options:
# max-size: "200k"
# max-file: "10"
# networks:
# internal:

# Underpass
underpass:
image: "ghcr.io/hotosm/underpass:${TAG_OVERRIDE:-debug}"
Expand All @@ -51,13 +73,16 @@ services:
target: ${TAG_OVERRIDE:-debug}
args:
APP_VERSION: ${APP_VERSION:-debug}
depends_on: [postgis]
depends_on: [underpass_db]
environment:
- REPLICATOR_UNDERPASS_DB_URL=underpass:underpass@postgis/underpass
- REPLICATOR_UNDERPASS_DB_URL=underpass:underpass@underpass_db/underpass
- REPLICATOR_OSM_DB_URL=underpass:underpass@underpass_db/underpass

command: tail -f /dev/null
# Un-comment for debugging
# volumes:
# - ${PWD}:/code
# - ./replication:/code/build/replication
# - ${PWD}:/code
# - ./replication:/usr/local/lib/underpass/data/replication
networks:
internal:

Expand All @@ -79,26 +104,8 @@ services:
networks:
internal:
environment:
- UNDERPASS_API_DB=postgresql://underpass:underpass@postgis/underpass

# Underpass UI
ui:
image: "ghcr.io/hotosm/underpass/ui:${APP_VERSION:-debug}"
container_name: "underpass_ui"
build:
context: .
dockerfile: docker/underpass-ui.dockerfile
target: debug
args:
APP_VERSION: ${APP_VERSION:-debug}
# # Mount underpass-ui repo
# volumes:
# - ../underpass-ui/src:/code/src
# - ../underpass-ui/playground:/code/playground
ports:
- "${UI_PORT:-8080}:5000"
networks:
internal:
- UNDERPASS_API_DB=postgresql://underpass:underpass@underpass/underpass
- UNDERPASS_API_OSM_DB=postgresql://underpass:underpass@underpass/underpass

networks:
internal:
4 changes: 3 additions & 1 deletion docker/underpass-config.yaml
Original file line number Diff line number Diff line change
@@ -1,7 +1,9 @@
# Underpass config file
config:
- underpass_osm_db_url:
- underpass:underpass@underpass_db:5432/underpass
- underpass_db_url:
- underpass@postgis/underpass
- underpass:underpass@underpass_db:5432/underpass
- planet_servers:
- planet.maps.mail.ru
- destdir_base:
Expand Down
1 change: 1 addition & 0 deletions docker/underpass.dockerfile
Original file line number Diff line number Diff line change
Expand Up @@ -48,6 +48,7 @@ RUN set -ex \
"librange-v3-dev" \
"libtool" \
"osm2pgsql" \
"rsync" \
&& rm -rf /var/lib/apt/lists/*


Expand Down
2 changes: 1 addition & 1 deletion docs/Doxyfile.in
Original file line number Diff line number Diff line change
Expand Up @@ -844,7 +844,7 @@ WARN_LOGFILE =
# spaces. See also FILE_PATTERNS and EXTENSION_MAPPING
# Note: If this tag is empty the current directory is searched.

INPUT = @SRCDIR@
INPUT = ../..

# This tag can be used to specify the character encoding of the source files
# that doxygen parses. Internally doxygen uses the UTF-8 encoding. Doxygen uses
Expand Down
4 changes: 2 additions & 2 deletions m4/ax_boost_timer.m4
Original file line number Diff line number Diff line change
Expand Up @@ -67,8 +67,8 @@ AC_DEFUN([AX_BOOST_TIMER],
[AC_LANG_PUSH([C++])
CXXFLAGS_SAVE=$CXXFLAGS
AC_COMPILE_IFELSE([AC_LANG_PROGRAM([[@%:@include <boost/timer.hpp>]],
[[boost::timer timer;]])],
AC_COMPILE_IFELSE([AC_LANG_PROGRAM([[@%:@include <boost/timer/timer.hpp>]],
[[boost::timer::auto_cpu_timer t;]])],
ax_cv_boost_timer=yes, ax_cv_boost_timer=no)
CXXFLAGS=$CXXFLAGS_SAVE
AC_LANG_POP([C++])
Expand Down
6 changes: 6 additions & 0 deletions python/dbapi/api/config.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
# Results per page on raw featues queries
RESULTS_PER_PAGE = 500
# Results per page on list featues queries
RESULTS_PER_PAGE_LIST = 10
# Print debug messages
DEBUG=False
33 changes: 21 additions & 12 deletions python/dbapi/api/db.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,21 +19,23 @@

import asyncpg
import json
from .config import DEBUG

class UnderpassDB():
# Default Underpass local DB configuration
# This might be replaced by an .ini config file
class DB():
# Default DB configuration

def __init__(self, connectionString = None):
self.connectionString = connectionString or "postgresql://underpass:underpass@postgis/underpass"
self.connectionString = connectionString or "postgresql://underpass:underpass@localhost:5432/underpass"
self.pool = None
# Extract the name of the database
self.name = self.connectionString[self.connectionString.rfind('/') + 1:]

async def __enter__(self):
await self.connect()

async def connect(self):
""" Connect to the database """
print("Connecting to DB ...")
print("Connecting to DB ... " + self.connectionString if DEBUG else "")
if not self.pool:
try:
self.pool = await asyncpg.create_pool(
Expand All @@ -50,20 +52,27 @@ def close(self):
if self.pool is not None:
self.pool.close()

async def run(self, query, singleObject = False):
async def run(self, query, singleObject = False, asJson=False):
if DEBUG:
print("Running query ...")
if not self.pool:
await self.connect()
if self.pool:
result = None
try:
conn = await self.pool.acquire()
result = await conn.fetch(query)
if singleObject:
return result[0]
return json.loads((result[0]['result']))
data = None
if asJson:
data = result[0]['result']
elif singleObject:
data = result[0]
else:
data = result
await self.pool.release(conn)
return data
except Exception as e:
print("\n******* \n" + query + "\n******* \n")
print(e)
return None
finally:
await self.pool.release(conn)
return None

44 changes: 39 additions & 5 deletions python/dbapi/api/filters.py
Original file line number Diff line number Diff line change
@@ -1,21 +1,55 @@
#!/usr/bin/python3
#
# Copyright (c) 2023, 2024 Humanitarian OpenStreetMap Team
#
# This file is part of Underpass.
#
# Underpass is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# Underpass is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with Underpass. If not, see <https://www.gnu.org/licenses/>.

def tagsQueryFilter(tagsQuery, table):
query = ""
tags = tagsQuery.split(",")
keyValue = tags[0].split("=")

if len(keyValue) == 2:
query += "{0}.tags->>'{1}' ~* '^{2}'".format(table, keyValue[0], keyValue[1])
query += "{table}.tags->>'{key}' ~* '^{value}'".format(
table=table,
key=keyValue[0],
value=keyValue[1]
)
else:
query += "{0}.tags->>'{1}' IS NOT NULL".format(table, keyValue[0])
query += "{table}.tags->>'{key}' IS NOT NULL".format(
table=table,
key=keyValue[0]
)

for tag in tags[1:]:
keyValue = tag.split("=")
if len(keyValue) == 2:
query += "OR {0}.tags->>'{1}' ~* '^{2}'".format(table, keyValue[0], keyValue[1])
query += "OR {table}.tags->>'{key}' ~* '^{value}'".format(
table=table,
key=keyValue[0],
value=keyValue[1]
)
else:
query += "OR {0}.tags->>'{1}' IS NOT NULL".format(table, keyValue[0])
query += "OR {table}.tags->>'{key}' IS NOT NULL".format(
table=table,
key=keyValue[0]
)
return query

def hashtagQueryFilter(hashtag, table):
return "'{0}' = ANY (hashtags)".format(hashtag)
return "'{hashtag}' = ANY (hashtags)".format(
hashtag=hashtag
)
14 changes: 7 additions & 7 deletions python/dbapi/api/queryHelper.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
#!/usr/bin/python3
#
# Copyright (c) 2023 Humanitarian OpenStreetMap Team
# Copyright (c) 2023, 2024 Humanitarian OpenStreetMap Team
#
# This file is part of Underpass.
#
Expand All @@ -17,14 +17,14 @@
# You should have received a copy of the GNU General Public License
# along with Underpass. If not, see <https://www.gnu.org/licenses/>.

RESULTS_PER_PAGE = 25

def hashtags(hashtagsList):
return "EXISTS ( SELECT * from unnest(hashtags) as h where {0} )".format(
' OR '.join(
map(lambda x: "h ~* '^{0}'".format(x), hashtagsList)
return "EXISTS ( SELECT * from unnest(hashtags) as h where {condition} )".format(
condition=' OR '.join(
map(lambda x: "h ~* '^{hashtag}'".format(hashtag=x), hashtagsList)
)
)

def bbox(wktMultipolygon):
return "ST_Intersects(bbox, ST_GeomFromText('{0}', 4326))".format(wktMultipolygon)
return "ST_Intersects(bbox, ST_GeomFromText('{area}', 4326))".format(
area=wktMultipolygon
)
Loading

0 comments on commit d9e5a41

Please sign in to comment.