-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathutils_functions.py
174 lines (137 loc) · 5.9 KB
/
utils_functions.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
# -*- coding: utf-8 -*-
"""
Created on Wed May 2 09:25:11 2018
@author: cenv0574
"""
import os
import json
from geopy.distance import vincenty
from boltons.iterutils import pairwise
import geopandas as gpd
import shapely.ops
import pandas as pd
import shapely.wkt
import shutil
def load_config():
"""Read config.json
"""
# config_path = os.path.join(os.path.dirname(__file__), '..', 'config.json')
config_path = os.path.join(os.path.dirname(__file__), 'config.json')
with open(config_path, 'r') as config_fh:
config = json.load(config_fh)
return config
def clean_dir(dirpath):
""""This function can be used to fully clear a directory
Arguments:
dirpath {string} -- path to directory to be cleared from files
"""
for filename in os.listdir(dirpath):
filepath = os.path.join(dirpath, filename)
try:
shutil.rmtree(filepath)
except OSError:
os.remove(filepath)
def remove_files(dirpath,startname):
"""This function can be used to delete specific files from a directory. In
general this function is used to clean country files from the 'calc' directory
Arguments:
dirpath {string} -- path to directory in which the files should be removed
startname {string} -- the substring to be searched for in the files
"""
for fname in os.listdir(dirpath):
if fname.startswith(startname):
os.remove(os.path.join(dirpath, fname))
def create_folder_structure(data_path,regionalized=True):
"""Create the directory structure for the output
Arguments:
base_path {string} -- path to directory where folder structure should be created
Keyword Arguments:
regionalized {bool} -- specify whether also the folders for a regionalized analyse should be created (default: {True})
"""
data_path = load_config()['paths']['data']
# create calc dir
calc_dir = os.path.join(data_path,'calc')
if not os.path.exists(calc_dir):
os.makedirs(calc_dir)
# create country osm dir
dir_osm = os.path.join(data_path,'country_osm')
if not os.path.exists(dir_osm):
os.makedirs(dir_osm)
if regionalized == True:
dir_osm_region = os.path.join(data_path,'region_osm')
if not os.path.exists(dir_osm_region):
os.makedirs(dir_osm_region)
# create output dirs for country level analysis
dir_out = os.path.join(data_path,'country_data')
if not os.path.exists(dir_out):
os.makedirs(dir_out)
# create dir and output file for continent outputs
continent_dir_high = os.path.join(data_path,'continental_highway')
if not os.path.exists(continent_dir_high):
os.makedirs(continent_dir_high)
continent_dir_rail = os.path.join(data_path,'continental_railway')
if not os.path.exists(continent_dir_rail):
os.makedirs(continent_dir_rail)
continent_dir_avi = os.path.join(data_path,'continental_aviation')
if not os.path.exists(continent_dir_avi):
os.makedirs(continent_dir_avi)
continent_dir_shi = os.path.join(data_path,'continental_shipping')
if not os.path.exists(continent_dir_shi):
os.makedirs(continent_dir_shi)
continent_dir_mm = os.path.join(data_path,'continental_multimodal')
if not os.path.exists(continent_dir_mm):
os.makedirs(continent_dir_mm)
def simplify_paths(x):
"""simpilify the geometry of the specific geometric feature. This will
reduce the calculation times quite a bit.
Arguments:
x {shapely geometry} -- a shapely geometry to be simplified.
"""
return x.simplify(tolerance=0.0001, preserve_topology=True)
def feather_to_gpd(file_name,save_shapefile=False):
"""Read a pandas dataframe saved as a feather file (.ft) and convert to a
geodatabase.
Arguments:
file_name {string} -- full path to feather file to load
Keyword Arguments:
save_shapefile {bool} -- Save the created GeoDatabase as shapefile (default: {False})
Returns:
Geopandas geodataframe of the .feather file
"""
# load feather file, convert geometry to normal linestrings again and
# fix some othe rsmall issues before saving it to a shapefile
gpd_in = gpd.GeoDataFrame(pd.read_feather(file_name))
gpd_in['geometry'] = gpd_in['geometry'].apply(shapely.wkb.loads)
gpd_in.crs = {'init' :'epsg:4326'}
gpd_in['infra_type'] = gpd_in['infra_type'].astype(str)
gpd_in['country'] = gpd_in['country'].astype(str)
gpd_in = gpd.GeoDataFrame(gpd_in,geometry='geometry')
if save_shapefile == True:
# set new file name
output_name = file_name[:-3]+'.shp'
gpd_in.to_file(output_name)
return gpd_in
def line_length(line, ellipsoid='WGS-84',shipping=True):
"""Length of a line in meters, given in geographic coordinates
Adapted from https://gis.stackexchange.com/questions/4022/looking-for-a-pythonic-way-to-calculate-the-length-of-a-wkt-linestring#answer-115285
Arguments:
line {Shapely LineString} -- a shapely LineString object with WGS-84 coordinates
ellipsoid {String} -- string name of an ellipsoid that `geopy` understands (see
http://geopy.readthedocs.io/en/latest/#module-geopy.distance)
Returns:
Length of line in meters
"""
if shipping == True:
if line.geometryType() == 'MultiLineString':
return sum(line_length(segment) for segment in line)
return sum(
vincenty(tuple(reversed(a)), tuple(reversed(b)), ellipsoid=ellipsoid).kilometers
for a, b in pairwise(line.coords)
)
else:
if line.geometryType() == 'MultiLineString':
return sum(line_length(segment) for segment in line)
return sum(
vincenty(a, b, ellipsoid=ellipsoid).kilometers
for a, b in pairwise(line.coords)
)