forked from pmlrsg/arsf_tools
-
Notifications
You must be signed in to change notification settings - Fork 0
/
assign_projection.py
executable file
·221 lines (174 loc) · 7.13 KB
/
assign_projection.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
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
#!/usr/bin/env python
# -*- coding: utf-8 -*-
"""
Add missing projection information to GDAL dataset from:
* Existing image
* WKT file
* Proj4 string
* EPSG code
For where GDAL dataset has correct geoinformation but projection
has not been stored correctly.
For example when exporting files from APL using projections other than
Lat/Long WGS84, UTM or OSGB36 NG this script can be used to fix the
following error:
"Unknown map projection: You will have to fill in the projection name
and datum in the map info in the .hdr file yourself."
Note this script will not reproject data. For this use gdalwarp
Requires GDAL Python bindings to be installed and available
"""
###########################################################
# This file has been created by ARSF Data Analysis Node and
# is licensed under the GPL v3 Licence. A copy of this
# licence is available to download with this file.
###########################################################
import argparse
import sys
import glob
try:
from osgeo import gdal, osr
except ImportError:
print('ERROR: Could not import GDAL Python library.')
print('Check GDAL is installed and if necessary added to "PYTHONPATH" environmental variable')
sys.exit(2)
def assign_from_image(in_image, in_basefile):
"""
Assign projection to one GDAL dataset
from another.
Arguments:
* in_image - Input image to assign projection to
* in_basefile - Input image to copy projection information from
"""
print('Copying projection from {} to {}'.format(in_basefile, in_image))
# Open the image file, in update mode
# so that the image can be edited.
baseDS = gdal.Open(in_basefile, gdal.GA_ReadOnly)
outputDS = gdal.Open(in_image, gdal.GA_Update)
if (not baseDS is None) and (not outputDS is None):
outputDS.SetProjection(baseDS.GetProjection())
else:
raise Exception("Could not open input / output file")
baseDS = None
outputDS = None
def assign_from_wkt(in_image, in_wkt_file):
"""
Assign projection to a GDAL dataset
from WKT file
Arguments:
* in_image - Input image to assign projection to
* in_wkt_file - Input WKT file
"""
print('Adding projection from {} to {}'.format(in_wkt_file, in_image))
# Read WKT file to a string
in_wkt_handler = open(in_wkt_file,'r')
in_wkt_str = in_wkt_handler.read()
in_wkt_handler.close()
# Open the image file, in update mode
# so that the image can be edited.
outputDS = gdal.Open(in_image, gdal.GA_Update)
if not outputDS is None:
outputDS.SetProjection(in_wkt_str)
else:
raise Exception("Could not open input / output file")
outputDS = None
def assign_from_proj4(in_image, in_proj4):
"""
Assign projection to a GDAL dataset
from Proj4 string
Arguments:
* in_image - Input image to assign projection to
* in_proj4 - Proj4 string
"""
print('Adding projection {} to {}'.format(in_proj4, in_image))
# Check if Proj4 string is within quotes
# remove if so.
if in_proj4[0] == '"' or in_proj4[0] == '\'':
in_proj4 = in_proj4[1:-1]
spatial_ref = osr.SpatialReference()
osr_out = spatial_ref.ImportFromProj4(in_proj4)
if osr_out != 0:
raise Exception('Could not create projection. '\
'Is "{}" a valid proj4 string'.format(in_proj4))
wkt_str = spatial_ref.ExportToWkt()
# Open the image file, in update mode
# so that the image can be edited.
outputDS = gdal.Open(in_image, gdal.GA_Update)
if not outputDS is None:
outputDS.SetProjection(wkt_str)
else:
raise Exception("Could not open input / output file")
outputDS = None
def assign_from_epsg_code(in_image, epsg_code):
"""
Assign projection to a GDAL dataset
from EPSG code
Arguments:
* in_image - Input image to assign projection to
* epsg_code - EPSG code
"""
# Tidy up EPSG code format
if isinstance(epsg_code,int):
epsg_code_int = epsg_code
elif isinstance(epsg_code,str):
epsg_code_int = int(epsg_code.split(':')[-1])
else:
raise Exception('Did not understand input format, expected string or integer')
print('Adding projection from EPSG code {} to {}'.format(epsg_code, in_image))
spatial_ref = osr.SpatialReference()
osr_out = spatial_ref.ImportFromEPSG(epsg_code_int)
if osr_out != 0:
raise Exception('Could not create projection. '\
'Is "{}" a valid EPGS code'.format(epsg_code_int))
wkt_str = spatial_ref.ExportToWkt()
# Open the image file, in update mode
# so that the image can be edited.
outputDS = gdal.Open(in_image, gdal.GA_Update)
if not outputDS is None:
outputDS.SetProjection(wkt_str)
else:
raise Exception("Could not open input / output file")
outputDS = None
if __name__ == '__main__':
parser = argparse.ArgumentParser(description='''Assign missing projection information to a GDAL dataset.
Note this script does not reproject data it
only adds an existing projection in a format GDAL can recognise.
Created by ARSF-DAN at Plymouth Marine Laboratory.
Latest version available from https://github.com/pmlrsg/arsf_tools/.''')
parser.add_argument("inputfile", nargs='+',type=str, help="Input Files(s)")
parser.add_argument("-b", "--baseimage", type=str,
help="Copy projection information from specified image",
required=False,
default=None)
parser.add_argument("-w", "--wkt", type=str,
help="Assign projection information from specified WKT file",
required=False,
default=None)
parser.add_argument("-p", "--proj4", type=str,
help="Assign projection information from specified Proj4 string",
required=False,
default=None)
parser.add_argument("-e", "--epsg", type=str,
help="Assign projection information from EPSG code",
required=False,
default=None)
args = parser.parse_args()
# On Windows don't have shell expansion so fake it using glob
if args.inputfile[0].find('*') > -1:
input_file_list = glob.glob(args.inputfile[0])
else:
input_file_list = args.inputfile
try:
for image in input_file_list:
if args.baseimage is not None:
assign_from_image(image, args.baseimage)
elif args.wkt is not None:
assign_from_wkt(image, args.wkt)
elif args.proj4 is not None:
assign_from_proj4(image, args.proj4)
elif args.epsg is not None:
assign_from_epsg_code(image, args.epsg)
else:
print("ERROR: must provide one of '--baseimage','--wkt','--proj4' or '--epsg'")
sys.exit(1)
except Exception as err:
print(err)
sys.exit(1)