https://github.com/GeoscienceAustralia/PyRate
Tip revision: 20300854948723d29f27ee4bc799adff4c698f8d authored by Sudipta Basak on 15 June 2021, 21:06:36 UTC
using shared instead could be faster
using shared instead could be faster
Tip revision: 2030085
gdal_calc_local.py
#! /usr/bin/python
# -*- coding: utf-8 -*-
#******************************************************************************
#
# Project: GDAL
# Purpose: Command line raster calculator with numpy syntax
# Author: Chris Yesson, chris.yesson@ioz.ac.uk
#
#******************************************************************************
# Copyright (c) 2010, Chris Yesson <chris.yesson@ioz.ac.uk>
# Copyright (c) 2010-2011, Even Rouault <even dot rouault at mines-paris dot org>
# Copyright (c) 2016, Piers Titus van der Torren <pierstitus@gmail.com>
#
# Permission is hereby granted, free of charge, to any person obtaining a
# copy of this software and associated documentation files (the "Software"),
# to deal in the Software without restriction, including without limitation
# the rights to use, copy, modify, merge, publish, distribute, sublicense,
# and/or sell copies of the Software, and to permit persons to whom the
# Software is furnished to do so, subject to the following conditions:
#
# The above copyright notice and this permission notice shall be included
# in all copies or substantial portions of the Software.
#
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
# OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
# THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
# FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
# DEALINGS IN THE SOFTWARE.
#******************************************************************************
################################################################
# Command line raster calculator with numpy syntax. Use any basic arithmetic supported by numpy arrays such as +-*\ along with logical operators such as >. Note that all files must have the same dimensions, but no projection checking is performed. Use gdal_calc.py --help for list of options.
# This script is a copy of gdal_calc.py included in the GDAL library, modified for use with PyRate.
################################################################
from optparse import OptionParser, Values
import os
import sys
import numpy
from osgeo import gdal
from osgeo import gdalnumeric
# create alphabetic list for storing input layers
AlphaList=["A","B","C","D","E","F","G","H","I","J","K","L","M",
"N","O","P","Q","R","S","T","U","V","W","X","Y","Z"]
# set up some default nodatavalues for each datatype
DefaultNDVLookup={'Byte':255, 'UInt16':65535, 'Int16':-32767, 'UInt32':4294967293, 'Int32':-2147483647, 'Float32':3.402823466E+38, 'Float64':1.7976931348623158E+308}
################################################################
def doit(opts, args):
if opts.debug:
print("gdal_calc.py starting calculation %s" %(opts.calc))
# set up global namespace for eval with all functions of gdalnumeric
global_namespace = dict([(key, getattr(gdalnumeric, key))
for key in dir(gdalnumeric) if not key.startswith('__')])
if not opts.calc:
raise Exception("No calculation provided.")
elif not opts.outF:
raise Exception("No output file provided.")
################################################################
# fetch details of input layers
################################################################
# set up some lists to store data for each band
myFiles=[]
myBands=[]
myAlphaList=[]
myDataType=[]
myDataTypeNum=[]
myNDV=[]
DimensionsCheck=None
# loop through input files - checking dimensions
for myI, myF in opts.input_files.items():
if not myI.endswith("_band"):
# check if we have asked for a specific band...
if "%s_band" % myI in opts.input_files:
myBand = opts.input_files["%s_band" % myI]
else:
myBand = 1
myFile = gdal.Open(myF, gdal.GA_ReadOnly)
if not myFile:
raise IOError("No such file or directory: '%s'" % myF)
myFiles.append(myFile)
myBands.append(myBand)
myAlphaList.append(myI)
myDataType.append(gdal.GetDataTypeName(myFile.GetRasterBand(myBand).DataType))
myDataTypeNum.append(myFile.GetRasterBand(myBand).DataType)
myNDV.append(myFile.GetRasterBand(myBand).GetNoDataValue())
# check that the dimensions of each layer are the same
if DimensionsCheck:
if DimensionsCheck != [myFile.RasterXSize, myFile.RasterYSize]:
raise Exception("Error! Dimensions of file %s (%i, %i) are different from other files (%i, %i). Cannot proceed" % \
(myF, myFile.RasterXSize, myFile.RasterYSize, DimensionsCheck[0], DimensionsCheck[1]))
else:
DimensionsCheck = [myFile.RasterXSize, myFile.RasterYSize]
if opts.debug:
print("file %s: %s, dimensions: %s, %s, type: %s" %(myI,myF,DimensionsCheck[0],DimensionsCheck[1],myDataType[-1]))
# process allBands option
allBandsIndex=None
allBandsCount=1
if opts.allBands:
try:
allBandsIndex=myAlphaList.index(opts.allBands)
except ValueError:
raise Exception("Error! allBands option was given but Band %s not found. Cannot proceed" % (opts.allBands))
allBandsCount=myFiles[allBandsIndex].RasterCount
if allBandsCount <= 1:
allBandsIndex=None
################################################################
# set up output file
################################################################
# open output file exists
if os.path.isfile(opts.outF) and not opts.overwrite:
if allBandsIndex is not None:
raise Exception("Error! allBands option was given but Output file exists, must use --overwrite option!")
if opts.debug:
print("Output file %s exists - filling in results into file" %(opts.outF))
myOut=gdal.Open(opts.outF, gdal.GA_Update)
if [myOut.RasterXSize,myOut.RasterYSize] != DimensionsCheck:
raise Exception("Error! Output exists, but is the wrong size. Use the --overwrite option to automatically overwrite the existing file")
myOutB=myOut.GetRasterBand(1)
myOutNDV=myOutB.GetNoDataValue()
myOutType=gdal.GetDataTypeName(myOutB.DataType)
else:
# remove existing file and regenerate
if os.path.isfile(opts.outF):
os.remove(opts.outF)
# create a new file
if opts.debug:
print("Generating output file %s" %(opts.outF))
# find data type to use
if not opts.type:
# use the largest type of the input files
myOutType=gdal.GetDataTypeName(max(myDataTypeNum))
else:
myOutType=opts.type
# create file
myOutDrv = gdal.GetDriverByName(opts.format)
myOut = myOutDrv.Create(
opts.outF, DimensionsCheck[0], DimensionsCheck[1], allBandsCount,
gdal.GetDataTypeByName(myOutType), opts.creation_options)
# set output geo info based on first input layer
myOut.SetGeoTransform(myFiles[0].GetGeoTransform())
myOut.SetProjection(myFiles[0].GetProjection())
if opts.NoDataValue!=None:
myOutNDV=opts.NoDataValue
else:
myOutNDV=DefaultNDVLookup[myOutType]
for i in range(1,allBandsCount+1):
myOutB=myOut.GetRasterBand(i)
myOutB.SetNoDataValue(myOutNDV)
# write to band
myOutB=None
if opts.debug:
print("output file: %s, dimensions: %s, %s, type: %s" %(opts.outF,myOut.RasterXSize,myOut.RasterYSize,myOutType))
################################################################
# find block size to chop grids into bite-sized chunks
################################################################
# use the block size of the first layer to read efficiently
myBlockSize=myFiles[0].GetRasterBand(myBands[0]).GetBlockSize();
# store these numbers in variables that may change later
nXValid = myBlockSize[0]
nYValid = myBlockSize[1]
# find total x and y blocks to be read
nXBlocks = (int)((DimensionsCheck[0] + myBlockSize[0] - 1) / myBlockSize[0]);
nYBlocks = (int)((DimensionsCheck[1] + myBlockSize[1] - 1) / myBlockSize[1]);
myBufSize = myBlockSize[0]*myBlockSize[1]
if opts.debug:
print("using blocksize %s x %s" %(myBlockSize[0], myBlockSize[1]))
# variables for displaying progress
ProgressCt=-1
ProgressMk=-1
ProgressEnd=nXBlocks*nYBlocks*allBandsCount
################################################################
# start looping through each band in allBandsCount
################################################################
for bandNo in range(1,allBandsCount+1):
################################################################
# start looping through blocks of data
################################################################
# loop through X-lines
for X in range(0,nXBlocks):
# in the rare (impossible?) case that the blocks don't fit perfectly
# change the block size of the final piece
if X==nXBlocks-1:
nXValid = DimensionsCheck[0] - X * myBlockSize[0]
myBufSize = nXValid*nYValid
# find X offset
myX=X*myBlockSize[0]
# reset buffer size for start of Y loop
nYValid = myBlockSize[1]
myBufSize = nXValid*nYValid
# loop through Y lines
for Y in range(0,nYBlocks):
ProgressCt+=1
if 10*ProgressCt/ProgressEnd%10!=ProgressMk and not opts.quiet:
ProgressMk=10*ProgressCt/ProgressEnd%10
from sys import version_info
if version_info >= (3,0,0):
exec('print("%d.." % (10*ProgressMk), end=" ")')
else:
exec('print 10*ProgressMk, "..",')
# change the block size of the final piece
if Y==nYBlocks-1:
nYValid = DimensionsCheck[1] - Y * myBlockSize[1]
myBufSize = nXValid*nYValid
# find Y offset
myY=Y*myBlockSize[1]
# create empty buffer to mark where nodata occurs
myNDVs = None
# make local namespace for calculation
local_namespace = {}
# fetch data for each input layer
for i, Alpha in enumerate(myAlphaList):
# populate lettered arrays with values
if allBandsIndex is not None and allBandsIndex==i:
myBandNo=bandNo
else:
myBandNo=myBands[i]
myval=gdalnumeric.BandReadAsArray(myFiles[i].GetRasterBand(myBandNo),
xoff=myX, yoff=myY,
win_xsize=nXValid, win_ysize=nYValid)
# fill in nodata values
if myNDV[i] is not None:
if myNDVs is None:
myNDVs = numpy.zeros(myBufSize)
myNDVs.shape=(nYValid,nXValid)
myNDVs=1*numpy.logical_or(myNDVs==1, myval==myNDV[i])
# add an array of values for this block to the eval namespace
local_namespace[Alpha] = myval
myval=None
# try the calculation on the array blocks
try:
myResult = eval(opts.calc, global_namespace, local_namespace)
except:
print("evaluation of calculation %s failed" %(opts.calc))
raise
if myNDVs is not None:
myResult[numpy.isclose(myResult, -99999, atol=1e-5)] = myOutNDV
elif not isinstance(myResult, numpy.ndarray):
myResult = numpy.ones( (nYValid,nXValid) ) * myResult
# write data block to the output file
myOutB=myOut.GetRasterBand(bandNo)
gdalnumeric.BandWriteArray(myOutB, myResult, xoff=myX, yoff=myY)
if not opts.quiet:
print("100 - Done")
#print("Finished - Results written to %s" %opts.outF)
return
################################################################
def Calc(calc, outfile, NoDataValue=None, type=None, format='GTiff', creation_options=[], allBands='', overwrite=False, debug=False, quiet=False, **input_files):
""" Perform raster calculations with numpy syntax.
Use any basic arithmetic supported by numpy arrays such as +-*\ along with logical
operators such as >. Note that all files must have the same dimensions, but no projection checking is performed.
Keyword arguments:
[A-Z]: input files
[A_band - Z_band]: band to use for respective input file
Examples:
add two files together:
Calc("A+B", A="input1.tif", B="input2.tif", outfile="result.tif")
average of two layers:
Calc(calc="(A+B)/2", A="input1.tif", B="input2.tif", outfile="result.tif")
set values of zero and below to null:
Calc(calc="A*(A>0)", A="input.tif", A_Band=2, outfile="result.tif", NoDataValue=0)
"""
opts = Values()
opts.input_files = input_files
opts.calc = calc
opts.outF = outfile
opts.NoDataValue = NoDataValue
opts.type = type
opts.format = format
opts.creation_options = creation_options
opts.allBands = allBands
opts.overwrite = overwrite
opts.debug = debug
opts.quiet = quiet
doit(opts, None)
def store_input_file(option, opt_str, value, parser):
if not hasattr(parser.values, 'input_files'):
parser.values.input_files = {}
parser.values.input_files[opt_str.lstrip('-')] = value
def main():
usage = """usage: %prog --calc=expression --outfile=out_filename [-A filename]
[--A_band=n] [-B...-Z filename] [other_options]"""
parser = OptionParser(usage)
# define options
parser.add_option("--calc", dest="calc", help="calculation in gdalnumeric syntax using +-/* or any numpy array functions (i.e. log10())", metavar="expression")
# limit the input file options to the ones in the argument list
given_args = set([a[1] for a in sys.argv if a[1:2] in AlphaList] + ['A'])
for myAlpha in given_args:
parser.add_option("-%s" % myAlpha, action="callback", callback=store_input_file, type=str, help="input gdal raster file, you can use any letter (A-Z)", metavar='filename')
parser.add_option("--%s_band" % myAlpha, action="callback", callback=store_input_file, type=int, help="number of raster band for file %s (default 1)" % myAlpha, metavar='n')
parser.add_option("--outfile", dest="outF", help="output file to generate or fill", metavar="filename")
parser.add_option("--NoDataValue", dest="NoDataValue", type=float, help="output nodata value (default datatype specific value)", metavar="value")
parser.add_option("--type", dest="type", help="output datatype, must be one of %s" % list(DefaultNDVLookup.keys()), metavar="datatype")
parser.add_option("--format", dest="format", default="GTiff", help="GDAL format for output file (default 'GTiff')", metavar="gdal_format")
parser.add_option(
"--creation-option", "--co", dest="creation_options", default=[], action="append",
help="Passes a creation option to the output format driver. Multiple "
"options may be listed. See format specific documentation for legal "
"creation options for each format.", metavar="option")
parser.add_option("--allBands", dest="allBands", default="", help="process all bands of given raster (A-Z)", metavar="[A-Z]")
parser.add_option("--overwrite", dest="overwrite", action="store_true", help="overwrite output file if it already exists")
parser.add_option("--debug", dest="debug", action="store_true", help="print debugging information")
parser.add_option("--quiet", dest="quiet", action="store_true", help="suppress progress messages")
(opts, args) = parser.parse_args()
if not hasattr(opts, "input_files"):
opts.input_files = {}
if len(sys.argv) == 1:
parser.print_help()
sys.exit(1)
elif not opts.calc:
print("No calculation provided. Nothing to do!")
parser.print_help()
sys.exit(1)
elif not opts.outF:
print("No output file provided. Cannot proceed.")
parser.print_help()
sys.exit(1)
else:
try:
doit(opts, args)
except IOError as e:
print(e)
sys.exit(1)
if __name__ == "__main__":
main()