Is there a way to perform the same task as the gdalbuildvrt utility using GDAL Python bindings? So far I have not found any way to do this other than creating a vrt of a single dataset and manually editing the xml. I would like to create a vrt from multiple rasters (essentially performing a mosaic). Is this possible using pure Python? My other option is to use subprocess to simply call gdalbuildvrt.
3 Answers
The answer of @rcoup only worked for me, if modify it as follows:
from osgeo import gdal
vrt_options = gdal.BuildVRTOptions(resampleAlg='cubic', addAlpha=True)
my_vrt = gdal.BuildVRT('my.vrt', ['one.tif', 'two.tif'], options=vrt_options)
my_vrt = None
Otherwise, the file is not written to disk.
- 351
- 3
- 3
Note that as of 2021, this answer below is now the "offical" way to do this, and directly supported within the GDAL Python Bindings. It is no longer necessary to manually generate the VRT.
Honestly it's easier to do this by using gdalbuildvrt in a subprocess or os.system.
Should you wish to do this through Python it can be done. Using the standard dataset creation methods within GDAL Python we can easily create the base dataset VRT.
from osgeo import gdal
drv = gdal.GetDriverByName("VRT")
vrt = drv.Create("test.vrt", x_size, y_size, 0)
Note that we are creating the dataset with no bands initially. From the documentation on VRTs that VRT datasets are one of the few dataset types that can accept AddBand arguments.
vrt.AddBand(gdal.GDT_Float32)
band = vrt.GetRasterBand(1)
Now for each band we have to set the metadata items manually:
simple_source = '<SourceFilename relativeToVRT="1">%s</SourceFilename>' % source_path + \
'<SourceBand>%i</SourceBand>' % source_band + \
'<SourceProperties RasterXSize="%i" RasterYSize="%i" DataType="Real" BlockXSize="%i" BlockYSize="%i"/>' % (x_size, y_size, x_block, y_block) + \
'<SrcRect xOff="%i" yOff="%i" xSize="%i" ySize="%i"/>' % (x_offset, y_offset, x_source_size, y_source_size) + \
'<DstRect xOff="%i" yOff="%i" xSize="%i" ySize="%i"/>' % (dest_x_offset, dest_y_offset, x_dest_size, y_dest_size)
band.SetMetadataItem("SimpleSource", simple_source)
band.SetMetadataItem("NoDataValue", -9999)
SetMetadatItem takes two arguments, the first a string of the metadata item, the second the item itself. This means that you can't subset a metadata item, so for data sources you have to set the entire contents as a string.
Note that we can use this method to create complex sources (ComplexSource) that contain look-up-tables of values, Kernel filter sources (KernelFilteredSource) of arbitrary sizes and shapes, and Mask Bands (MaskBand).
- 115
- 4
- 15,642
- 2
- 46
- 86
-
Thank you @om_henners - I ended up using subprocess to call gdalbuildvrt. I am more experienced with Python rather than command line so I was hoping I could do this directly in Python, but its not worth the trouble of messing with creating XML strings as you described. It's certainly good to know that I can do that if need in the future though. – Brian Dec 17 '12 at 03:19
-
Just found a use-case for having a python equivalent: adding unsupported features. For example the vrt file format supports an
overviewselement, but gdalbuildvrt doesn't use it. Thanks for providing a stub how this might be added in python. – matt wilkie Feb 04 '14 at 20:30 -
@om_henners is there any way to drv.CreateCopy('path/to/file.vrt',input_ds) with absolute path to input_ds file in python? there is relativeToVRT="1" option, but how change it or set while createing VRT? – Dmitriy Litvinov Mar 21 '17 at 07:03
Since GDAL 2.1 the CLI tools are available as library functions, and in fact that's what the CLI tools now call internally.
For example:
gdalbuildvrt -r cubic -addalpha my.vrt one.tif two.tif
Is the equivalent of:
from osgeo import gdal
vrt_options = gdal.BuildVRTOptions(resampleAlg='cubic', addAlpha=True)
gdal.BuildVRT('my.vrt', ['one.tif', 'two.tif'], options=vrt_options)
The available CLI options directly map to the parameters of BuildVRTOptions, plus there's some extras like progress callbacks available.
- 741
- 8
- 12
-
This appears to be for only certain CLI tools. For example, I'm trying to get gdaladdo to work but it doesn't show up. Same with gdalwarp. Do you know if they plan to support these as well? Would be very helpful . – franchyze923 Mar 30 '20 at 19:16
-
@fpolig01 most of them are there - see
RegenerateOverviews()andWarp()in the API reference. Arguments generally match the CLI commands. – rcoup Mar 30 '20 at 23:28 -
@rccoup Thanks for the reply. Is RegenerateOverviews() the same as gdaladdo? Do you have an example of it working? I'm trying to do something similar to
gdaladdo -r average "D:\image.tif"
– franchyze923 Mar 31 '20 at 15:04 -
@fpolig01 this post suggests
BuildOverviews()(which is actually what I went searching for when I foundRegenerateOverviews) — maybe give that a try? – rcoup Apr 01 '20 at 20:12 -
For me two things were required to make
BuildVRTwork: 1. all paths (inputs and outputs) need to be resolved, 2. (in jupyter notebook) the result ofBuildVRTneeded to be assigned to a variable – Георгий Савельев Sep 28 '23 at 10:48
closingfor python, you have to bring yourvrtout of scope, by assign it toNone. – JensL Apr 02 '19 at 06:52gdal.BuildVRT(dest_path, vsi_hrefs).FlushCache()to force writing the file. – Rob Jul 30 '21 at 17:44