forked from perrygeo/gdal_utils
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathreclass_raster.py
101 lines (84 loc) · 2.67 KB
/
reclass_raster.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
#!/usr/bin/env python
USAGE = """
Reclassify raster from continuous values to discrete classes
Example usage:
python reclass_raster.py input.tif classes.cfg output.tif
where classes.cfg contains lines like:
min,max,class
"""
import sys
import numpy
from osgeo import gdal
from progressbar import ProgressBar, Percentage, ETA, SimpleProgress
def parse_config(fh):
lines = fh.readlines()
cfg = []
for line in lines:
items = [float(x) for x in line.strip().split(',')]
items[2] == int(items[2])
if len(items) == 3:
cfg.append(items)
else:
raise Exception
if len(cfg) < 1:
raise Exception
return cfg
def classify_val(val, cfg):
for rule in cfg:
if val >= rule[0] and val < rule[1]:
return rule[2]
print "WARNING: value %s did not fall into any class"
return None
if __name__ == '__main__':
format = 'GTiff'
gdal.AllRegister()
argv = gdal.GeneralCmdLineProcessor( sys.argv )
if len(argv) < 4:
print USAGE
sys.exit( 0 )
cont = gdal.Open( argv[1], gdal.GA_ReadOnly )
if cont is None:
print "%s is not a valid raster file" % argv[1]
sys.exit( 0 )
try:
cfg_fh = open(argv[2], 'r')
cfg = parse_config(cfg_fh)
except:
print "Configuration file '%s' is not valid" % argv[4]
sys.exit( 0 )
geotransform = cont.GetGeoTransform()
xsize = cont.RasterXSize
ysize = cont.RasterYSize
projection = cont.GetProjection()
if projection == '':
projection = None
# Create output
driver = gdal.GetDriverByName(format)
if driver is None:
print 'Format driver %s not found, pick a supported driver.' % format
sys.exit( 1 )
bands = 1
band_type = gdal.GDT_Int16
out = driver.Create( argv[3], xsize, ysize, bands, band_type )
if out is None:
print 'Creation failed, terminating.'
sys.exit( 1 )
out.SetGeoTransform(geotransform)
if projection:
out.SetProjection(projection)
inband = cont.GetRasterBand(1)
indata = inband.ReadAsArray()
outband = out.GetRasterBand(1)
progress = ProgressBar(widgets=['Processed Lines ', SimpleProgress(), ' (', Percentage(), ') ' , ETA(), ' remaining' ],maxval=ysize)
progress.start()
progress.update_interval = 1
progress.next_update = 1
for line in range(ysize):
zones_line = []
for xpos in range(xsize):
val = indata[line, xpos]
zone = classify_val(val, cfg)
zones_line.append(zone)
outband.WriteArray( numpy.array([zones_line]), 0, line )
progress.update(line)
print