Source code for aequilibrae.project.zoning

from copy import deepcopy
from os.path import join, realpath
from typing import Union, Dict

import geopandas as gpd
import shapely.wkb
from shapely.geometry import Point, Polygon, LineString, MultiLineString
from shapely import union_all

from aequilibrae.project.basic_table import BasicTable
from aequilibrae.project.data_loader import DataLoader
from aequilibrae.project.project_creation import run_queries_from_sql_file
from aequilibrae.project.table_loader import TableLoader
from aequilibrae.project.zone import Zone
from aequilibrae.utils.db_utils import commit_and_close
from aequilibrae.utils.geo_index import GeoIndex
from aequilibrae.utils.spatialite_utils import connect_spatialite


[docs] class Zoning(BasicTable): """ Access to the API resources to manipulate the 'zones' table in the project .. code-block:: python >>> project = create_example(project_path, "coquimbo") >>> zoning = project.zoning >>> zone_downtown = zoning.get(1) >>> zone_downtown.population = 637 >>> zone_downtown.employment = 10039 >>> zone_downtown.save() # We can also add one more field to the table >>> fields = zoning.fields >>> fields.add('parking_spots', 'Total licensed parking spots', 'INTEGER') """
[docs] def __init__(self, network): super().__init__(network.project) self.__items: Dict[int, Zone] = {} self.network = network self.__table_type__ = "zones" self.__fields = [] self.__geo_index = GeoIndex() if self.__has_zoning(): self.__load()
[docs] def new(self, zone_id: int) -> Zone: """Creates a new zone :Returns: **zone** (:obj:`Zone`): A new zone object populated only with ``zone_id`` (but not saved in the model yet) """ if zone_id in self.__items: raise Exception(f"Zone ID {zone_id} already exists") data = {key: None for key in self.__fields} data["zone_id"] = zone_id self.project.logger.info(f"Zone with id {zone_id} was created") return self.__create_return_zone(data)
[docs] def create_zoning_layer(self): """Creates the 'zones' table for project files that did not previously contain it""" if not self.__has_zoning(): qry_file = join(realpath(__file__), "database_specification", "tables", "zones.sql") with commit_and_close(connect_spatialite(self.project.path_to_file)) as conn: run_queries_from_sql_file(conn, self.project.logger, qry_file) self.__load() else: self.project.warning("zones table already exists. Nothing was done", Warning)
[docs] def coverage(self) -> Polygon: """Returns a single polygon for the entire zoning coverage :Returns: **model coverage** (:obj:`Polygon`): Shapely (Multi)polygon of the zoning system. """ with commit_and_close(connect_spatialite(self.project.path_to_file)) as conn: dt = conn.execute('Select ST_asBinary("geometry") from zones;').fetchall() polygons = [shapely.wkb.loads(x[0]) for x in dt] return union_all(polygons)
[docs] def get(self, zone_id: str) -> Zone: """Get a zone from the model by its ``zone_id``""" if zone_id not in self.__items: raise ValueError(f"Zone {zone_id} does not exist in the model") return self.__items[zone_id]
[docs] def all_zones(self) -> dict: """Returns a dictionary with all Zone objects available in the model, using ``zone_id`` as key""" return self.__items
[docs] def save(self): for item in self.__items.values(): item.save()
[docs] def get_closest_zone(self, geometry: Union[Point, LineString, MultiLineString]) -> int: """Returns the zone in which the given geometry is located. If the geometry is not fully enclosed by any zone, the zone closest to the geometry is returned :Arguments: **geometry** (:obj:`Point` or :obj:`LineString`): A Shapely geometry object :Returns: **zone_id** (:obj:`int`): ID of the zone applicable to the point provided """ nearest = self.__geo_index.nearest(geometry, 10) dists = {} for zone_id in nearest: geo = self.__items[zone_id].geometry if geo.contains(geometry): return zone_id dists[geo.distance(geometry)] = zone_id return dists[min(dists.keys())]
[docs] def refresh_geo_index(self): self.__geo_index.reset() for zone_id, zone in self.__items.items(): self.__geo_index.insert(feature_id=zone_id, geometry=zone.geometry)
def __has_zoning(self): with commit_and_close(connect_spatialite(self.project.path_to_file)) as conn: dt = conn.execute("SELECT name FROM sqlite_master WHERE type='table';").fetchall() return any("zone" in x[0].lower() for x in dt) def __load(self): tl = TableLoader() with commit_and_close(connect_spatialite(self.project.path_to_file)) as conn: zones_list = tl.load_table(conn, "zones") self.__fields = deepcopy(tl.fields) existing_list = [zn["zone_id"] for zn in zones_list] if zones_list: self.__properties = list(zones_list[0].keys()) for zn in zones_list: if zn["zone_id"] not in self.__items: self.__items[zn["zone_id"]] = Zone(zn, self) to_del = [key for key in self.__items.keys() if key not in existing_list] for key in to_del: del self.__items[key] self.refresh_geo_index() def _remove_zone(self, zone_id: int): del self.__items[zone_id] def __create_return_zone(self, data): zone = Zone(data, self) self.__items[zone.zone_id] = zone return zone @property def data(self) -> gpd.GeoDataFrame: """Returns all zones data as a Pandas DataFrame :Returns: **table** (:obj:`GeoDataFrame`): GeoPandas GeoDataFrame with all the nodes """ dl = DataLoader(self.project.path_to_file, "zones") return dl.load_table()