diff --git a/src/__init__.py b/src/__init__.py new file mode 100644 index 0000000..40a96af --- /dev/null +++ b/src/__init__.py @@ -0,0 +1 @@ +# -*- coding: utf-8 -*- diff --git a/src/backend/__init__.py b/src/backend/__init__.py new file mode 100644 index 0000000..40a96af --- /dev/null +++ b/src/backend/__init__.py @@ -0,0 +1 @@ +# -*- coding: utf-8 -*- diff --git a/src/backend/postgis/__init__.py b/src/backend/postgis/__init__.py new file mode 100644 index 0000000..4271ed8 --- /dev/null +++ b/src/backend/postgis/__init__.py @@ -0,0 +1,96 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# This file is part of kothic, the realtime map renderer. + +# kothic 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. + +# kothic 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 kothic. If not, see . + +#from debug import debug +from twms import projections +import psycopg2 +import shapely.wkb + +class Way: + def __init__(self, tags, geom): + + self.cs = None + #print [x.split("=") for x in tags.split(";")] + self.tags = tags + # calculating center point + #c= geom + #sumz = [(c[0],c[1])] + #for k in range(2, len(c), 2): + # sumz.append((c[k], c[k + 1])) + self.coords = geom + # left for the better times: + self.center = reduce(lambda x, y: (x[0]+y[0],x[1]+y[1]), self.coords) + self.center = (self.center[0]/len(self.coords),self.center[1]/len(self.coords)) + #debug(self.center) + +class PostGisBackend: + """ + A class that gives out vector data on demand. + """ + + + def __init__(self,database = "dbname=gis user=mapz",max_zoom = 16,proj = "EPSG:3857", path = "tiles", lang = "ru", ): + + + # debug("Bakend created") + self.database=database + self.max_zoom = max_zoom # no better tiles available + self.path = path # path to tile files + self.lang = lang # map language to use + self.tiles = {} # loaded vector tiles go here + self.data_projection = proj # which projection used to cut map in tiles + self.keep_tiles = 190 # a number of tiles to cache in memory + self.tile_load_log = [] # used when selecting which tile to unload + + def get_vectors (self, bbox, zoom): + a = psycopg2.connect(self.database) + b = a.cursor() + bbox = tuple(projections.from4326(bbox,self.data_projection)) + ### FIXME: hardcoded EPSG:3857 in database + tables = ("planet_osm_line","planet_osm_polygon") # FIXME: points + resp = {} + for table in tables: + b.execute("SELECT * FROM %s WHERE way && SetSRID('BOX3D(%s %s,%s %s)'::box3d,900913);"%(table,bbox[0],bbox[1],bbox[2],bbox[3])) + names = [q[0] for q in b.description] + for row in b.fetchall(): + + row_dict = dict(map(None,names,row)) + for k,v in row_dict.items(): + if not v: + del row_dict[k] + geom = shapely.wkb.loads(row_dict["way"].decode('hex')) + ### FIXME: a dirty hack to basically support polygons, needs lots of rewrite + try: + geom = list(geom.coords) + except NotImplementedError: + "trying polygons" + try: + geom = geom.boundary + geom = list(geom.coords) + except NotImplementedError: + "multipolygon" + continue + ### FIXME + + geom = projections.to4326(geom, self.data_projection) + del row_dict["way"] + oid = row_dict["osm_id"] + del row_dict["osm_id"] + w = Way(row_dict, geom) + #print row_dict + resp[oid] = w + return resp \ No newline at end of file diff --git a/src/vtiles_backend.py b/src/backend/vtile/__init__.py similarity index 100% rename from src/vtiles_backend.py rename to src/backend/vtile/__init__.py diff --git a/src/generate_image.py b/src/generate_image.py index 5eb5eee..94e9376 100644 --- a/src/generate_image.py +++ b/src/generate_image.py @@ -17,7 +17,8 @@ from debug import debug, Timer -from vtiles_backend import QuadTileBackend as DataBackend +#from backend.vtile import QuadTileBackend as DataBackend +from backend.postgis import PostGisBackend as DataBackend #from style import Styling from mapcss import MapCSS @@ -25,7 +26,7 @@ from render import RasterTile style = MapCSS(1, 19) #zoom levels -style.parse(open("styles/default.mapcss","r").read()) +style.parse(open("styles/openstreetinfo.mapcss","r").read()) bbox = (27.115768874532,53.740327031764,28.028320754378,54.067187302158) diff --git a/src/twms_fetcher.py b/src/twms_fetcher.py new file mode 100644 index 0000000..0f4ebe1 --- /dev/null +++ b/src/twms_fetcher.py @@ -0,0 +1,51 @@ +# -*- coding: utf-8 -*- +# This file is part of tWMS. + +# tWMS 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. + +# tWMS 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 tWMS. If not, see . + +import StringIO +import Image +import os + +from twms import projections + +#from vtiles_backend import QuadTileBackend as DataBackend +from backend.postgis import PostGisBackend as DataBackend +from mapcss import MapCSS +from render import RasterTile +from tempfile import NamedTemporaryFile + +style = MapCSS(1,19) +style.parse(open("/home/kom/osm/kothic/src/styles/openstreetinfo.mapcss","r").read()) + + + +def kothic_fetcher (z, x, y, this_layer): + if "max_zoom" in this_layer: + if z >= this_layer["max_zoom"]: + return None + bbox = projections.bbox_by_tile(z,x,y,"EPSG:3857") + db = DataBackend(path="/home/kom/osm/kothic/src/tiles") + res = RasterTile(256, 256, 1, db, "EPSG:3857") + res.update_surface(bbox, z, style) + f = NamedTemporaryFile() + f.close() + res.surface.write_to_png(f.name) + del res + del db + im = Image.open(f.name) + os.unlink(f.name) + im = im.convert("RGBA") + + return im