# Batch Processing¶

PDAL doesn’t handle matching multiple file inputs except for glob handling for merge operations, but does allow for command line substitution parameters to make batch processing simpler, substitutions. Substitions work with both Pipeline operations as well as with other applications such as translate.

## Operating system variations¶

How substitutions are passed generally depends on the operating system and tools available. In the unix/linux environments, this is primarily using the find and ls programs to get lists of files (either with directories or just filenames) and the xargs or parallel program to pass those files to the pdal application (although -exec with find can also be used). These tools are available in the docker environment if you are running PDAL under docker. They are also available under Windows one installs Cygwin or MinGW. They are also available if Git for Windows is installed. They are also available as win32 command line programs installed from the GNU Findutils (https://www.gnu.org/software/findutils/findutils.html). They are available for MacOS and Linux.

## Windows native tools¶

Subtitions can be handled directly in windows using PowerShell syntax.

While there are a number of ways to generate lists of files, the Get-ChildItem is used here, along with the foreach option to pass each separate filepath to the pdal application.

## Example - Batch compression of LAS files to LAZ - PowerShell:¶

To compress a series of LAS files in one directory into compressed LAZ files in another directory, the PowerShell syntax would be:

Get-ChildItem .\DIR1\*.las | foreach {pdal translate -i .\DIR1\$($_.BaseName).las ^
-o .\DIR2\$($_.BaseName).laz}


Note the use of the $($_.BaseName) syntax for the files passed. This option on the $($_) shortcut for the full filename, removes the directory and the extension on the file and allows the user to set the path and extension manually.

## Example - Parallel Batch compression of LAS files to LAZ - PowerShell:¶

This use of the PowerShell syntax doesn’t allow a user to execute more than one process at a time. There is a free download of the xargs program that provides parallel execution available at http://www.pirosa.co.uk/demo/wxargs/ppx2.exe. For this tool, the file names are passed with using the {} syntax.

Get-ChildItem .\dir1\ | Select-Object -ExpandProperty BaseName ^
| .\ppx2.exe -P 3 pdal translate -i ".\dir1\{}.las" -o ".\dir2\{}.laz"


## Example - Batch compression of LAS files to LAZ - Bash:¶

To compress a series of LAS files in one directory into compressed LAZ files in another directory, the Bash syntax would be:

ls ./dir1/*.las | parallel -I{} \
pdal translate -i ./dir1/{/.}.las -o ./dir2/{/.}.laz


In Parallel, then {/.} syntax means strip the directory and the extension and just use the basename of the file. This allows you to easily change the output format and the location.

## Example - Parallel Batch compression of LAS files to LAZ - Bash:¶

Parallel, as its name implies, allows paralell operations. Adding the -j syntax indicates the number simultaneous jobs to run

ls ./dir1/*.las | parallel -I{} -j 4 \
pdal translate -i ./dir1/{/.}.las -o ./dir2/{/.}.laz


## Exercise - Pipeline Substitions:¶

For the most flexibility, pipelines are used to apply a series of opertations to a file (or group of files). In this excersise, we build on the Generating a DTM pipeline example, but run this pipline over 4 files and reproject, calculate a bare earth using the filters.smrf filter, remove those points that aren’t bare earth with filters.range and then write the output using the writers.gdal.

The pipeline we are using is:

{
"pipeline": [
{
},
{
"type": "filters.reprojection"
},
{
"type": "filters.smrf"
},
{
"type":"filters.range",
"limits":"Classification[2:2]"
},
{
"gdaldriver":"GTiff",
"output_type":"idw",
"resolution" :"2.0",
"type": "writers.gdal"
}
]
}


You might have spotted that this pipeline doesn’t have any input or output file references, or a value for the output spatial reference. We will be adding those at the command line, not within the actual pipeline and using the substitutions syntax to do this.

PS ./exercises/batch> Get-ChildItem  ./exercises/batch/source/*.laz | ^
foreach {pdal pipeline ./exercises/batch/batch_srs_gdal.json ^
--readers.las.filename=./source/$($_.BaseName).laz ^
--writers.gdal.filename=./dtm/$($_.BaseName).tif ^
--filters.reprojection.in_srs=epsg:3794 ^
--filters.reprojection.in_srs=epsg:32733}

ls ./exercises/batch_processing/source/*.laz | \
parallel -I{} pdal pipeline ./exercises/batch_processing/batch_srs_gdal.json \

gdalbuildvrt  ./exercises/batch_processing/dtm.vrt ./exercises/batch_processing/dtm*.tif