Measuring Natural Surveillance at Scale: An Automated Methodology for Investigating the Relation Between the "Eyes on the Street" and Urban Safety
This repository contains the code belonging to my master thesis: Measuring Natural Surveillance at Scale: An Automated Methodology for Investigating the Relation Between the "Eyes on the Street" and Urban Safety. See the document for a detailed descripton of the project and the obained results.
To create safe urban areas, it is important to gain insight into what influences the (perceived) safety of our cities and human settlements. One of the factors that can contribute to safety is the way urban spaces are designed. Previous work has highlighted the importance of natural surveillance: a type of surveillance that is a byproduct of how citizens normally and routinely use the environment. However, studying this concept is not a trivial task. Manual approaches such as observation studies are costly and time-consuming and have therefore often limited themselves to smaller geographical areas.
In this work, we present a methodology that can automatically provide an estimate of natural surveillance by detecting building openings (i.e. windows and doors) in street level imagery and localizing them in 3 dimensions. The proposed method is able to estimate natural surveillance at the street segment level, while simultaneously being able to gather data on a whole city in a matter of hours. We then apply our method to the city of Amsterdam to analyze the relationship between natural surveillance and urban safety using the Amsterdam Safety Index.
We conclude that our chosen operationalization of natural surveillance (road surveillability and occupant surveillability) is correlated with decreases in high impact crime and nuisance as well as increases in perceived safety. Furthermore we provide evidence for the existence of a threshold after which extra natural surveillance is no longer associated with higher degrees of safety.
This project is written and tested in Python 3.9.
Clone the repository:
git clone https://github.com/timovanasten/natural-surveillance.git
cd natural-surveillance
Optionally, create and activate a virtual environment:
For MacOS/Linux:
python3 -m venv ./venv
source ./venv/bin/activate
or for Windows:
python3 -m venv ./venv
.\venv\Scripts\activate.bat
Lastly, install the dependencies:
pip install -r requirements.txt
The trained model used for detecting building openings within the street view imagery can be downloaded here.
Next, create the /labeling/heatmap_fusion/model
directory. For example by running
mkdir ./labeling/heatmap_fusion/model
and place resnet18_model_latest.pth.tar
into the natural-surveillance/labeling/heatmap_fusion/model
directory.
Alternatively, point to the path the model is located in settings.yaml
.
Obtain an API key for the Google Street View Static API within the Google Cloud platform and add it to .secrets.yaml
.
To not overshoot available credit, enter your credit in dollar in data/gsv_budget.txt
.
A dataset in CSV format with the estimated natural surveillance scores for over 6500 street segments spread over 43 neighborhoods in Amsterdam can be found in the data/results
directory.
To create a nice visualization of the dataset, such as the one above, check out kepler.gl.
A dataset containing the estimated geolocation of 872 360 building openings is also available. Please contact me for access.
The code that was ran for the experiment outlined in the thesis document can be found in main.py
.
To obtain your own data, update the EPSG of the local coordinate reference system in settings.yaml
to one of the area of interest and create a Pipeline
object using either a polygon, address or neighborhood name*:
from shapely.geometry import Polygon
from pipeline import Pipeline, PipelineConfig, PipelineStep
# Option 1: using area polygon
polygon = Polygon([4.4359826, 52.2269100],
...,
[4.4364993, 52.2271558])
config = PipelineConfig(pipeline_name,
field_of_view,
sightline_distance,
viewing_angle,
polygon=polygon)
# Option 2: using address
config = PipelineConfig(pipeline_name,
field_of_view,
sightline_distance,
viewing_angle,
address="Mekelweg 4, Delft",
distance_around_address=100)
# Option 3: using neighborhood name
config = PipelineConfig(pipeline_name,
field_of_view,
sightline_distance,
viewing_angle,
neighborhood_name="Staatliedenbuurt")
# Create the pipeline
pipeline = Pipeline(config)
# Execute all steps in the pipeline:
pipeline.execute_all()
# ...or only certain step(s):
pipeline.execute_step(PipelineStep.LOCALIZE_OPENINGS)
pipeline.execute_up_to_and_including(PipelineStep.LOCALIZE_OPENINGS)
pipeline.execute_from(PipelineStep.CALCULATE_SIGHTLINES)
# Pipelines are saved after each step.
# To load a prevously saved pipeline into memory use:
pipeline = Pipeline.load_from_state_file(pipeline_name)
*This repository only contains the neighborhood geometries for Amsterdam. To use other neighborhood boundaries,
alter neighborhoods.py
and settings.yaml
.
For any questions or requests, feel free to contact me at [email protected].