pygeoprocessing


Namepygeoprocessing JSON
Version 2.4.6 PyPI version JSON
download
home_pagehttps://github.com/natcap/pygeoprocessing
SummaryPyGeoprocessing: Geoprocessing routines for GIS
upload_time2024-10-15 19:10:44
maintainerNatural Capital Project Software Team
docs_urlNone
authorNone
requires_python>=3.6
licenseIn this license, "Natural Capital Project" is defined as the parties of the Board of Trustees of the Leland Stanford Junior University, The Nature Conservancy, World Wildlife Fund Inc., and the Regents of the University of Minnesota. This tool has an open license. All people are invited to use the tool under the following conditions and terms: Copyright (c) 2021, Natural Capital Project All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. * Neither the name of Natural Capital Project nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
keywords gis geoprocessing pygeoprocessing
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            .. default-role:: code

About PyGeoprocessing
=====================

PyGeoprocessing is a Python/Cython based library that provides a set of
commonly used raster, vector, and hydrological operations for GIS processing.
Similar functionality can be found in ArcGIS/QGIS raster algebra, ArcGIS zonal
statistics, and ArcGIS/GRASS/TauDEM hydrological routing routines.

PyGeoprocessing is developed at the Natural Capital Project to create a
programmable, open source, and free Python based GIS processing library to
support the InVEST toolset.  PyGeoprocessing's design prioritizes
computation and memory efficient runtimes, easy installation and cross
compatibility with other open source and proprietary software licenses, and a
simplified set of orthogonal GIS processing routines that interact with GIS
data via filename. Specifically the functionally provided by PyGeoprocessing
includes:

* a suite of raster manipulation functions (warp, align, raster calculator,
  reclassification, distance transform, convolution, and fast iteration)
* a suite of vector based manipulation function (zonal statistics,
  rasterization, interpolate points, reprojection, and disjoint polygon sets)
* a simplified hydrological routing library (D8inf/MFD flow direction,
  plateau drainage, weighted and unweighted flow accumulation, and weighted
  and unweighted flow distance)

Installing PyGeoprocessing
==========================

.. code-block:: console

    $ pip install pygeoprocessing

If you `import pygeoprocessing` and see a `ValueError: numpy.dtype has the
wrong size, try recompiling`, this is the result of a version compatibility
issue with the numpy API in the precompiled pygeoprocessing binaries.
The solution is to recompile pygeoprocessing on your computer:

.. code-block:: console

    $ pip uninstall -y pygeoprocessing
    $ pip install pygeoprocessing --no-deps --no-binary :all:

Requirements
------------

Note the pip-installable requirements in `requirements.txt` are for best
results, but older package versions may also work. If necessary,
PyGeoprocessing can be installed without dependencies with `pip install
--no-deps`.

API Documentation
=================

API documentation is available at https://pygeoprocessing.readthedocs.io/en/latest/

Release History
===============

.. Unreleased Changes

2.4.6 (2024-10-15)
------------------
* Removing the ``numpy<2`` constraint for requirements.txt that should have
  been included in the 2.4.5 release. https://github.com/natcap/pygeoprocessing/issues/396
* Handling GDAL-based ``RuntimeError``s raised during ``pygeoprocessing.reproject_vector``.
  https://github.com/natcap/pygeoprocessing/issues/409

2.4.5 (2024-10-08)
------------------
* Updating for numpy 2.0 API changes.  Pygeoprocessing is now compatible with
  numpy 2.0 and later.  https://github.com/natcap/pygeoprocessing/issues/396

2.4.4 (2024-05-21)
------------------
* Our github actions for building python distributions now use
  `actions/setup-python@v5`, which uses node 20.
  https://github.com/natcap/pygeoprocessing/issues/384
* ``warp_raster`` and ``build_overviews`` no longer raise a ``ValueError`` if
  called with an invalid resampling algorithm. We now fall back to the
  underlying GDAL functions' error messages.
  https://github.com/natcap/pygeoprocessing/issues/387
* Implementing decaying flow accumulation for D8 routing.
  https://github.com/natcap/pygeoprocessing/issues/386
* Updated to Cython 3.
* Dropped support for Python 3.7.
* Enabled GDAL exceptions (``gdal.UseExceptions``) throughout pygeoprocessing.
  This is done with a context manager so should not affect other code or
  libraries. https://github.com/natcap/pygeoprocessing/issues/391

2.4.3 (2024-03-06)
------------------
* Wheels for python 3.12 are now built during our github actions runs.
  https://github.com/natcap/pygeoprocessing/issues/381
* ``get_gis_type`` can accept a path to a remote file, allowing the GDAL driver
  to open it if the driver supports the protocol.
  https://github.com/natcap/pygeoprocessing/issues/375
* If running on a SLURM system (identified by the presence of ``SLURM*``
  environment variables), the GDAL cache max is checked against the amount of
  memory available on the compute node.  If GDAL may exceed the available slurm
  memory, a warning is issued or logged.
  https://github.com/natcap/pygeoprocessing/issues/361
* Fixed an issue in ``extract_strahler_streams_d8`` where a nodata pixel
  could be mistakenly treated as a stream seed point, ultimately creating
  a stream feature with no geometry.
  https://github.com/natcap/pygeoprocessing/issues/361
* Improved ``align_and_resize_raster_stack`` so that rasterization of a vector
  mask only happens once, regardless of the number of rasters in the stack.
  In addition, the created mask raster's path may be defined by the caller so
  that it persists across calls to ``align_and_resize_raster_stack``.
  https://github.com/natcap/pygeoprocessing/issues/366
* Improved ``warp_raster`` to allow for a pre-defined mask raster to be
  provided instead of a vector.  If both are provided, the mask raster alone is
  used.  The new mask raster must have the same dimensions and geotransform as
  the output warped raster. https://github.com/natcap/pygeoprocessing/issues/366
* Fixed a bug in ``zonal_statistics`` where the wrong number of disjoint
  polygon sets were being reported in the logs.
  https://github.com/natcap/pygeoprocessing/issues/368
* Pygeoprocessing is now tested against python 3.12.
  https://github.com/natcap/pygeoprocessing/issues/355

2.4.2 (2023-10-24)
------------------
* Fixed an issue where MFD flow direction was producing many nodata holes given
  a large-enough DEM.  These nodata holes would then propagate to flow
  accumulation and stream extraction, producing very disjointed stream
  networks. https://github.com/natcap/pygeoprocessing/issues/350
* Improved progress logging in MFD flow direction, MFD flow accumulation, MFD
  stream extraction to prevent integer overflows in percentages and improve
  the readability of log messages. https://github.com/natcap/pygeoprocessing/issues/246
* Fixed an issue introduced in 2.4.1 where ``zonal_statistics`` results could
  vary depending on the bounding boxes of the disjoint polygon sets calculated
  from the input vector.
* Added a new function, ``pygeoprocessing.align_bbox``, which pads a bounding
  box to align with the grid of a given geotransform.
* Added support for int64 and uint64 (GDAL 3.5+)
  https://github.com/natcap/pygeoprocessing/issues/352
* Added support for signed bytes (GDAL 3.7+)
  https://github.com/natcap/pygeoprocessing/issues/329
* Fixed a bug where ``calculate_slope`` did not work as expected when the DEM
  raster's nodata value is NaN (https://github.com/natcap/pygeoprocessing/issues/352)

2.4.1 (2023-09-05)
------------------
* The ``pygeoprocessing`` package metadata has been updated to use
  ``importlib.metadata`` (python >= 3.8) or ``importlib_metadata``
  (python < 3.8) for retrieving the package version, in keeping with
  recommendations from ``setuptools_scm``.  The dependency
  ``importlib_metadata`` is now required for installs on python < 3.8.
* Fixed another memory leak in ``pygeoprocessing.raster_calculator``, where
  shared memory objects under certain circumstances were not being unlinked at
  the end of the function, resulting in excess memory usage and a warning
  during the shutdown of the python process.
  https://github.com/natcap/pygeoprocessing/issues/247
* Added a new function, ``pygeoprocessing.array_equals_nodata``, which returns
  a boolean array indicating which elements have nodata. It handles integer,
  float, and ``nan`` comparison, and the case where the nodata value is `None`.
* Standardized the approach used in ``warp_raster`` and
  ``create_raster_from_bounding_box`` for determining the dimensions of the
  target raster given a target bounding box and pixel sizes.
  https://github.com/natcap/pygeoprocessing/issues/321
* ``pygeoprocessing.routing.delineate_watersheds_d8`` now handles the case
  where the input flow direction raster does not have a defined spatial
  reference. https://github.com/natcap/pygeoprocessing/issues/254
* Updating internal documentation describing TauDEM flow directions, and adding
  for how to convert from a flow direction raster from what TauDEM expects to
  what pygeoprocessing expects.
  https://github.com/natcap/pygeoprocessing/issues/255
* Users may now specify the overview level to use when calling ``warp_raster``.
  By default, ``pygeoprocessing`` will use the base layer.
  https://github.com/natcap/pygeoprocessing/issues/326
* Fixed a bug across ``pygeoprocessing`` where some valid resampling methods
  would throw an exception because they were not recognized.  This was only
  happening when ``pygeoprocessing`` was installed alongside GDAL < 3.4.
* Fixing an issue with ``pygeoprocessing.multiprocessing.raster_calculator``
  where the function would raise an Exception when the target raster path was
  provided as a filename only, not within a directory, even though the parent
  directory could be inferred. https://github.com/natcap/pygeoprocessing/issues/313
* Fixing a bug where the statistics worker in
  ``pygeoprocessing.raster_calculator`` may return a complex value. This is
  only an issue when pygeoprocessing is compiled against Cython 3.0.0 and
  later. https://github.com/natcap/pygeoprocessing/issues/342
* Added the new function ``pygeoprocessing.raster_map``, a higher-level version
  of ``raster_calculator``. https://github.com/natcap/pygeoprocessing/issues/235

2.4.0 (2023-03-03)
------------------
* A new submodule, ``pygeoprocessing.kernels`` has been added to facilitate the
  creation of kernel rasters needed for calls to
  ``pygeoprocessing.convolve_2d``. Functions for creating common decay kernels
  have been added, along with functions to facilitate the creation of
  distance-based kernels using a user-defined function
  (``pygeoprocessing.create_kernel``) and to facilitate the creation of kernels
  using custom 2D numpy arrays, such as those commonly used in image processing
  (``pygeoprocessing.kernel_from_numpy_array``).
  https://github.com/natcap/pygeoprocessing/issues/268
* Logging across functions in ``pygeoprocessing.geoprocessing`` now correctly
  reports the function that it's being called from rather than ``<lambda>``.
  https://github.com/natcap/pygeoprocessing/issues/300
* The function ``pygeoprocessing.reproject_vector`` now accepts an optional
  parameter ``layer_name`` to allow the target vector layer name to be defined
  by the user.  If the user does not provide a ``layer_name``, the layer name
  will be copied from the source vector.
  https://github.com/natcap/pygeoprocessing/issues/301
* Implement the proposed new function ``pygeoprocessing.raster_reduce``, a
  wrapper around ``pygeoprocessing.iterblocks``
  (https://github.com/natcap/pygeoprocessing/issues/285)
* Nodata value checking in ``pygeoprocessing.routing`` now correctly handles
  comparison of ``nan`` values.  This is explicitly tested in
  ``pygeoprocessing.routing.fill_pits``, but should also improve the
  experience of other routing functions as well.
  https://github.com/natcap/pygeoprocessing/issues/248
* Added a function to build overviews for a raster,
  ``pygeoprocessing.build_overviews``. Related to this,
  ``pygeoprocessing.get_raster_info()`` now includes an ``'overviews'`` key
  listing the pixel dimensions of each overview layer in a raster.
  https://github.com/natcap/pygeoprocessing/issues/280
* Added a D8 stream extraction function at
  ``pygeoprocessing.routing.extract_streams_d8`` which takes a D8 flow
  accumulation raster and a flow accumulation threshold, setting all pixels
  with accumulation above that threshold to 1 and all other valid pixels to 0.
  https://github.com/natcap/pygeoprocessing/issues/272
* Adding a new function, ``pygeoprocessing.create_raster_from_bounding_box``,
  that enables the creation of a new raster from a bounding box.
  https://github.com/natcap/pygeoprocessing/issues/276
* Win32 wheels of PyGeoprocessing are no longer created through our GitHub
  Actions workflows and will no longer be produced or distributed as part of
  our release checklist.  For details (and metrics!) see:
  https://github.com/natcap/pygeoprocessing/issues/232

2.3.5 (2022-12-13)
------------------
* ``pygeoprocessing.calculate_disjoint_polygon_set`` now offers an optional
  parameter, ``geometries_may_touch`` for cases where geometries are known to
  have intersecting boundaries but nonintersecting interiors.
  https://github.com/natcap/pygeoprocessing/issues/269
* Pygeoprocessing is now tested against Python 3.11.
* Adding the target filename to progress logging in
  ``pygeoprocessing.raster_calculator``.
* ``pygeoprocessing.zonal_statistics`` will now optionally include a count of
  the number of pixels per value encountered under each polygon. A warning
  will be logged when invoked on floating-point rasters, as using this on
  continuous rasters can result in excessive memory consumption. To use this
  feature, set ``include_value_counts=True`` when calling ``zonal_statistics``.
* ``pygeoprocessing.get_gis_type`` will now raise a ``ValueError`` if the file
  cannot be opened as ``gdal.OF_RASTER`` or ``gdal.OF_VECTOR``.
  https://github.com/natcap/pygeoprocessing/issues/244
* Fixing an error message in ``convolve_2d`` when the signal or kernel is a
  row based blocksize. https://github.com/natcap/pygeoprocessing/issues/228.

2.3.4 (2022-08-22)
------------------
* Fixing an issue with imports at the ``pygeoprocessing`` module level that was
  causing linters like PyLint and IDE command-completion programs like JEDI-vim
  to not be able to identify the attributes of the ``pygeoprocessing`` module
  namespace.
* Reducing the amount of memory used by
  ``pygeoprocessing.new_raster_from_base`` when filling an array with values to
  only as much memory as is needed for the datatype.
* Fixing a memory leak in ``pygeoprocessing.raster_calculator`` where
  shared memory objects were being inadvertently created when they should not
  have been and then they were not subsequently destroyed.
* ``calculate_disjoint_polygon_set`` will now skip over empty geometries.
  Previously, the presence of empty geometries would cause an error to be
  raised.
* Fixed a ``DeprecationWarning`` in ``calculate_disjoint_polygon_set`` caused
  by the use of a deprecated logging ``warn`` method.

2.3.3.post0 (2022-01-28)
------------------------
* Post-release due to corrupted sdist released on Github and PyPI. The sdist
  for 2.3.3 failed to install. The wheels were unaffected. No code changes.

2.3.3 (2022-01-26)
------------------
* Fixing a bug in ``shapely_geometry_to_vector`` where a feature name mismatch
  between the ``fields`` and ``attribute_list`` inputs would silently pass
  under most circumstances.  Now an informative ``ValueError`` is raised.
* Testing against Python 3.10.
* Pinned ``numpy`` versions in ``pyproject.toml`` to the lowest compatible
  version for each supported python version. This prevents issues when
  ``pygeoprocessing`` is used in an environment with a lower numpy version
  than it was built with (https://github.com/cython/cython/issues/4452).

2.3.2 (2021-09-08)
------------------
* Restore functionality in ``reclassify_raster`` that allows for nodata
  values to be reclassified. This was accidentally removed in 2.3.1.

2.3.1 (2021-08-24)
------------------
* Slightly change the error message displayed for a
  ``ReclassificationMissingValuesError``.
* Handle float nodata values correctly in ``reclassify_raster``.
* Fix a bug in ``distance_to_channel_mfd`` where pixels that only partially
  drained to a stream were given incorrect distances.
* Fixed an issue with ``raster_band_percentile`` that would raise an
  exception if an input raster had an undefined nodata value and/or
  would cause an invalid result if the raster contained non-finite
  values.
* After many years with the Natural Capital Project, Rich Sharp has stepped
  down from the Project and as the maintainer of ``pygeoprocessing``.  James
  Douglass is taking his place, and this change is now reflected in
  ``setup.py``.

2.3.0 (2021-06-21)
------------------
* Added a ``single_outlet_tuple`` parameter to ``routing.fill_pits`` that
  forces a DEM to only have one outlet at any point on the raster. The
  fill effect is that all pixels will drain to the raster coordinate at
  ``single_outlet_tuple``.
* Added a ``detect_lowest_sink_and_drain`` function that finds the lowest
  DEM pixel that drains to nodata/edge and the lowest DEM pixel that could
  be a sink. The values that result from this call can be used to condition
  a DEM that is known to have a single drain using the
  ``single_outlet_tuple`` parameter in ``routing.fill_pits``.
* Fixed a bug in ``routing.fill_pits`` that could cause the nodata region of
  a DEM to be incorrectly filled with non-nodata values.
* Fixed an issue where ``create_raster_from_vector_extent`` would use an
  approximation of the layer extent to infer the input vector extent. Now
  uses the individual geometry to infer vector extents.
* Fixed a bug in ``routing.distance_to_channel_mfd`` where pixels that drained
  off the edge of the raster had negative values rather than nodata values.
* Expanded the error message raised by ``transform_bounding_box`` when the
  bounding box cannot be transformed to provide more helpful details.
* Add support and testing for GDAL 3.3.0.
* Fixed exception occuring in ``stitch_rasters`` when base rasters were
  larger than the target raster on the ``y`` axis.

2.2.0 (2021-05-14)
------------------
* Adding explicit support for Python 3.9 and testing on Python 3.9.
* Fixed an issue in ``create_raster_from_vector_extents`` that would cause a
  confusing exception to be raised if there was no geometry in the vector.
  Now raises a ``ValueError`` with a helpful error message.
* Changed parameters in ``convolve_2d`` to allow API to set
  ``ignore_nodata_and_edges=True`` while ``mask_nodata=False`` and updated
  docstring to indicate this is useful in cases such as filling nodata holes
  in missing datasets. Additionally added a logger ``debug`` message to note
  this "unusual" setting of these parameters in case of accidental usage
  which could be noted during development.
* Fixed issue in zonal statistics that could cause the aggregate vector to be
  locked due to a dereferencing race condition. This was present in some
  cases with a flaky unit test but could have been seen in practice if the
  vector was deleted immediately after the call to ``zonal_statistics``.
* Fixed issue in ``routing.fill_pits`` that used a numerical "is close" test
  when determining if two pixels were at equal height. In the case of
  hydrological pitfilling, pixels must be exactly equal height to be
  considered a plateau otherwise a drain into or out of the pixel is
  resolved. Testing is close allowed a hydrological pit to remain since it
  was "close" to the same height as a draining pixel.
* Added a D8 Strahler order stream extraction function at
  ``pygeoprocessing.extract_strahler_streams_d8``. Creates segmented
  geometric lines which correspond to streams on the landscape.
* Added a D8 subwatershed calculation function at
  ``pygeoprocessing.calculate_subwatershed_boundary``. Creates subwatersheds
  that are segmented at the junctions of the streams created by
  ``pygeoprocessing.extract_strahler_streams_d8``.
* Removing all instances of ``__swig_destroy__`` to prevent multiprocessing
  memory corruption.
* Exposing a ``use_shared_memory`` flag on ``raster_calculator`` to allow
  a user to use shared memory objects when calculating statistics. This
  feature is only available for Python >= 3.8. If available, this
  feature creates a significant runtime improvement but can be unstable
  in multiprocessing configurations. For this reason it is set to
  ``False`` as the default value.
* Added a ``max_timeout`` parameter to ``convolve_2d`` and
  ``raster_calculator`` to allow the user to specify the maximum amount of
  time to wait for worker threads to terminate. In normal operation these
  threads should terminate in a short amount of time but are generously
  timed with the ``pygeoprocessing._MAX_TIMEOUT`` parameter. This parameter
  allows a user to tune in cases that may involve significant latency such
  as in a heavy multiprocess environment.
* Added a ``pygeoprocessing.routing.detect_outlets`` function that creates a
  vector with points placed on the center of the pixels in a D8 or MFD flow
  direction raster that are outlet pixels. The vector has additional metadata
  about the (i, j) locations of the pixel in raster coordinate space.
* Replacing ``merge_rasters`` with ``stitch_rasters`` that can take an
  arbitrary list of raster path bands and stitch them into an existing raster
  of same or different projection. Additional functionality to preserve
  area variant values when stitching from a linear meters projection to
  WGS84. Useful when the units of the base raster are in "units per pixel"
  rather than "units per area".
* Added a ``max_pixel_fill_count`` parameter to ``routing.fill_pits`` to
  guard against pitfilling large natural depression. Defaults to 500.
* Fixed an issue in ``align_and_resize_raster_stack`` that would ignore
  the bounds of a feature in a mask vector if the
  ``"mask_vector_where_filter"`` clause was invoked and instead only
  considered the entire bounds of the vector.
* Fixed an issue with ``convolve_2d`` that allowed output rasters to be
  created without a defined nodata value.
* Fixed a LOGGER message bug that occurred in ``zonal_statistics``.
* Fixed an issue in ``distance_to_channel_mfd`` that would generate a raster
  with distances to the edge of the raster even if there was no channel. Now
  generates nodata so it is consistent with ``distance_to_channel_d8``.
* The ``convolve_2d`` function now raises a ValueError if either the signal
  or kernel raster has a row based blocksize since this could result in
  very long runtimes due to gdal cache thrashing.
* Fixed an issue that would cause a ``convolve_2d`` worker to crash if the
  integrating kernel was not a float and also ``normalize_kernel=True``
  thus causing the entire function call to deadlock.
* Added a check on ``transform_bounding_box`` to ensure the target bounding
  box's coordinates were finite. This guards against cases where a transform
  into another coordinate system creates a degenerate bounding box.
  Previously the function would silently return non-finite coordinates.
* Fixing issue when calculating histogram for floating point rasters the
  logging progress percent would be incorrectly calculated.

2.1.2 (2020-12-03)
------------------
* ``pygeoprocessing.warp_raster`` now raises a ``ValueError`` when an invalid
  resampling method is provided.
* Fixed bug in ``pygeoprocessing.warp_raster`` that would not properly handle
  GDAL Byte type signing when masking warped raster with a vector.
* Fixed issue in ``convolve_2d`` that would cause excessive memory use
  leading to out of memory errors.
* Fixed issue in ``convolve_2d`` that could lead to a file removal race
  condition and raise a ``FileNotFoundException`` when ignoring nodata
  regions.

2.1.1 (2020-09-16)
------------------
* Fixed a critical bug introduced in 2.1.0 that generated invalid results in
  `convolve_2d` for any raster larger than 256x256.
* Added an optional parameter, ``target_layer_name`` to
  ``pygeoprocessing.routing.delineate_watersheds_d8`` for cases where a
  caller would like to define the output layer name.

2.1.0 (2020-08-25)
------------------
* Modified ``pygeoprocessing.numpy_array_to_raster`` to take boolean arrays.
* Modified ``pygeoprocessing.convolve_2d`` to guard against nonsensical queries
  to both ``ignore_nodata_and_edges=True`` but also ``mask_nodata=False``.
  A query of this combination now raises a ``ValueError`` to guard against
  programmer error.
* Added a custom exception class ``ReclassificationMissingValuesError`` to
  ``pygeoprocessing``. ``pygeoprocessing.reclassify_raster`` raises this
  exception instead of ``ValueError`` when a raster pixel value is not
  represented in ``value_map``. This custom exception provides a list of
  missing raster pixel values in a ``missing_values`` attribute that allows
  the caller access to the pixel values that are missing through a Python type
  rather than indirectly through an error message.
* Correcting the docstring for ``pygeoprocessing.numpy_array_to_raster`` to
  specify that the ``pixel_size`` parameter must be a tuple or list, not an
  int.
* ``pygeoprocessing.routing.delineate_watersheds_d8`` now has an optional
  parameter ``write_diagnostic_vector``.  When ``True``, this parameter will
  cause a new vector per outflow feature to be created in the ``working_dir``.
  This parameter defaults to ``False``.  This is a change from prior behavior,
  when the diagnostic vectors were always created, which could occupy a lot of
  computational time under large outflow geometries.
* Added a ``pygeoprocessing.multiprocessing.raster_calculator`` function which
  matches the API and results of ``pygeoprocessing.raster_calculator`` but uses
  multiple processing cores to compute raster calculation blocks.
  when the diagnostic vectors were always created, which could occupy
  significant computational time under large outflow geometries.
* Minor performance improvement to ``pygeoprocessing.convolve_2d`` by
  preventing a pre-processing step that initialized temporary rasters with zero
  values as well as added asynchronous work distribution for kernel/signal
  block processing.
* Modified logging message for ``pygeoprocessing.new_raster_from_base`` when
  filling a raster such that an informative error message is printed with
  context as to the function, file, status, and value being filled.

2.0.0 (05-19-2020)
------------------
* Adding Python 3.8 support and dropping Python 3.6 support.
* Adding GDAL 3 support and dropping GDAL 2 support. The only non-backwards
  compatible issue in GDAL 2 to GDAL 3 is the need to handle Axis Ordering with
  osr.SetAxisMappingStrategy(osr.OAMS_TRADITIONAL_GIS_ORDER) because of
  https://trac.osgeo.org/gdal/wiki/rfc73_proj6_wkt2_srsbarn#Axisorderissues?.
  Since Axis ordering now matters for Geographic CRS the expected order is
  Lat,Lon but we use osr.OAMS_TRADITIONAL_GIS_ORDER to swap to Lon,Lat.
* Using osr.CreateCoordinateTransformation() instead of
  osr.CoordinateTransformation() as the GDAL 3 call.
* Fixed a bug in convolve_2d that would not ``ignore_nodata`` if the signal
  raster's nodata value was undefined. Changed the name of this flag to
  ``ignore_nodata_and_edges`` to reflect its expected functionality.
* Warped signed byte rasters are now also signed byte rasters.
* Adding a GitHub Actions-based build job for building wheels and a source
  distribution for a given commit of pygeoprocessing.
* Updated ``setup.py`` to point the URL project link to the project's new
  home on GitHub.
* Updated ``MANIFEST.in`` to only include files that should be there in the
  pygeoprocessing source distribution.  This fixes an issue where files
  matching a variety of extensions anywhere in the pygeoprocessing directory
  might be included with the source distribution.
* Added ``set_tol_to_zero`` to ``convolve_2d`` to allow for in-function masking
  of near-zero results to be set to 0.0.
* Fixed malformed logging outputs which could be seen during long running
  ``rasterize`` calls.
* Renamed all parameters involving Spatial Projections to the form
  ``[var_id]_projection_wkt``, this involves optional arguments in
  ``reproject_vector``, ``warp_raster``, ``transform_bounding_box``,
  and ``align_and_resize_raster_stack`` as well as the return value from
  ``get_raster_info`` and ``get_vector_info``.
* Fixed an issue in ``zonal_statistics`` that would crash if an aggregate
  vector had a feature with no geometry defined. Now the function ignores
  such features and prints a warning to the log.
* Fixed a malformed ``ValueError`` message when a corrupt raster was
  encountered in ``raster_calculator``.
* Fixes an unnecessary calculation that pre-fills slope raster GeoTIFFs with
  nodata values.
* Added a check to ``convolve_2d`` to verify that raster path/band tuples were
  passed where expected and raise a useful Exception  if not.
* Removed most of the `pygeoprocessing.testing` module and added the following
  convenience functions to ``pygeoprocessing``, which should not be used
  for production code but are useful for testing and scripting:

  * ``raster_to_numpy_array`` - read a single band of a raster into a ``numpy``
    array, runs the risk of memory error if the raster is too large.
  * ``numpy_array_to_raster`` - writes a ``numpy`` array to a raster on disk.
  * ``shapely_geometry_to_vector`` - creates a vector from a list of
    ``Shapely`` geometry.

* Fixed an issue in ``flow_dir_mfd`` that would cause invalid flow directions
  on DEMs that had very small numerical delta heights.
* Fixes an issue in ``convolve_2d`` that would occasionally cause undefined
  numerical noise in regions where the signal was nodata but ``mask_nodata``
  was set to ``False``.

1.9.2 (2020-02-06)
------------------
* Removed the ``multiprocessing`` dependency to avoid an occasional deadlock
  that occurred on Mac OS X during ``align_and_resize_raster_stack``.
  That function now operates serially, but multithreading can be used by
  passing ``gdal_warp_options``.

1.9.1 (2019-12-19)
------------------
* Fixed a compilation issue on Mac OS X Catalina related to the compilation
  of a template in the file iteration component of the out-of-core percentile
  function.
* Resolved a compilation issue on Mac OS X (Mavericks and later) where
  pygeoprocessing would not compile unless some additional compiler and linker
  flags were provided.  These are now accounted for in the package's compilation
  steps in ``setup.py``.
* ``pygeoprocessing.symbolic.evaluate_raster_calculator_expression``
  no longer depends on ``sympy`` for its expression evaluation.

1.9.0 (2019-10-22)
------------------
* Fixed a memory error issue that could occur on multiple flow direction flow
  accumulation calculations.
* Added PEP518-compatible build dependencies to ``pyproject.toml``, which has
  been added to source distributions of pygeoprocessing.
* Added an out-of-core high performance raster percentile function at
  pygeoprocessing.raster_band_percentile.
* Added a ``pygeoprocessing.symbolic.evaluate_raster_calculator_expression``
  function that can parse a string expression and a map of symbols to
  calculate a raster calculation operation.
* Modified ``pygeoprocessing.routing.fill_pits`` to create a single band
  raster of the input DEM raster/path band rather than a copy of the input
  DEM raster irrespective of the band number.
* Fixed an issue that would cause the bounding box generated by
  ```transform_bounding_box`` to flip the min and max coordinates in some
  spatial contexts.
* Defaulting compression algorithm back to LZW as ZSTD is not a widely
  implemented compression standard for geotiffs.

1.8.0 (2019-08-12)
------------------
* Added a ``'numpy_type'`` field to the result of ``get_raster_info`` that
  contains the equivalent numpy datatype of the GDAL type in the raster. This
  includes functionality differentate between the unsigned and signed
  ``gdal.GDT_Byte`` vs. ``numpy.int8`` and ``numpy.uint8``.
* Changed default compression routine for GeoTIFFs to ZSTD (thanks Facebook
  https://facebook.github.io/zstd/).
* Added a **non-backwards compatible change** by replacing the
  ``gtiff_creation_options`` string to a driver/option string named
  ``raster_driver_creation_tuple``. This allows the caller to create any type
  of ``GDAL`` writable driver along with the option list associated with that
  driver.
* Added a ``'numpy_type'`` field to the result of ``get_raster_info`` that
  contains the equivalent numpy datatype of the GDAL type in the raster. This
  includes functionality differentate between the unsigned and signed
  ``gdal.GDT_Byte`` vs. ``numpy.int8`` and ``numpy.uint8``.
* Changed default compression routine for GeoTIFFs to ZSTD (thanks Facebook
  https://facebook.github.io/zstd/).
* Added a ``'file_list'`` key to the dictionary returned by
  ``get_raster_info`` and ``get_vector_info`` that contains a list of all the
  files associated with that GIS object. The first parameter of these lists
  can be passed to ``gdal.OpenEx`` to open the object directly.
* Added a ``get_gis_type`` function to ``pygeoprocessing`` that takes a
  filepath and returns a bitmask of ``pygeoprocessing.RASTER_TYPE`` and/or
  ``pygeoprocessing.VECTOR_TYPE``.
* Modified ``iterblocks`` to raise a helpful ValueError instead of a general
  NoneTypeError if a raster does not open.

1.7.0 (2019-06-27)
------------------
* Removing support for Python 2.7.
* Adding D8 watershed delineation as
  ``pygeoprocessing.routing.delineate_watersheds_d8``.
* Corrected an issue with ``pygeoprocessing.create_raster_from_vector_extents``
  where a vector with no width or no height (a vector with a single point, for
  example) would result in invalid raster dimensions being passed to GDAL.
  These edge cases are now guarded against.
* ``pygeoprocessing.calculate_disjoint_polygon_set`` will now raise
  ``RuntimeError`` if it is passed a vector with no features in it.
* ``pygeoprocessing.rasterize`` will now raise ``RuntimeError`` if the
  underlying call to ``gdal.RasterizeLayer`` encounters an error.
* Correcting an issue with the docstring in
  ``pygeoprocessing.reclassify_raster`` to reflect the current parameters.
* Changed ``zonal_statistics`` to always return a ``dict`` instead of
  sometimes a ``defaultdict``. This allows pickling of the result, if desired.
* Adding automated testing via bitbucket pipelines.
* Correcting an issue with ``pygeoprocessing.zonal_statistics`` that was
  causing test failures on Python 3.6.
* Pygeoprocessing is now tested against Python 3.7.
* Fixed an issue in distance transform where a vertical striping artifact
  would occur in the masked region of some large rasters when distances should
  be 0.
* Fixed an issue in all functionality that used a cutline polygon with
  invalid geometry which would cause a crash. This was caused by `gdal.Warp`
  when using the cutline functionality. Instead this functionality was
  replaced with manual rasterization. In turn this introduces two optional
  parameters:

    * ``rasterize`` and ``mask_raster`` have a ``where_clause`` parameter
      which takes a string argument in SQL WHERE syntax to filter
      rasterization based on attribute values.
    * ``warp_raster`` takes a ``working_dir`` parameter to manage local
      temporary mask rasters.

* Removing a temporary working directory that is created when executing
  pygeoprocessing.convolve_2d.
* Changed optional parameters involving layer indexes to be either indexes
  or string ids. In all cases changing ``layer_index`` to ``layer_id`` in
  the functions: ``get_vector_info``, ``reproject_vector``, ``warp_raster``,
  ``rasterize``, ``calculate_disjoint_polygon_set``, and ``mask_raster``.

1.6.1 (2019-02-13)
------------------
* Added error checking in ``raster_calculator`` to help ensure that the
  ``target_datatype`` value is a valid GDAL type.
* Fixed an issue in ``distance_transform_edt`` that would occasionally
  cause incorrect distance calculations when the x sampling distance was > 1.

1.6.0 (2019-01-23)
------------------
* Changed ``iterblocks`` API to take a raster/path band as an input rather
  than a path and a list of bands. Also removed the ``astype_list`` due to
  its lack of orthogonality.
* Fixed bugs in ``convolve_2d`` involving inputs with nodata masking.
* Changing default raster creation compression algorithm from LZW to DEFLATE,
  this is to address issues where we were seeing recreatable, but
  unexplainable LZWDecode errors in large raster data.
* Fixed an issue that could cause the distance transform to be incorrect when
  the sampling distance was in the noninclusive range of (0.0, 1.0).

1.5.0 (2018-12-12)
------------------
* Specific type checking for ``astype_list`` in ``iterblocks`` to avoid
  confusing exceptions.
* Renamed test suite to be consistent with the pattern
  ``test_[component].tif``.
* Added a function ``pygeoprocessing.routing.extract_streams_mfd`` that
  creates a contiguous stream layer raster to accounts for the divergent flow
  that can occur with multiple flow direction. If the flow direction raster is
  otherwise directly thresholded, small disjoint streams can appear where
  the downstream flow drops below the threshold level.
* Fixed an issue that could cause some custom arguments to geotiff creation
  options to be ignored.
* Added a ``mask_raster`` function that can be used to mask out pixels in
  an existing raster that don't overlap with a given vector.
* Fixed a bug in the ``distance_transform_edt`` function that would cause
  incorrect distances to be calculated in the case of nodata pixels in the
  region raster. The algorithm has been modified to treat nodata as though
  pixel values were 0 (non-region) and the distance transform will be defined
  for the entire raster.
* Added a ``sampling_distance`` parameter to ``distance_transform_edt`` that
  linearly scales the distance transform by this value.
* Fixed an issue in ``calculate_slope`` that would raise an exception if the
  input dem did not have a nodata value defined.
* Changed the behavior of ``zonal_statistics`` for polygons that that do not
  intersect any pixels. These FIDs are now also included in the result from
  ``zonal_statistics`` where previously they were absent. This is to remain
  consistent with how other GIS libraries calculate zonal stats.

1.4.1 (2018-11-12)
------------------
* Hotfix that fixes an issue that would cause ``zonal_statistics`` to crash if
  a polygon were outside of the raster's bounding box.

1.4.0 (2018-11-12)
------------------
* Adding error checking to ensure that ``target_pixel_size`` passed to
  ``warp_raster`` and ``align_and_resize_raster_stack`` are validated to ensure
  they are in the correct format. This solves an issue where an incorrect
  value, such as a single numerical value, resolve into readable exception
  messages.
* Added a ``gdal_warp_options`` parameter to ``align_and_resize_raster_stack``
  and ``warp_raster`` whose contents get passed to gdal.Warp's ``warpOptions``
  parameter. This was implemented to expose the CUTLINE_TOUCH_ALL
  functionality but could be used for any gdal functionality.
* Modified ``rasterize`` API call to make ``burn_values`` and ``option_list``
  both optional parameters, along with error checking to ensure a bad input's
  behavior is understood.
* Exposing GeoTIFF creation options for all the ``pygeoprocessing.routing``
  functions which create rasters. This is consistent with the creation
  options exposed in the main ``pygeoprocessing`` API.
* Removing ``'mean_pixel_size'`` as a return value from ``get_raster_info``,
  this is because this parameter is easily misused and easily calculated if
  needed. This is a "What good programmers need, not what bad programmers
  want." feature.

1.3.1 (2018-10-25)
------------------
* Hotfix to patch an infinite loop when aggregating upstream or downstream
  with custom rasters.

1.3.0 (2018-10-25)
------------------
* Fixed a handful of docstring errors.
* Improved runtime of ``zonal_statistics`` by a couple of orders of magnitude
  for large vectors by using spatial indexes when calculating disjoint polygon
  overlap sets, using database transactions, and memory buffers.
* Improved runtime performance of ``reproject_vector`` by using database
  transactions.
* Improved logging for long runtimes in ``zonal_statistics``.
* Changed ``zonal_statistics`` API and functionality to aggregate across the
  FIDs of the aggregate vector. This is to be consistent with QGIS and other
  zonal statistics functionality. Additionally, fixed a bug where very small
  polygons might not get aggregated if they lie in the same pixel as another
  polygon that does not intersect it. The algorithm now runs in two passes:

    * aggregate pixels whose centers intersect the aggregate polygons
    * any polygons that were not aggregated are geometrically intersected
      with pixels to determine coverage.

* Removed the ``calculate_raster_stats`` function since it duplicates GDAL
  functionality, but with a slower runtime, and now functions in
  ``pygeoprocessing`` that create rasters also calculate stats on the fly if
  desired.
* Fixes an issue in ``get_raster_info`` and ``get_vector_info`` where the path
  to the raster/vector includes non-standard OS pathing (such as a NETCDF),
  info will still calculate info.
* Added functionality to ``align_raster_stack`` and ``warp_raster`` to define
  a base spatial reference system for rasters if not is not defined or one
  wishes to override the existing one. This functionality is useful when
  reprojecting a rasters that does not have a spatial reference defined in the
  dataset but is otherwise known.
* Added a ``weight_raster_path_band`` parameter to both
  ``flow_accumulation_d8`` and ``flow_accumulation_mfd`` that allows the
  caller to use per-pixel weights from a parallel raster as opposed to
  assuming a weight of 1 per pixel.
* Added a ``weight_raster_path_band`` parameter to both
  ``distance_to_channel_mfd`` and ``distance_to_channel_d8`` that allows the
  caller to use per-pixel weights from a parallel raster as opposed to
  assuming a distance of 1 between neighboring pixels or sqrt(2) between
  diagonal ones.
* Added an option to ``reproject_vector`` that allows a caller to specify
  which fields, if any, to copy to the target vector after reprojection.
* Adding a check in ``align_and_resize_raster_stack`` for duplicate target
  output paths to avoid problems where multiple rasters are being warped to
  the same path.
* Created a public ``merge_bounding_box_list`` function that's useful for
  union or intersection of bounding boxes consistent with the format in
  PyGeoprocessing.
* Added functionality in ``align_and_resize_raster_stack`` and ``warp_raster``
  to use a vector to mask out pixel values that lie outside of the polygon
  coverage area. This parameter is called ``vector_mask_options`` and is
  fully documented in both functions. It is similar to the cutline
  functionality provided in ``gdal.Warp``.
* Fixed an issue in the ``flow_accumulation_*`` functions where a weight
  raster whose values were equal to the nodata value of the flow accumulation
  raster OR simply nodata would cause infinite loops.

1.2.3 (2018-07-25)
------------------
* Exposing a parameter and setting reasonable defaults for the number of
  processes to allocate to ``convolve_2d`` and ``warp_raster``. Fixes an issue
  where the number of processes could exponentiate if many processes were
  calling these functions.
* Fixing an issue on ``zonal_statistics`` and ``convolve_2d`` that would
  attempt to both read and write to the target raster with two different GDAL
  objects. This caused an issue on Linux where the read file was not caught up
  with the written one. Refactored to use only one handle.
* Fixing a rare race condition where an exception could occur in
  ``raster_calculator`` that would be obscured by an access to an object that
  had not yet been assigned.
* ``align_and_resize_raster_stack`` now terminates its process pool.
* Increased the timeout in joining ``raster_calculator``'s stats worker.
  On a slow system 5 seconds was not quite enough time.

1.2.2 (2018-07-25)
------------------
* Hotfixed a bug that would cause numpy arrays to be treated as broadcastable
  even if they were passed in "raw".

1.2.1 (2018-07-22)
------------------
* Fixing an issue with ``warp_raster`` that would round off bounding boxes
  for rasters that did not fit perfectly into the target raster's provided
  pixel size.
* Cautiously ``join``\ing all process pools to avoid a potential bug where a
  deamonized subprocess in a process pool may still have access to a raster
  but another process may require write access to it.

1.2.0 (2018-07-19)
------------------
* Several PyGeoprocessing functions now take advantage of multiple CPU cores:

  * ``raster_calculator`` uses a separate thread to calculate raster
    statistics in a ``nogil`` section of Cython code. In timing with a big
    rasters we saw performance improvements of about 35%.
  * ``align_and_resize_raster_stack`` uses as many CPU cores, up to the number
    of CPUs reported by multiprocessing.cpu_count (but no less than 1), to
    process each raster warp while also accounting for the fact that
    ``gdal.Warp`` uses 2 cores on its own.
  * ``warp_raster`` now directly uses ``gdal.Warp``'s multithreading directly.
    In practice it seems to utilize two cores.
  * ``convolve_2d`` attempts to use ``multiprocessing.cpu_count`` cpus to
    calculate separable convolutions per block while using the main thread to
    aggregate  and write the result to the target raster. In practice we saw
    this improve runtimes by about 50% for large rasters.
* Fixed a bug that caused some nodata values to not be treated as nodata
  if there was a numerical roundoff.
* A recent GDAL upgrade (might have been 2.0?) changed the reference to
  nearest neighbor interpolation from 'nearest' to 'near'. This PR changes
  PyGeoprocessing to be consistent with that change.
* ``raster_calculator`` can now also take "raw" arguments in the form of a
  (value, "raw") tuple. The parameter ``value`` will be passed directly to
  ``local_op``. Scalars are no longer a special case and need to be passed as
  "raw" parameters.
* Raising ``ValueError`` in ``get_raster_info`` and ``get_vector_info`` in
  cases where non-filepath non-GIS values are passed as parameters. Previously
  such an error would result in an unhelpful error in the GDAL library.

1.1.0 (2018-07-06)
------------------
* PyGeoprocessing now supports Python 2 and 3, and is tested on python 2.7
  and 3.6  Testing across multiple versions is configured to be run via
  ``tox``.
* After testing (tox configuration included under ``tox-libcompat.ini``),
  numpy requirement has been dropped to ``numpy>=1.10.0`` and scipy has been
  modified to be ``scipy>=0.14.1,!=0.19.1``.
* A dependency on ``future`` has been added for compatibility between python
  versions.
* Fixed a crash in ``pygeoprocessing.routing.flow_dir_mfd`` and
  ``flow_dir_d8`` if a base raster was passed in that did not have a power of
  two blocksize.
* ``raster_calculator`` can now take numpy arrays and scalar values along with
  raster path band tuples. Arrays and scalars are broadcast to the raster size
  according to numpy array broadcasting rules.
* ``align_and_resize_raster_stack`` can now take a desired target projection
  which causes all input rasters to be warped to that projection on output.

1.0.1 (2018-05-16)
------------------
* Hotfix patch to remove upper bound on required numpy version. This was
  causing a conflict with InVEST's looser requirement. Requirement is now
  set to >=1.13.0.

1.0.0 (2018-04-29)
------------------
* This release marks a feature-complete version of PyGeoprocessing with a
  full suite of routing and geoprocessing capabilities.
* ``pygeoprocessing.routing`` module has a ``flow_dir_mfd`` function that
  calculates a 32 bit multiple flow direction raster.
* ``pygeoprocessing.routing`` module has a ``flow_accumulation_mfd`` function
  that uses the flow direction raster from
  ``pygeoprocessing.routing.flow_dir_mfd`` to calculate a per-pixel continuous
  flow accumulation raster.
* ``pygeoprocessing.routing`` module has a ``distance_to_channel_mfd``
  function that calculates distance to a channel raster given a
  pygeoprocessing MFD raster.
* ``pygeoprocessing.routing`` module has a ``distance_to_channel_d8`` function
  that calculates distance to a channel raster given a pygeoprocessing D8
  raster.

0.7.0 (2018-04-18)
------------------
* Versioning is now handled by ``setuptools_scm`` rather than
  ``natcap.versioner``.  ``pygeoprocessing.__version__`` is now fetched from
  the package metadata.
* Raster creation defaults now set "COMPRESS=LZW" for all rasters created in
  PyGeoprocessing, including internal temporary rasters. This option was
  chosen after profiling large raster creation runs on platter hard drives.
  In many cases processing time was dominated by several orders of magnitude
  as a write-to-disk. When compression is turned on overall runtime of very
  large rasters is significantly reduced. Note this otherwise increases the
  runtime small raster creation and processing by a small amount.
* ``pygeoprocessing.routing`` module now has a ``fill_pits``, function which
   fills hydrological pits with a focus on runtime efficiency, memory space
   efficiency, and cache locality.
* ``pygeoprocessing.routing`` module has a ``flow_dir_d8`` that uses largest
  slope to determine the downhill flow direction.
* ``pygeoprocessing.routing`` module has a ``flow_accumulation_d8`` that uses
  a pygeoprocessing D8 flow direction raster to calculate per-pixel flow
  accumulation.
* Added a ``merge_rasters`` function to ``pygeoprocessing`` that will mosaic a
  set of rasters in the same projection, pixel size, and band count.

0.6.0 (2017-01-10)
------------------
* Added an optional parameter to ``iterblocks`` to allow the ``largest_block``
  to be set something other than the PyGeoprocessing default. This in turn
  allows the ``largest_block`` parameter in ``raster_calculator`` to be passed
  through to ``iterblocks``.
* Upgraded PyGeoprocessing GDAL dependency to >=2.0.
* Added a ``working_dir`` optional parameter to ``zonal_statistics``,
  ``distance_transform_edt``, and ``convolve_2d`` which specifies a directory
  in which temporary files will be created during execution of the function.
  If set to ``None`` files are created in the default system temporary
  directory.

0.5.0 (2017-09-14)
------------------
* Fixed an issue where NETCDF files incorrectly raised Exceptions in
  ``raster_calculator``  and ``rasterize`` because they aren't filepaths.
* Added a NullHandler so that users wouldn't get an error that a logger
  handler was undefined.
* Added ``ignore_nodata``, ``mask_nodata``, and ``normalize_kernel`` options
  to ``convolve_2d`` which make this function capable of adapting the nodata
  overlap with the kernel rather than zero out the result, as well as on
  the fly normalization of the kernel for weighted averaging purposes. This
  is in part to make this functionality more consistent with ArcGIS's
  spatial filters.

0.4.4 (2017-08-18)
------------------
* When testing for raster alignment ``raster_calculator`` no longer checks the
  string equality for projections or geotransforms.  Instead it only checks
  raster size equality.  This fixes issues where users rasters DO align, but
  have a slightly different text format of the WKT of projection.  It also
  abstracts the problem of georeferencing away from raster_calculator that is
  only a grid based operation.

0.4.3 (2017-08-16)
------------------
* Changed the error message in ``reclassify_raster`` so it's more informative
  about how many values are missing and the values in the input lookup table.
* Added an optional parameter ``target_nodata`` to ``convolve_2d`` to set the
  desired target nodata value.

0.4.2 (2017-06-20)
------------------
* Hotfix to fix an issue with ``iterblocks`` that would return signed values
  on unsigned raster types.
* Hotfix to correctly cite Natural Capital Project partners in license and
  update the copyright year.
* Hotfix to patch an issue that gave incorrect results in many PyGeoprocessing
  functions when a raster was passed with an NoData value.  In these cases the
  internal raster block masks would blindly pass through on the first row
  since a test for ``numpy.ndarray == None`` is ``False`` and later
  ``x[False]`` is the equivalent of indexing the first row of the array.

0.4.1 (2017-06-19)
------------------
* Non-backwards compatible refactor of core PyGeoprocessing geoprocessing
  pipeline. This is to in part expose only orthogonal functionality, address
  runtime complexity issues, and follow more conventional GIS naming
  conventions. Changes include:

    * Full test coverage for ``pygeoprocessing.geoprocessing`` module
    * Dropping "uri" moniker in lieu of "path".
    * If a raster path is specified and operation requires a single band,
      argument is passed as a "(path, band)" tuple where the band index starts
      at 1 as convention for raster bands.
    * Shapefile paths are assumed to operate on the first layer.  It is so
      rare for a shapefile to have more than one layer, functions that would
      be confused by multiple layers have a layer_index that defaults to 0
      that can be overridden in the call.
    * Be careful, many of the parameter orders have been changed and renamed.
      Generally inputs come first, outputs last.  Input parameters are
      often prefixed with "base\_" while output parameters are prefixed with
      "target\_".
    * Functions that take rasters as inputs must have their rasters aligned
      before the call to that function.  The function
      ``align_and_resize_raster_stack`` can handle this.
    * ``vectorize_datasets`` refactored to ``raster_calculator`` since that
      name is often used as a convention when referring to raster
      calculations.
    * ``vectorize_points`` refactored to meaningful ``interpolate_points``.
    * ``aggregate_by_shapefile`` refactored to ``zonal_statistics`` and now
      returns a dictionary rather than a named tuple.
    * All functions that create rasters expose the underlying GeoTIFF options
      through a default parameter ``gtiff_creation_options`` which default to
      "('TILED=YES', 'BIGTIFF=IF_SAFER')".
    * Individual functions for raster and vector properties have been
      aggregated into ``get_raster_info`` and ``get_vector_info``
      respectively.
    * Introducing ``warp_raster`` to wrap GDAL's ``ReprojectImage``
      functionality that also works on bounding box clips.
    * Removed the ``temporary_filename()`` paradigm.  Users should manage
      temporary filenames directly.
    * Numerous API changes from the 0.3.x version of PyGeoprocessing.
* Fixing an issue with aggregate_raster_values that caused a crash if feature
  IDs were not in increasing order starting with 0.
* Removed "create_rat/create_rat_uri" and migrated it to
  natcap.invest.wind_energy; the only InVEST model that uses that function.
* Fixing an issue with aggregate_raster_values that caused a crash if feature
  IDs were not in increasing order starting with 0.
* Removed "create_rat/create_rat_uri" and migrated it to
  natcap.invest.wind_energy; the only InVEST model that uses that function.

0.3.3 (2017-02-09)
------------------
* Fixing a memory leak with large polygons when calculating disjoint set.

0.3.2 (2017-01-24)
------------------
* Hotfix to patch an issue with watershed delineation packing that causes some
  field values to lose precision due to default field widths being set.

0.3.1 (2017-01-18)
------------------
* Hotfix patch to address an issue in watershed delineation that doesn't pack
  the target watershed output file.  Half the shapefile consists of features
  polygonalized around nodata values that are flagged for deletion, but not
  removed from the file.  This patch packs those features and returns a clean
  watershed.

0.3.0 (2016-10-21)
------------------
* Added ``rel_tol`` and ``abs_tol`` parameters to ``testing.assertions`` to be
  consistent with PEP485 and deal with real world testing situations that
  required an absolute tolerance.
* Removed calls to ``logging.basicConfig`` throughout pygeoprocessing.  Client
  applications may need to adjust their logging if pygeoprocessing's log
  messages are desired.
* Added a flag  to ``aggregate_raster_values_uri`` that can be used to
  indicate incoming polygons do not overlap, or the user does not care about
  overlap. This can be used in cases where there is a computational or memory
  bottleneck in calculating the polygon disjoint sets that would ultimately be
  unnecessary if it is known a priori that such a check is unnecessary.
* Fixed an issue where in some cases different nodata values for 'signal' and
  'kernel' would cause incorrect convolution results in ``convolve_2d_uri``.
* Added functionality to ``pygeoprocessing.iterblocks`` to iterate over
  largest memory aligned block that fits into the number of elements provided
  by the parameter.  With default parameters, this uses a ceiling around 16MB
  of memory per band.
* Added functionality to ``pygeoprocessing.iterblocks`` to return only the
  offset dictionary.  This functionality would be used in cases where memory
  aligned writes are desired without first reading arrays from the band.
* Refactored ``pygeoprocessing.convolve_2d_uri`` to use ``iterblocks`` to take
  advantage of large block sizes for FFT summing window method.
* Refactoring source side to migrate source files from [REPO]/pygeoprocessing
  to [REPO]/src/pygeoprocessing.
* Adding a pavement script with routines to fetch SVN test data, build a
  virtual environment, and clean the environment in a Windows based operating
  system.
* Adding ``transform_bounding_box`` to calculate the largest projected
  bounding box given the four corners on a local coordinate system.
* Removing GDAL, Shapely from the hard requirements in setup.py.  This will
  allow pygeoprocessing to be built by package managers like pip without these
  two packages being installed.  GDAL and Shapely will still need to be
  installed for pygeoprocessing to run as expected.
* Fixed a defect in ``pygeoprocessing.testing.assert_checksums_equal``
  preventing BSD-style checksum files from being analyzed correctly.
* Fixed an issue in reclassify_dataset_uri that would cause an exception if
  the incoming raster didn't have a nodata value defined.
* Fixed a defect in ``pygeoprocessing.geoprocessing.get_lookup_from_csv``
  where the dialect was unable to be detected when analyzing a CSV that was
  larger than 1K in size.  This fix enables the correct detection of comma or
  semicolon delimited CSV files, so long as the header row by itself is not
  larger than 1K.
* Intra-package imports are now relative.  Addresses an import issue for users
  with multiple copies of pygeoprocessing installed across multiple Python
  installations.
* Exposed cython routing functions so they may be imported from C modules.
* ``get_lookup_from_csv`` attempts to determine the dialect of the CSV instead
  of assuming comma delimited.
* Added relative numerical tolerance parameters to the PyGeoprocessing raster
  and csv tests with in the same API style as ``numpy.testing.allclose``.
* Fixed an incomparability with GDAL 1.11.3 bindings that expects a boolean
  type in ``band.ComputeStatistics``.  Before this fix PyGeoprocessing would
  crash with a TypeError on many operations.
* Fixed a defect in pygeoprocessing.routing.calculate_transport where the
  nodata types were cast as int even though the base type of the routing
  rasters were floats.  In extreme cases this could cause a crash on a type
  that could not be converted to an int, like an ``inf``, and in subtle cases
  this would result in nodata values in the raster being ignored during
  routing.
* Added functions to construct raster and vectors on disk from reasonable
  datatypes (numpy matrices for rasters, lists of Shapely geometries for
  vectors).
* Fixed an issue where reproject_datasource_uri would add geometry that
  couldn't be projected directly into the output datasource.  Function now
  only adds geometries that transformed without error and reports if any
  features failed to transform.
* Added file flushing and dataset swig deletion in reproject_datasource_uri to
  handle a race condition that might have been occurring.
* Fixed an issue when "None" was passed in on new raster creation that would
  attempt to directly set that value as the nodata value in the raster.
* Added basic filetype-specific assertions for many geospatial filetypes, and
  tests for these assertions.  These assertions are exposed in
  ``pygeoprocessing.testing``.
* Pygeoprocessing package tests can be run by invoking
  ``python setup.py nosetests``.  A subset of tests may also be run from an
  installed pygeoprocessing distribution by calling
  ``pygeoprocessing.test()``.
* Fixed an issue with reclassify dataset that would occur when small rasters
  whose first memory block would extend beyond the size of the raster thus
  passing in "0" values in the out of bounds area. Reclassify dataset
  identified these as valid pixels, even though vectorize_datsets would mask
  them out later.  Now vectorize_datasets only passes memory blocks that
  contain valid pixel data to its kernel op.
* Added support for very small AOIs that result in rasters less than a pixel
  wide.  Additionally an ``all_touched`` flag was added to allow the
  ALL_TOUCHED=TRUE option to be passed to RasterizeLayer in the AOI mask
  calculation.
* Added watershed delineation routine to
  pygeoprocessing.routing.delineate_watershed.  Operates on a DEM and point
  shapefile, optionally snaps outlet points to nearest stream as defined by a
  thresholded flow accumulation raster and copies the outlet point fields into
  the constructed watershed shapefile.
* Fixing a memory leak in block caches that held on to dataset, band, and
  block references even after the object was destroyed.
* Add an option to route_flux that lets the current pixel's source be included
  in the flux, or not.  Previous version would include on the source no matter
  what.
* Now using natcap.versioner for versioning instead of local versioning logic.

0.2.2 (2015-05-07)
------------------
* Adding MinGW-specific compiler flags for statically linking pygeoprocessing
  binaries against libstdc++ and libgcc.  Fixes an issue on many user's
  computers when installing from a wheel on the Python Package Index without
  having two needed DLLs on the PATH, resulting in an ImportError on pygeoprocessing.geoprocessing_core.pyd.
* Fixing an issue with versioning where 'dev' was displayed instead of the
  version recorded in pygeoprocessing/__init__.py.
* Adding all pygeoprocessing.geoprocessing functions to
  pygeoprocessing.__all__, which allows those functions to appear when
  calling help(pygeoprocessing).
* Adding routing_core.pxd to the manifest.  This fixes an issue where some
  users were unable to compiler pygeoprocessing from source.

0.2.1 (2015-04-23)
------------------
* Fixed a bug on the test that determines if a raster should be memory
  blocked.  Rasters were not getting square blocked if the memory block was
  row aligned.  Now creates 256x256 blocks on rasters larger than 256x256.
* Updates to reclassify_dataset_uri to use numpy.digitize rather than Python
  loops across the number of keys.
* More informative error messages raised on incorrect bounding box mode.
* Updated docstring on get_lookup_from_table to indicate the headers are case
  insensitive.
* Added updates to align dataset list that report which dataset is being
  aligned.  This is helpful for logging feedback when many datasets are passed
  in that don't take long enough to get a report from the underlying reproject
  dataset function.
* pygeoprocessing.routing.routing_core includes pxd to be ``cimport``\able
  from a Cython module.

0.2.0 (2015-04-14)
------------------
* Fixed a library wide issue relating to the underlying numpy types of
  GDT_Byte Datasets.  Now correctly identify the signed and unsigned versions
  and removed all instances where code used to mod byte data to unsigned data
  and correctly creates signed/unsigned byte datasets during resampling.
* Removed extract_band_and_nodata function since it exposes the underlying
  GDAL types.
* Removed reclassify_by_dictionary since reclassify_dataset_uri provided
  almost the same functionality and was widely used.
* Removed the class OrderedDict that was not used.
* Removed the function calculate_value_not_in_dataset since it loaded the
  entire dataset into memory and was not useful.

0.1.8 (2015-04-13)
------------------
* Fixed an issue on reclassifying signed byte rasters that had negative nodata
  values but the internal type stored for vectorize datasets was unsigned.

0.1.7 (2015-04-02)
------------------
* Package logger objects are now identified by python hierarchical package
  paths (e.g. pygeoprocessing.routing)
* Fixed an issue where rasters that had undefined nodata values caused
  striping in the reclassify_dataset_uri function.

0.1.6 (2015-03-24)
------------------
* Fixing LICENSE.TXT to .txt issue that keeps reoccurring.

0.1.5 (2015-03-16)
------------------
* Fixed an issue where int32 dems with INT_MIN as the nodata value were being
  treated as real DEM values because of an internal cast to a float for the
  nodata type, but a cast to double for the DEM values.
* Fixed an issue where flat regions, such as reservoirs, that could only drain
  off the edge of the DEM now correctly drain as opposed to having undefined
  flow directions.

0.1.4 (2015-03-13)
------------------
* Fixed a memory issue for DEMs on the order of 25k X 25k, still may have
  issues with larger DEMs.

0.1.3 (2015-03-08)
------------------
* Fixed an issue so tox correctly executes on the repository.
* Created a history file to document current and previous releases.
* Created an informative README.rst.

0.1.2 (2015-03-04)
------------------
* Fixing issue that caused "LICENSE.TXT not found" during pip install.

0.1.1 (2015-03-04)
------------------
* Fixing issue with automatic versioning scheme.

0.1.0 (2015-02-26)
------------------
* First release on PyPI.


            

Raw data

            {
    "_id": null,
    "home_page": "https://github.com/natcap/pygeoprocessing",
    "name": "pygeoprocessing",
    "maintainer": "Natural Capital Project Software Team",
    "docs_url": null,
    "requires_python": ">=3.6",
    "maintainer_email": null,
    "keywords": "gis, geoprocessing, pygeoprocessing",
    "author": null,
    "author_email": null,
    "download_url": "https://files.pythonhosted.org/packages/91/25/760ce41928c4208a238efe37a183893e1bbf93c85437ff1938bb287e5d5c/pygeoprocessing-2.4.6.tar.gz",
    "platform": null,
    "description": ".. default-role:: code\n\nAbout PyGeoprocessing\n=====================\n\nPyGeoprocessing is a Python/Cython based library that provides a set of\ncommonly used raster, vector, and hydrological operations for GIS processing.\nSimilar functionality can be found in ArcGIS/QGIS raster algebra, ArcGIS zonal\nstatistics, and ArcGIS/GRASS/TauDEM hydrological routing routines.\n\nPyGeoprocessing is developed at the Natural Capital Project to create a\nprogrammable, open source, and free Python based GIS processing library to\nsupport the InVEST toolset.  PyGeoprocessing's design prioritizes\ncomputation and memory efficient runtimes, easy installation and cross\ncompatibility with other open source and proprietary software licenses, and a\nsimplified set of orthogonal GIS processing routines that interact with GIS\ndata via filename. Specifically the functionally provided by PyGeoprocessing\nincludes:\n\n* a suite of raster manipulation functions (warp, align, raster calculator,\n  reclassification, distance transform, convolution, and fast iteration)\n* a suite of vector based manipulation function (zonal statistics,\n  rasterization, interpolate points, reprojection, and disjoint polygon sets)\n* a simplified hydrological routing library (D8inf/MFD flow direction,\n  plateau drainage, weighted and unweighted flow accumulation, and weighted\n  and unweighted flow distance)\n\nInstalling PyGeoprocessing\n==========================\n\n.. code-block:: console\n\n    $ pip install pygeoprocessing\n\nIf you `import pygeoprocessing` and see a `ValueError: numpy.dtype has the\nwrong size, try recompiling`, this is the result of a version compatibility\nissue with the numpy API in the precompiled pygeoprocessing binaries.\nThe solution is to recompile pygeoprocessing on your computer:\n\n.. code-block:: console\n\n    $ pip uninstall -y pygeoprocessing\n    $ pip install pygeoprocessing --no-deps --no-binary :all:\n\nRequirements\n------------\n\nNote the pip-installable requirements in `requirements.txt` are for best\nresults, but older package versions may also work. If necessary,\nPyGeoprocessing can be installed without dependencies with `pip install\n--no-deps`.\n\nAPI Documentation\n=================\n\nAPI documentation is available at https://pygeoprocessing.readthedocs.io/en/latest/\n\nRelease History\n===============\n\n.. Unreleased Changes\n\n2.4.6 (2024-10-15)\n------------------\n* Removing the ``numpy<2`` constraint for requirements.txt that should have\n  been included in the 2.4.5 release. https://github.com/natcap/pygeoprocessing/issues/396\n* Handling GDAL-based ``RuntimeError``s raised during ``pygeoprocessing.reproject_vector``.\n  https://github.com/natcap/pygeoprocessing/issues/409\n\n2.4.5 (2024-10-08)\n------------------\n* Updating for numpy 2.0 API changes.  Pygeoprocessing is now compatible with\n  numpy 2.0 and later.  https://github.com/natcap/pygeoprocessing/issues/396\n\n2.4.4 (2024-05-21)\n------------------\n* Our github actions for building python distributions now use\n  `actions/setup-python@v5`, which uses node 20.\n  https://github.com/natcap/pygeoprocessing/issues/384\n* ``warp_raster`` and ``build_overviews`` no longer raise a ``ValueError`` if\n  called with an invalid resampling algorithm. We now fall back to the\n  underlying GDAL functions' error messages.\n  https://github.com/natcap/pygeoprocessing/issues/387\n* Implementing decaying flow accumulation for D8 routing.\n  https://github.com/natcap/pygeoprocessing/issues/386\n* Updated to Cython 3.\n* Dropped support for Python 3.7.\n* Enabled GDAL exceptions (``gdal.UseExceptions``) throughout pygeoprocessing.\n  This is done with a context manager so should not affect other code or\n  libraries. https://github.com/natcap/pygeoprocessing/issues/391\n\n2.4.3 (2024-03-06)\n------------------\n* Wheels for python 3.12 are now built during our github actions runs.\n  https://github.com/natcap/pygeoprocessing/issues/381\n* ``get_gis_type`` can accept a path to a remote file, allowing the GDAL driver\n  to open it if the driver supports the protocol.\n  https://github.com/natcap/pygeoprocessing/issues/375\n* If running on a SLURM system (identified by the presence of ``SLURM*``\n  environment variables), the GDAL cache max is checked against the amount of\n  memory available on the compute node.  If GDAL may exceed the available slurm\n  memory, a warning is issued or logged.\n  https://github.com/natcap/pygeoprocessing/issues/361\n* Fixed an issue in ``extract_strahler_streams_d8`` where a nodata pixel\n  could be mistakenly treated as a stream seed point, ultimately creating\n  a stream feature with no geometry.\n  https://github.com/natcap/pygeoprocessing/issues/361\n* Improved ``align_and_resize_raster_stack`` so that rasterization of a vector\n  mask only happens once, regardless of the number of rasters in the stack.\n  In addition, the created mask raster's path may be defined by the caller so\n  that it persists across calls to ``align_and_resize_raster_stack``.\n  https://github.com/natcap/pygeoprocessing/issues/366\n* Improved ``warp_raster`` to allow for a pre-defined mask raster to be\n  provided instead of a vector.  If both are provided, the mask raster alone is\n  used.  The new mask raster must have the same dimensions and geotransform as\n  the output warped raster. https://github.com/natcap/pygeoprocessing/issues/366\n* Fixed a bug in ``zonal_statistics`` where the wrong number of disjoint\n  polygon sets were being reported in the logs.\n  https://github.com/natcap/pygeoprocessing/issues/368\n* Pygeoprocessing is now tested against python 3.12.\n  https://github.com/natcap/pygeoprocessing/issues/355\n\n2.4.2 (2023-10-24)\n------------------\n* Fixed an issue where MFD flow direction was producing many nodata holes given\n  a large-enough DEM.  These nodata holes would then propagate to flow\n  accumulation and stream extraction, producing very disjointed stream\n  networks. https://github.com/natcap/pygeoprocessing/issues/350\n* Improved progress logging in MFD flow direction, MFD flow accumulation, MFD\n  stream extraction to prevent integer overflows in percentages and improve\n  the readability of log messages. https://github.com/natcap/pygeoprocessing/issues/246\n* Fixed an issue introduced in 2.4.1 where ``zonal_statistics`` results could\n  vary depending on the bounding boxes of the disjoint polygon sets calculated\n  from the input vector.\n* Added a new function, ``pygeoprocessing.align_bbox``, which pads a bounding\n  box to align with the grid of a given geotransform.\n* Added support for int64 and uint64 (GDAL 3.5+)\n  https://github.com/natcap/pygeoprocessing/issues/352\n* Added support for signed bytes (GDAL 3.7+)\n  https://github.com/natcap/pygeoprocessing/issues/329\n* Fixed a bug where ``calculate_slope`` did not work as expected when the DEM\n  raster's nodata value is NaN (https://github.com/natcap/pygeoprocessing/issues/352)\n\n2.4.1 (2023-09-05)\n------------------\n* The ``pygeoprocessing`` package metadata has been updated to use\n  ``importlib.metadata`` (python >= 3.8) or ``importlib_metadata``\n  (python < 3.8) for retrieving the package version, in keeping with\n  recommendations from ``setuptools_scm``.  The dependency\n  ``importlib_metadata`` is now required for installs on python < 3.8.\n* Fixed another memory leak in ``pygeoprocessing.raster_calculator``, where\n  shared memory objects under certain circumstances were not being unlinked at\n  the end of the function, resulting in excess memory usage and a warning\n  during the shutdown of the python process.\n  https://github.com/natcap/pygeoprocessing/issues/247\n* Added a new function, ``pygeoprocessing.array_equals_nodata``, which returns\n  a boolean array indicating which elements have nodata. It handles integer,\n  float, and ``nan`` comparison, and the case where the nodata value is `None`.\n* Standardized the approach used in ``warp_raster`` and\n  ``create_raster_from_bounding_box`` for determining the dimensions of the\n  target raster given a target bounding box and pixel sizes.\n  https://github.com/natcap/pygeoprocessing/issues/321\n* ``pygeoprocessing.routing.delineate_watersheds_d8`` now handles the case\n  where the input flow direction raster does not have a defined spatial\n  reference. https://github.com/natcap/pygeoprocessing/issues/254\n* Updating internal documentation describing TauDEM flow directions, and adding\n  for how to convert from a flow direction raster from what TauDEM expects to\n  what pygeoprocessing expects.\n  https://github.com/natcap/pygeoprocessing/issues/255\n* Users may now specify the overview level to use when calling ``warp_raster``.\n  By default, ``pygeoprocessing`` will use the base layer.\n  https://github.com/natcap/pygeoprocessing/issues/326\n* Fixed a bug across ``pygeoprocessing`` where some valid resampling methods\n  would throw an exception because they were not recognized.  This was only\n  happening when ``pygeoprocessing`` was installed alongside GDAL < 3.4.\n* Fixing an issue with ``pygeoprocessing.multiprocessing.raster_calculator``\n  where the function would raise an Exception when the target raster path was\n  provided as a filename only, not within a directory, even though the parent\n  directory could be inferred. https://github.com/natcap/pygeoprocessing/issues/313\n* Fixing a bug where the statistics worker in\n  ``pygeoprocessing.raster_calculator`` may return a complex value. This is\n  only an issue when pygeoprocessing is compiled against Cython 3.0.0 and\n  later. https://github.com/natcap/pygeoprocessing/issues/342\n* Added the new function ``pygeoprocessing.raster_map``, a higher-level version\n  of ``raster_calculator``. https://github.com/natcap/pygeoprocessing/issues/235\n\n2.4.0 (2023-03-03)\n------------------\n* A new submodule, ``pygeoprocessing.kernels`` has been added to facilitate the\n  creation of kernel rasters needed for calls to\n  ``pygeoprocessing.convolve_2d``. Functions for creating common decay kernels\n  have been added, along with functions to facilitate the creation of\n  distance-based kernels using a user-defined function\n  (``pygeoprocessing.create_kernel``) and to facilitate the creation of kernels\n  using custom 2D numpy arrays, such as those commonly used in image processing\n  (``pygeoprocessing.kernel_from_numpy_array``).\n  https://github.com/natcap/pygeoprocessing/issues/268\n* Logging across functions in ``pygeoprocessing.geoprocessing`` now correctly\n  reports the function that it's being called from rather than ``<lambda>``.\n  https://github.com/natcap/pygeoprocessing/issues/300\n* The function ``pygeoprocessing.reproject_vector`` now accepts an optional\n  parameter ``layer_name`` to allow the target vector layer name to be defined\n  by the user.  If the user does not provide a ``layer_name``, the layer name\n  will be copied from the source vector.\n  https://github.com/natcap/pygeoprocessing/issues/301\n* Implement the proposed new function ``pygeoprocessing.raster_reduce``, a\n  wrapper around ``pygeoprocessing.iterblocks``\n  (https://github.com/natcap/pygeoprocessing/issues/285)\n* Nodata value checking in ``pygeoprocessing.routing`` now correctly handles\n  comparison of ``nan`` values.  This is explicitly tested in\n  ``pygeoprocessing.routing.fill_pits``, but should also improve the\n  experience of other routing functions as well.\n  https://github.com/natcap/pygeoprocessing/issues/248\n* Added a function to build overviews for a raster,\n  ``pygeoprocessing.build_overviews``. Related to this,\n  ``pygeoprocessing.get_raster_info()`` now includes an ``'overviews'`` key\n  listing the pixel dimensions of each overview layer in a raster.\n  https://github.com/natcap/pygeoprocessing/issues/280\n* Added a D8 stream extraction function at\n  ``pygeoprocessing.routing.extract_streams_d8`` which takes a D8 flow\n  accumulation raster and a flow accumulation threshold, setting all pixels\n  with accumulation above that threshold to 1 and all other valid pixels to 0.\n  https://github.com/natcap/pygeoprocessing/issues/272\n* Adding a new function, ``pygeoprocessing.create_raster_from_bounding_box``,\n  that enables the creation of a new raster from a bounding box.\n  https://github.com/natcap/pygeoprocessing/issues/276\n* Win32 wheels of PyGeoprocessing are no longer created through our GitHub\n  Actions workflows and will no longer be produced or distributed as part of\n  our release checklist.  For details (and metrics!) see:\n  https://github.com/natcap/pygeoprocessing/issues/232\n\n2.3.5 (2022-12-13)\n------------------\n* ``pygeoprocessing.calculate_disjoint_polygon_set`` now offers an optional\n  parameter, ``geometries_may_touch`` for cases where geometries are known to\n  have intersecting boundaries but nonintersecting interiors.\n  https://github.com/natcap/pygeoprocessing/issues/269\n* Pygeoprocessing is now tested against Python 3.11.\n* Adding the target filename to progress logging in\n  ``pygeoprocessing.raster_calculator``.\n* ``pygeoprocessing.zonal_statistics`` will now optionally include a count of\n  the number of pixels per value encountered under each polygon. A warning\n  will be logged when invoked on floating-point rasters, as using this on\n  continuous rasters can result in excessive memory consumption. To use this\n  feature, set ``include_value_counts=True`` when calling ``zonal_statistics``.\n* ``pygeoprocessing.get_gis_type`` will now raise a ``ValueError`` if the file\n  cannot be opened as ``gdal.OF_RASTER`` or ``gdal.OF_VECTOR``.\n  https://github.com/natcap/pygeoprocessing/issues/244\n* Fixing an error message in ``convolve_2d`` when the signal or kernel is a\n  row based blocksize. https://github.com/natcap/pygeoprocessing/issues/228.\n\n2.3.4 (2022-08-22)\n------------------\n* Fixing an issue with imports at the ``pygeoprocessing`` module level that was\n  causing linters like PyLint and IDE command-completion programs like JEDI-vim\n  to not be able to identify the attributes of the ``pygeoprocessing`` module\n  namespace.\n* Reducing the amount of memory used by\n  ``pygeoprocessing.new_raster_from_base`` when filling an array with values to\n  only as much memory as is needed for the datatype.\n* Fixing a memory leak in ``pygeoprocessing.raster_calculator`` where\n  shared memory objects were being inadvertently created when they should not\n  have been and then they were not subsequently destroyed.\n* ``calculate_disjoint_polygon_set`` will now skip over empty geometries.\n  Previously, the presence of empty geometries would cause an error to be\n  raised.\n* Fixed a ``DeprecationWarning`` in ``calculate_disjoint_polygon_set`` caused\n  by the use of a deprecated logging ``warn`` method.\n\n2.3.3.post0 (2022-01-28)\n------------------------\n* Post-release due to corrupted sdist released on Github and PyPI. The sdist\n  for 2.3.3 failed to install. The wheels were unaffected. No code changes.\n\n2.3.3 (2022-01-26)\n------------------\n* Fixing a bug in ``shapely_geometry_to_vector`` where a feature name mismatch\n  between the ``fields`` and ``attribute_list`` inputs would silently pass\n  under most circumstances.  Now an informative ``ValueError`` is raised.\n* Testing against Python 3.10.\n* Pinned ``numpy`` versions in ``pyproject.toml`` to the lowest compatible\n  version for each supported python version. This prevents issues when\n  ``pygeoprocessing`` is used in an environment with a lower numpy version\n  than it was built with (https://github.com/cython/cython/issues/4452).\n\n2.3.2 (2021-09-08)\n------------------\n* Restore functionality in ``reclassify_raster`` that allows for nodata\n  values to be reclassified. This was accidentally removed in 2.3.1.\n\n2.3.1 (2021-08-24)\n------------------\n* Slightly change the error message displayed for a\n  ``ReclassificationMissingValuesError``.\n* Handle float nodata values correctly in ``reclassify_raster``.\n* Fix a bug in ``distance_to_channel_mfd`` where pixels that only partially\n  drained to a stream were given incorrect distances.\n* Fixed an issue with ``raster_band_percentile`` that would raise an\n  exception if an input raster had an undefined nodata value and/or\n  would cause an invalid result if the raster contained non-finite\n  values.\n* After many years with the Natural Capital Project, Rich Sharp has stepped\n  down from the Project and as the maintainer of ``pygeoprocessing``.  James\n  Douglass is taking his place, and this change is now reflected in\n  ``setup.py``.\n\n2.3.0 (2021-06-21)\n------------------\n* Added a ``single_outlet_tuple`` parameter to ``routing.fill_pits`` that\n  forces a DEM to only have one outlet at any point on the raster. The\n  fill effect is that all pixels will drain to the raster coordinate at\n  ``single_outlet_tuple``.\n* Added a ``detect_lowest_sink_and_drain`` function that finds the lowest\n  DEM pixel that drains to nodata/edge and the lowest DEM pixel that could\n  be a sink. The values that result from this call can be used to condition\n  a DEM that is known to have a single drain using the\n  ``single_outlet_tuple`` parameter in ``routing.fill_pits``.\n* Fixed a bug in ``routing.fill_pits`` that could cause the nodata region of\n  a DEM to be incorrectly filled with non-nodata values.\n* Fixed an issue where ``create_raster_from_vector_extent`` would use an\n  approximation of the layer extent to infer the input vector extent. Now\n  uses the individual geometry to infer vector extents.\n* Fixed a bug in ``routing.distance_to_channel_mfd`` where pixels that drained\n  off the edge of the raster had negative values rather than nodata values.\n* Expanded the error message raised by ``transform_bounding_box`` when the\n  bounding box cannot be transformed to provide more helpful details.\n* Add support and testing for GDAL 3.3.0.\n* Fixed exception occuring in ``stitch_rasters`` when base rasters were\n  larger than the target raster on the ``y`` axis.\n\n2.2.0 (2021-05-14)\n------------------\n* Adding explicit support for Python 3.9 and testing on Python 3.9.\n* Fixed an issue in ``create_raster_from_vector_extents`` that would cause a\n  confusing exception to be raised if there was no geometry in the vector.\n  Now raises a ``ValueError`` with a helpful error message.\n* Changed parameters in ``convolve_2d`` to allow API to set\n  ``ignore_nodata_and_edges=True`` while ``mask_nodata=False`` and updated\n  docstring to indicate this is useful in cases such as filling nodata holes\n  in missing datasets. Additionally added a logger ``debug`` message to note\n  this \"unusual\" setting of these parameters in case of accidental usage\n  which could be noted during development.\n* Fixed issue in zonal statistics that could cause the aggregate vector to be\n  locked due to a dereferencing race condition. This was present in some\n  cases with a flaky unit test but could have been seen in practice if the\n  vector was deleted immediately after the call to ``zonal_statistics``.\n* Fixed issue in ``routing.fill_pits`` that used a numerical \"is close\" test\n  when determining if two pixels were at equal height. In the case of\n  hydrological pitfilling, pixels must be exactly equal height to be\n  considered a plateau otherwise a drain into or out of the pixel is\n  resolved. Testing is close allowed a hydrological pit to remain since it\n  was \"close\" to the same height as a draining pixel.\n* Added a D8 Strahler order stream extraction function at\n  ``pygeoprocessing.extract_strahler_streams_d8``. Creates segmented\n  geometric lines which correspond to streams on the landscape.\n* Added a D8 subwatershed calculation function at\n  ``pygeoprocessing.calculate_subwatershed_boundary``. Creates subwatersheds\n  that are segmented at the junctions of the streams created by\n  ``pygeoprocessing.extract_strahler_streams_d8``.\n* Removing all instances of ``__swig_destroy__`` to prevent multiprocessing\n  memory corruption.\n* Exposing a ``use_shared_memory`` flag on ``raster_calculator`` to allow\n  a user to use shared memory objects when calculating statistics. This\n  feature is only available for Python >= 3.8. If available, this\n  feature creates a significant runtime improvement but can be unstable\n  in multiprocessing configurations. For this reason it is set to\n  ``False`` as the default value.\n* Added a ``max_timeout`` parameter to ``convolve_2d`` and\n  ``raster_calculator`` to allow the user to specify the maximum amount of\n  time to wait for worker threads to terminate. In normal operation these\n  threads should terminate in a short amount of time but are generously\n  timed with the ``pygeoprocessing._MAX_TIMEOUT`` parameter. This parameter\n  allows a user to tune in cases that may involve significant latency such\n  as in a heavy multiprocess environment.\n* Added a ``pygeoprocessing.routing.detect_outlets`` function that creates a\n  vector with points placed on the center of the pixels in a D8 or MFD flow\n  direction raster that are outlet pixels. The vector has additional metadata\n  about the (i, j) locations of the pixel in raster coordinate space.\n* Replacing ``merge_rasters`` with ``stitch_rasters`` that can take an\n  arbitrary list of raster path bands and stitch them into an existing raster\n  of same or different projection. Additional functionality to preserve\n  area variant values when stitching from a linear meters projection to\n  WGS84. Useful when the units of the base raster are in \"units per pixel\"\n  rather than \"units per area\".\n* Added a ``max_pixel_fill_count`` parameter to ``routing.fill_pits`` to\n  guard against pitfilling large natural depression. Defaults to 500.\n* Fixed an issue in ``align_and_resize_raster_stack`` that would ignore\n  the bounds of a feature in a mask vector if the\n  ``\"mask_vector_where_filter\"`` clause was invoked and instead only\n  considered the entire bounds of the vector.\n* Fixed an issue with ``convolve_2d`` that allowed output rasters to be\n  created without a defined nodata value.\n* Fixed a LOGGER message bug that occurred in ``zonal_statistics``.\n* Fixed an issue in ``distance_to_channel_mfd`` that would generate a raster\n  with distances to the edge of the raster even if there was no channel. Now\n  generates nodata so it is consistent with ``distance_to_channel_d8``.\n* The ``convolve_2d`` function now raises a ValueError if either the signal\n  or kernel raster has a row based blocksize since this could result in\n  very long runtimes due to gdal cache thrashing.\n* Fixed an issue that would cause a ``convolve_2d`` worker to crash if the\n  integrating kernel was not a float and also ``normalize_kernel=True``\n  thus causing the entire function call to deadlock.\n* Added a check on ``transform_bounding_box`` to ensure the target bounding\n  box's coordinates were finite. This guards against cases where a transform\n  into another coordinate system creates a degenerate bounding box.\n  Previously the function would silently return non-finite coordinates.\n* Fixing issue when calculating histogram for floating point rasters the\n  logging progress percent would be incorrectly calculated.\n\n2.1.2 (2020-12-03)\n------------------\n* ``pygeoprocessing.warp_raster`` now raises a ``ValueError`` when an invalid\n  resampling method is provided.\n* Fixed bug in ``pygeoprocessing.warp_raster`` that would not properly handle\n  GDAL Byte type signing when masking warped raster with a vector.\n* Fixed issue in ``convolve_2d`` that would cause excessive memory use\n  leading to out of memory errors.\n* Fixed issue in ``convolve_2d`` that could lead to a file removal race\n  condition and raise a ``FileNotFoundException`` when ignoring nodata\n  regions.\n\n2.1.1 (2020-09-16)\n------------------\n* Fixed a critical bug introduced in 2.1.0 that generated invalid results in\n  `convolve_2d` for any raster larger than 256x256.\n* Added an optional parameter, ``target_layer_name`` to\n  ``pygeoprocessing.routing.delineate_watersheds_d8`` for cases where a\n  caller would like to define the output layer name.\n\n2.1.0 (2020-08-25)\n------------------\n* Modified ``pygeoprocessing.numpy_array_to_raster`` to take boolean arrays.\n* Modified ``pygeoprocessing.convolve_2d`` to guard against nonsensical queries\n  to both ``ignore_nodata_and_edges=True`` but also ``mask_nodata=False``.\n  A query of this combination now raises a ``ValueError`` to guard against\n  programmer error.\n* Added a custom exception class ``ReclassificationMissingValuesError`` to\n  ``pygeoprocessing``. ``pygeoprocessing.reclassify_raster`` raises this\n  exception instead of ``ValueError`` when a raster pixel value is not\n  represented in ``value_map``. This custom exception provides a list of\n  missing raster pixel values in a ``missing_values`` attribute that allows\n  the caller access to the pixel values that are missing through a Python type\n  rather than indirectly through an error message.\n* Correcting the docstring for ``pygeoprocessing.numpy_array_to_raster`` to\n  specify that the ``pixel_size`` parameter must be a tuple or list, not an\n  int.\n* ``pygeoprocessing.routing.delineate_watersheds_d8`` now has an optional\n  parameter ``write_diagnostic_vector``.  When ``True``, this parameter will\n  cause a new vector per outflow feature to be created in the ``working_dir``.\n  This parameter defaults to ``False``.  This is a change from prior behavior,\n  when the diagnostic vectors were always created, which could occupy a lot of\n  computational time under large outflow geometries.\n* Added a ``pygeoprocessing.multiprocessing.raster_calculator`` function which\n  matches the API and results of ``pygeoprocessing.raster_calculator`` but uses\n  multiple processing cores to compute raster calculation blocks.\n  when the diagnostic vectors were always created, which could occupy\n  significant computational time under large outflow geometries.\n* Minor performance improvement to ``pygeoprocessing.convolve_2d`` by\n  preventing a pre-processing step that initialized temporary rasters with zero\n  values as well as added asynchronous work distribution for kernel/signal\n  block processing.\n* Modified logging message for ``pygeoprocessing.new_raster_from_base`` when\n  filling a raster such that an informative error message is printed with\n  context as to the function, file, status, and value being filled.\n\n2.0.0 (05-19-2020)\n------------------\n* Adding Python 3.8 support and dropping Python 3.6 support.\n* Adding GDAL 3 support and dropping GDAL 2 support. The only non-backwards\n  compatible issue in GDAL 2 to GDAL 3 is the need to handle Axis Ordering with\n  osr.SetAxisMappingStrategy(osr.OAMS_TRADITIONAL_GIS_ORDER) because of\n  https://trac.osgeo.org/gdal/wiki/rfc73_proj6_wkt2_srsbarn#Axisorderissues?.\n  Since Axis ordering now matters for Geographic CRS the expected order is\n  Lat,Lon but we use osr.OAMS_TRADITIONAL_GIS_ORDER to swap to Lon,Lat.\n* Using osr.CreateCoordinateTransformation() instead of\n  osr.CoordinateTransformation() as the GDAL 3 call.\n* Fixed a bug in convolve_2d that would not ``ignore_nodata`` if the signal\n  raster's nodata value was undefined. Changed the name of this flag to\n  ``ignore_nodata_and_edges`` to reflect its expected functionality.\n* Warped signed byte rasters are now also signed byte rasters.\n* Adding a GitHub Actions-based build job for building wheels and a source\n  distribution for a given commit of pygeoprocessing.\n* Updated ``setup.py`` to point the URL project link to the project's new\n  home on GitHub.\n* Updated ``MANIFEST.in`` to only include files that should be there in the\n  pygeoprocessing source distribution.  This fixes an issue where files\n  matching a variety of extensions anywhere in the pygeoprocessing directory\n  might be included with the source distribution.\n* Added ``set_tol_to_zero`` to ``convolve_2d`` to allow for in-function masking\n  of near-zero results to be set to 0.0.\n* Fixed malformed logging outputs which could be seen during long running\n  ``rasterize`` calls.\n* Renamed all parameters involving Spatial Projections to the form\n  ``[var_id]_projection_wkt``, this involves optional arguments in\n  ``reproject_vector``, ``warp_raster``, ``transform_bounding_box``,\n  and ``align_and_resize_raster_stack`` as well as the return value from\n  ``get_raster_info`` and ``get_vector_info``.\n* Fixed an issue in ``zonal_statistics`` that would crash if an aggregate\n  vector had a feature with no geometry defined. Now the function ignores\n  such features and prints a warning to the log.\n* Fixed a malformed ``ValueError`` message when a corrupt raster was\n  encountered in ``raster_calculator``.\n* Fixes an unnecessary calculation that pre-fills slope raster GeoTIFFs with\n  nodata values.\n* Added a check to ``convolve_2d`` to verify that raster path/band tuples were\n  passed where expected and raise a useful Exception  if not.\n* Removed most of the `pygeoprocessing.testing` module and added the following\n  convenience functions to ``pygeoprocessing``, which should not be used\n  for production code but are useful for testing and scripting:\n\n  * ``raster_to_numpy_array`` - read a single band of a raster into a ``numpy``\n    array, runs the risk of memory error if the raster is too large.\n  * ``numpy_array_to_raster`` - writes a ``numpy`` array to a raster on disk.\n  * ``shapely_geometry_to_vector`` - creates a vector from a list of\n    ``Shapely`` geometry.\n\n* Fixed an issue in ``flow_dir_mfd`` that would cause invalid flow directions\n  on DEMs that had very small numerical delta heights.\n* Fixes an issue in ``convolve_2d`` that would occasionally cause undefined\n  numerical noise in regions where the signal was nodata but ``mask_nodata``\n  was set to ``False``.\n\n1.9.2 (2020-02-06)\n------------------\n* Removed the ``multiprocessing`` dependency to avoid an occasional deadlock\n  that occurred on Mac OS X during ``align_and_resize_raster_stack``.\n  That function now operates serially, but multithreading can be used by\n  passing ``gdal_warp_options``.\n\n1.9.1 (2019-12-19)\n------------------\n* Fixed a compilation issue on Mac OS X Catalina related to the compilation\n  of a template in the file iteration component of the out-of-core percentile\n  function.\n* Resolved a compilation issue on Mac OS X (Mavericks and later) where\n  pygeoprocessing would not compile unless some additional compiler and linker\n  flags were provided.  These are now accounted for in the package's compilation\n  steps in ``setup.py``.\n* ``pygeoprocessing.symbolic.evaluate_raster_calculator_expression``\n  no longer depends on ``sympy`` for its expression evaluation.\n\n1.9.0 (2019-10-22)\n------------------\n* Fixed a memory error issue that could occur on multiple flow direction flow\n  accumulation calculations.\n* Added PEP518-compatible build dependencies to ``pyproject.toml``, which has\n  been added to source distributions of pygeoprocessing.\n* Added an out-of-core high performance raster percentile function at\n  pygeoprocessing.raster_band_percentile.\n* Added a ``pygeoprocessing.symbolic.evaluate_raster_calculator_expression``\n  function that can parse a string expression and a map of symbols to\n  calculate a raster calculation operation.\n* Modified ``pygeoprocessing.routing.fill_pits`` to create a single band\n  raster of the input DEM raster/path band rather than a copy of the input\n  DEM raster irrespective of the band number.\n* Fixed an issue that would cause the bounding box generated by\n  ```transform_bounding_box`` to flip the min and max coordinates in some\n  spatial contexts.\n* Defaulting compression algorithm back to LZW as ZSTD is not a widely\n  implemented compression standard for geotiffs.\n\n1.8.0 (2019-08-12)\n------------------\n* Added a ``'numpy_type'`` field to the result of ``get_raster_info`` that\n  contains the equivalent numpy datatype of the GDAL type in the raster. This\n  includes functionality differentate between the unsigned and signed\n  ``gdal.GDT_Byte`` vs. ``numpy.int8`` and ``numpy.uint8``.\n* Changed default compression routine for GeoTIFFs to ZSTD (thanks Facebook\n  https://facebook.github.io/zstd/).\n* Added a **non-backwards compatible change** by replacing the\n  ``gtiff_creation_options`` string to a driver/option string named\n  ``raster_driver_creation_tuple``. This allows the caller to create any type\n  of ``GDAL`` writable driver along with the option list associated with that\n  driver.\n* Added a ``'numpy_type'`` field to the result of ``get_raster_info`` that\n  contains the equivalent numpy datatype of the GDAL type in the raster. This\n  includes functionality differentate between the unsigned and signed\n  ``gdal.GDT_Byte`` vs. ``numpy.int8`` and ``numpy.uint8``.\n* Changed default compression routine for GeoTIFFs to ZSTD (thanks Facebook\n  https://facebook.github.io/zstd/).\n* Added a ``'file_list'`` key to the dictionary returned by\n  ``get_raster_info`` and ``get_vector_info`` that contains a list of all the\n  files associated with that GIS object. The first parameter of these lists\n  can be passed to ``gdal.OpenEx`` to open the object directly.\n* Added a ``get_gis_type`` function to ``pygeoprocessing`` that takes a\n  filepath and returns a bitmask of ``pygeoprocessing.RASTER_TYPE`` and/or\n  ``pygeoprocessing.VECTOR_TYPE``.\n* Modified ``iterblocks`` to raise a helpful ValueError instead of a general\n  NoneTypeError if a raster does not open.\n\n1.7.0 (2019-06-27)\n------------------\n* Removing support for Python 2.7.\n* Adding D8 watershed delineation as\n  ``pygeoprocessing.routing.delineate_watersheds_d8``.\n* Corrected an issue with ``pygeoprocessing.create_raster_from_vector_extents``\n  where a vector with no width or no height (a vector with a single point, for\n  example) would result in invalid raster dimensions being passed to GDAL.\n  These edge cases are now guarded against.\n* ``pygeoprocessing.calculate_disjoint_polygon_set`` will now raise\n  ``RuntimeError`` if it is passed a vector with no features in it.\n* ``pygeoprocessing.rasterize`` will now raise ``RuntimeError`` if the\n  underlying call to ``gdal.RasterizeLayer`` encounters an error.\n* Correcting an issue with the docstring in\n  ``pygeoprocessing.reclassify_raster`` to reflect the current parameters.\n* Changed ``zonal_statistics`` to always return a ``dict`` instead of\n  sometimes a ``defaultdict``. This allows pickling of the result, if desired.\n* Adding automated testing via bitbucket pipelines.\n* Correcting an issue with ``pygeoprocessing.zonal_statistics`` that was\n  causing test failures on Python 3.6.\n* Pygeoprocessing is now tested against Python 3.7.\n* Fixed an issue in distance transform where a vertical striping artifact\n  would occur in the masked region of some large rasters when distances should\n  be 0.\n* Fixed an issue in all functionality that used a cutline polygon with\n  invalid geometry which would cause a crash. This was caused by `gdal.Warp`\n  when using the cutline functionality. Instead this functionality was\n  replaced with manual rasterization. In turn this introduces two optional\n  parameters:\n\n    * ``rasterize`` and ``mask_raster`` have a ``where_clause`` parameter\n      which takes a string argument in SQL WHERE syntax to filter\n      rasterization based on attribute values.\n    * ``warp_raster`` takes a ``working_dir`` parameter to manage local\n      temporary mask rasters.\n\n* Removing a temporary working directory that is created when executing\n  pygeoprocessing.convolve_2d.\n* Changed optional parameters involving layer indexes to be either indexes\n  or string ids. In all cases changing ``layer_index`` to ``layer_id`` in\n  the functions: ``get_vector_info``, ``reproject_vector``, ``warp_raster``,\n  ``rasterize``, ``calculate_disjoint_polygon_set``, and ``mask_raster``.\n\n1.6.1 (2019-02-13)\n------------------\n* Added error checking in ``raster_calculator`` to help ensure that the\n  ``target_datatype`` value is a valid GDAL type.\n* Fixed an issue in ``distance_transform_edt`` that would occasionally\n  cause incorrect distance calculations when the x sampling distance was > 1.\n\n1.6.0 (2019-01-23)\n------------------\n* Changed ``iterblocks`` API to take a raster/path band as an input rather\n  than a path and a list of bands. Also removed the ``astype_list`` due to\n  its lack of orthogonality.\n* Fixed bugs in ``convolve_2d`` involving inputs with nodata masking.\n* Changing default raster creation compression algorithm from LZW to DEFLATE,\n  this is to address issues where we were seeing recreatable, but\n  unexplainable LZWDecode errors in large raster data.\n* Fixed an issue that could cause the distance transform to be incorrect when\n  the sampling distance was in the noninclusive range of (0.0, 1.0).\n\n1.5.0 (2018-12-12)\n------------------\n* Specific type checking for ``astype_list`` in ``iterblocks`` to avoid\n  confusing exceptions.\n* Renamed test suite to be consistent with the pattern\n  ``test_[component].tif``.\n* Added a function ``pygeoprocessing.routing.extract_streams_mfd`` that\n  creates a contiguous stream layer raster to accounts for the divergent flow\n  that can occur with multiple flow direction. If the flow direction raster is\n  otherwise directly thresholded, small disjoint streams can appear where\n  the downstream flow drops below the threshold level.\n* Fixed an issue that could cause some custom arguments to geotiff creation\n  options to be ignored.\n* Added a ``mask_raster`` function that can be used to mask out pixels in\n  an existing raster that don't overlap with a given vector.\n* Fixed a bug in the ``distance_transform_edt`` function that would cause\n  incorrect distances to be calculated in the case of nodata pixels in the\n  region raster. The algorithm has been modified to treat nodata as though\n  pixel values were 0 (non-region) and the distance transform will be defined\n  for the entire raster.\n* Added a ``sampling_distance`` parameter to ``distance_transform_edt`` that\n  linearly scales the distance transform by this value.\n* Fixed an issue in ``calculate_slope`` that would raise an exception if the\n  input dem did not have a nodata value defined.\n* Changed the behavior of ``zonal_statistics`` for polygons that that do not\n  intersect any pixels. These FIDs are now also included in the result from\n  ``zonal_statistics`` where previously they were absent. This is to remain\n  consistent with how other GIS libraries calculate zonal stats.\n\n1.4.1 (2018-11-12)\n------------------\n* Hotfix that fixes an issue that would cause ``zonal_statistics`` to crash if\n  a polygon were outside of the raster's bounding box.\n\n1.4.0 (2018-11-12)\n------------------\n* Adding error checking to ensure that ``target_pixel_size`` passed to\n  ``warp_raster`` and ``align_and_resize_raster_stack`` are validated to ensure\n  they are in the correct format. This solves an issue where an incorrect\n  value, such as a single numerical value, resolve into readable exception\n  messages.\n* Added a ``gdal_warp_options`` parameter to ``align_and_resize_raster_stack``\n  and ``warp_raster`` whose contents get passed to gdal.Warp's ``warpOptions``\n  parameter. This was implemented to expose the CUTLINE_TOUCH_ALL\n  functionality but could be used for any gdal functionality.\n* Modified ``rasterize`` API call to make ``burn_values`` and ``option_list``\n  both optional parameters, along with error checking to ensure a bad input's\n  behavior is understood.\n* Exposing GeoTIFF creation options for all the ``pygeoprocessing.routing``\n  functions which create rasters. This is consistent with the creation\n  options exposed in the main ``pygeoprocessing`` API.\n* Removing ``'mean_pixel_size'`` as a return value from ``get_raster_info``,\n  this is because this parameter is easily misused and easily calculated if\n  needed. This is a \"What good programmers need, not what bad programmers\n  want.\" feature.\n\n1.3.1 (2018-10-25)\n------------------\n* Hotfix to patch an infinite loop when aggregating upstream or downstream\n  with custom rasters.\n\n1.3.0 (2018-10-25)\n------------------\n* Fixed a handful of docstring errors.\n* Improved runtime of ``zonal_statistics`` by a couple of orders of magnitude\n  for large vectors by using spatial indexes when calculating disjoint polygon\n  overlap sets, using database transactions, and memory buffers.\n* Improved runtime performance of ``reproject_vector`` by using database\n  transactions.\n* Improved logging for long runtimes in ``zonal_statistics``.\n* Changed ``zonal_statistics`` API and functionality to aggregate across the\n  FIDs of the aggregate vector. This is to be consistent with QGIS and other\n  zonal statistics functionality. Additionally, fixed a bug where very small\n  polygons might not get aggregated if they lie in the same pixel as another\n  polygon that does not intersect it. The algorithm now runs in two passes:\n\n    * aggregate pixels whose centers intersect the aggregate polygons\n    * any polygons that were not aggregated are geometrically intersected\n      with pixels to determine coverage.\n\n* Removed the ``calculate_raster_stats`` function since it duplicates GDAL\n  functionality, but with a slower runtime, and now functions in\n  ``pygeoprocessing`` that create rasters also calculate stats on the fly if\n  desired.\n* Fixes an issue in ``get_raster_info`` and ``get_vector_info`` where the path\n  to the raster/vector includes non-standard OS pathing (such as a NETCDF),\n  info will still calculate info.\n* Added functionality to ``align_raster_stack`` and ``warp_raster`` to define\n  a base spatial reference system for rasters if not is not defined or one\n  wishes to override the existing one. This functionality is useful when\n  reprojecting a rasters that does not have a spatial reference defined in the\n  dataset but is otherwise known.\n* Added a ``weight_raster_path_band`` parameter to both\n  ``flow_accumulation_d8`` and ``flow_accumulation_mfd`` that allows the\n  caller to use per-pixel weights from a parallel raster as opposed to\n  assuming a weight of 1 per pixel.\n* Added a ``weight_raster_path_band`` parameter to both\n  ``distance_to_channel_mfd`` and ``distance_to_channel_d8`` that allows the\n  caller to use per-pixel weights from a parallel raster as opposed to\n  assuming a distance of 1 between neighboring pixels or sqrt(2) between\n  diagonal ones.\n* Added an option to ``reproject_vector`` that allows a caller to specify\n  which fields, if any, to copy to the target vector after reprojection.\n* Adding a check in ``align_and_resize_raster_stack`` for duplicate target\n  output paths to avoid problems where multiple rasters are being warped to\n  the same path.\n* Created a public ``merge_bounding_box_list`` function that's useful for\n  union or intersection of bounding boxes consistent with the format in\n  PyGeoprocessing.\n* Added functionality in ``align_and_resize_raster_stack`` and ``warp_raster``\n  to use a vector to mask out pixel values that lie outside of the polygon\n  coverage area. This parameter is called ``vector_mask_options`` and is\n  fully documented in both functions. It is similar to the cutline\n  functionality provided in ``gdal.Warp``.\n* Fixed an issue in the ``flow_accumulation_*`` functions where a weight\n  raster whose values were equal to the nodata value of the flow accumulation\n  raster OR simply nodata would cause infinite loops.\n\n1.2.3 (2018-07-25)\n------------------\n* Exposing a parameter and setting reasonable defaults for the number of\n  processes to allocate to ``convolve_2d`` and ``warp_raster``. Fixes an issue\n  where the number of processes could exponentiate if many processes were\n  calling these functions.\n* Fixing an issue on ``zonal_statistics`` and ``convolve_2d`` that would\n  attempt to both read and write to the target raster with two different GDAL\n  objects. This caused an issue on Linux where the read file was not caught up\n  with the written one. Refactored to use only one handle.\n* Fixing a rare race condition where an exception could occur in\n  ``raster_calculator`` that would be obscured by an access to an object that\n  had not yet been assigned.\n* ``align_and_resize_raster_stack`` now terminates its process pool.\n* Increased the timeout in joining ``raster_calculator``'s stats worker.\n  On a slow system 5 seconds was not quite enough time.\n\n1.2.2 (2018-07-25)\n------------------\n* Hotfixed a bug that would cause numpy arrays to be treated as broadcastable\n  even if they were passed in \"raw\".\n\n1.2.1 (2018-07-22)\n------------------\n* Fixing an issue with ``warp_raster`` that would round off bounding boxes\n  for rasters that did not fit perfectly into the target raster's provided\n  pixel size.\n* Cautiously ``join``\\ing all process pools to avoid a potential bug where a\n  deamonized subprocess in a process pool may still have access to a raster\n  but another process may require write access to it.\n\n1.2.0 (2018-07-19)\n------------------\n* Several PyGeoprocessing functions now take advantage of multiple CPU cores:\n\n  * ``raster_calculator`` uses a separate thread to calculate raster\n    statistics in a ``nogil`` section of Cython code. In timing with a big\n    rasters we saw performance improvements of about 35%.\n  * ``align_and_resize_raster_stack`` uses as many CPU cores, up to the number\n    of CPUs reported by multiprocessing.cpu_count (but no less than 1), to\n    process each raster warp while also accounting for the fact that\n    ``gdal.Warp`` uses 2 cores on its own.\n  * ``warp_raster`` now directly uses ``gdal.Warp``'s multithreading directly.\n    In practice it seems to utilize two cores.\n  * ``convolve_2d`` attempts to use ``multiprocessing.cpu_count`` cpus to\n    calculate separable convolutions per block while using the main thread to\n    aggregate  and write the result to the target raster. In practice we saw\n    this improve runtimes by about 50% for large rasters.\n* Fixed a bug that caused some nodata values to not be treated as nodata\n  if there was a numerical roundoff.\n* A recent GDAL upgrade (might have been 2.0?) changed the reference to\n  nearest neighbor interpolation from 'nearest' to 'near'. This PR changes\n  PyGeoprocessing to be consistent with that change.\n* ``raster_calculator`` can now also take \"raw\" arguments in the form of a\n  (value, \"raw\") tuple. The parameter ``value`` will be passed directly to\n  ``local_op``. Scalars are no longer a special case and need to be passed as\n  \"raw\" parameters.\n* Raising ``ValueError`` in ``get_raster_info`` and ``get_vector_info`` in\n  cases where non-filepath non-GIS values are passed as parameters. Previously\n  such an error would result in an unhelpful error in the GDAL library.\n\n1.1.0 (2018-07-06)\n------------------\n* PyGeoprocessing now supports Python 2 and 3, and is tested on python 2.7\n  and 3.6  Testing across multiple versions is configured to be run via\n  ``tox``.\n* After testing (tox configuration included under ``tox-libcompat.ini``),\n  numpy requirement has been dropped to ``numpy>=1.10.0`` and scipy has been\n  modified to be ``scipy>=0.14.1,!=0.19.1``.\n* A dependency on ``future`` has been added for compatibility between python\n  versions.\n* Fixed a crash in ``pygeoprocessing.routing.flow_dir_mfd`` and\n  ``flow_dir_d8`` if a base raster was passed in that did not have a power of\n  two blocksize.\n* ``raster_calculator`` can now take numpy arrays and scalar values along with\n  raster path band tuples. Arrays and scalars are broadcast to the raster size\n  according to numpy array broadcasting rules.\n* ``align_and_resize_raster_stack`` can now take a desired target projection\n  which causes all input rasters to be warped to that projection on output.\n\n1.0.1 (2018-05-16)\n------------------\n* Hotfix patch to remove upper bound on required numpy version. This was\n  causing a conflict with InVEST's looser requirement. Requirement is now\n  set to >=1.13.0.\n\n1.0.0 (2018-04-29)\n------------------\n* This release marks a feature-complete version of PyGeoprocessing with a\n  full suite of routing and geoprocessing capabilities.\n* ``pygeoprocessing.routing`` module has a ``flow_dir_mfd`` function that\n  calculates a 32 bit multiple flow direction raster.\n* ``pygeoprocessing.routing`` module has a ``flow_accumulation_mfd`` function\n  that uses the flow direction raster from\n  ``pygeoprocessing.routing.flow_dir_mfd`` to calculate a per-pixel continuous\n  flow accumulation raster.\n* ``pygeoprocessing.routing`` module has a ``distance_to_channel_mfd``\n  function that calculates distance to a channel raster given a\n  pygeoprocessing MFD raster.\n* ``pygeoprocessing.routing`` module has a ``distance_to_channel_d8`` function\n  that calculates distance to a channel raster given a pygeoprocessing D8\n  raster.\n\n0.7.0 (2018-04-18)\n------------------\n* Versioning is now handled by ``setuptools_scm`` rather than\n  ``natcap.versioner``.  ``pygeoprocessing.__version__`` is now fetched from\n  the package metadata.\n* Raster creation defaults now set \"COMPRESS=LZW\" for all rasters created in\n  PyGeoprocessing, including internal temporary rasters. This option was\n  chosen after profiling large raster creation runs on platter hard drives.\n  In many cases processing time was dominated by several orders of magnitude\n  as a write-to-disk. When compression is turned on overall runtime of very\n  large rasters is significantly reduced. Note this otherwise increases the\n  runtime small raster creation and processing by a small amount.\n* ``pygeoprocessing.routing`` module now has a ``fill_pits``, function which\n   fills hydrological pits with a focus on runtime efficiency, memory space\n   efficiency, and cache locality.\n* ``pygeoprocessing.routing`` module has a ``flow_dir_d8`` that uses largest\n  slope to determine the downhill flow direction.\n* ``pygeoprocessing.routing`` module has a ``flow_accumulation_d8`` that uses\n  a pygeoprocessing D8 flow direction raster to calculate per-pixel flow\n  accumulation.\n* Added a ``merge_rasters`` function to ``pygeoprocessing`` that will mosaic a\n  set of rasters in the same projection, pixel size, and band count.\n\n0.6.0 (2017-01-10)\n------------------\n* Added an optional parameter to ``iterblocks`` to allow the ``largest_block``\n  to be set something other than the PyGeoprocessing default. This in turn\n  allows the ``largest_block`` parameter in ``raster_calculator`` to be passed\n  through to ``iterblocks``.\n* Upgraded PyGeoprocessing GDAL dependency to >=2.0.\n* Added a ``working_dir`` optional parameter to ``zonal_statistics``,\n  ``distance_transform_edt``, and ``convolve_2d`` which specifies a directory\n  in which temporary files will be created during execution of the function.\n  If set to ``None`` files are created in the default system temporary\n  directory.\n\n0.5.0 (2017-09-14)\n------------------\n* Fixed an issue where NETCDF files incorrectly raised Exceptions in\n  ``raster_calculator``  and ``rasterize`` because they aren't filepaths.\n* Added a NullHandler so that users wouldn't get an error that a logger\n  handler was undefined.\n* Added ``ignore_nodata``, ``mask_nodata``, and ``normalize_kernel`` options\n  to ``convolve_2d`` which make this function capable of adapting the nodata\n  overlap with the kernel rather than zero out the result, as well as on\n  the fly normalization of the kernel for weighted averaging purposes. This\n  is in part to make this functionality more consistent with ArcGIS's\n  spatial filters.\n\n0.4.4 (2017-08-18)\n------------------\n* When testing for raster alignment ``raster_calculator`` no longer checks the\n  string equality for projections or geotransforms.  Instead it only checks\n  raster size equality.  This fixes issues where users rasters DO align, but\n  have a slightly different text format of the WKT of projection.  It also\n  abstracts the problem of georeferencing away from raster_calculator that is\n  only a grid based operation.\n\n0.4.3 (2017-08-16)\n------------------\n* Changed the error message in ``reclassify_raster`` so it's more informative\n  about how many values are missing and the values in the input lookup table.\n* Added an optional parameter ``target_nodata`` to ``convolve_2d`` to set the\n  desired target nodata value.\n\n0.4.2 (2017-06-20)\n------------------\n* Hotfix to fix an issue with ``iterblocks`` that would return signed values\n  on unsigned raster types.\n* Hotfix to correctly cite Natural Capital Project partners in license and\n  update the copyright year.\n* Hotfix to patch an issue that gave incorrect results in many PyGeoprocessing\n  functions when a raster was passed with an NoData value.  In these cases the\n  internal raster block masks would blindly pass through on the first row\n  since a test for ``numpy.ndarray == None`` is ``False`` and later\n  ``x[False]`` is the equivalent of indexing the first row of the array.\n\n0.4.1 (2017-06-19)\n------------------\n* Non-backwards compatible refactor of core PyGeoprocessing geoprocessing\n  pipeline. This is to in part expose only orthogonal functionality, address\n  runtime complexity issues, and follow more conventional GIS naming\n  conventions. Changes include:\n\n    * Full test coverage for ``pygeoprocessing.geoprocessing`` module\n    * Dropping \"uri\" moniker in lieu of \"path\".\n    * If a raster path is specified and operation requires a single band,\n      argument is passed as a \"(path, band)\" tuple where the band index starts\n      at 1 as convention for raster bands.\n    * Shapefile paths are assumed to operate on the first layer.  It is so\n      rare for a shapefile to have more than one layer, functions that would\n      be confused by multiple layers have a layer_index that defaults to 0\n      that can be overridden in the call.\n    * Be careful, many of the parameter orders have been changed and renamed.\n      Generally inputs come first, outputs last.  Input parameters are\n      often prefixed with \"base\\_\" while output parameters are prefixed with\n      \"target\\_\".\n    * Functions that take rasters as inputs must have their rasters aligned\n      before the call to that function.  The function\n      ``align_and_resize_raster_stack`` can handle this.\n    * ``vectorize_datasets`` refactored to ``raster_calculator`` since that\n      name is often used as a convention when referring to raster\n      calculations.\n    * ``vectorize_points`` refactored to meaningful ``interpolate_points``.\n    * ``aggregate_by_shapefile`` refactored to ``zonal_statistics`` and now\n      returns a dictionary rather than a named tuple.\n    * All functions that create rasters expose the underlying GeoTIFF options\n      through a default parameter ``gtiff_creation_options`` which default to\n      \"('TILED=YES', 'BIGTIFF=IF_SAFER')\".\n    * Individual functions for raster and vector properties have been\n      aggregated into ``get_raster_info`` and ``get_vector_info``\n      respectively.\n    * Introducing ``warp_raster`` to wrap GDAL's ``ReprojectImage``\n      functionality that also works on bounding box clips.\n    * Removed the ``temporary_filename()`` paradigm.  Users should manage\n      temporary filenames directly.\n    * Numerous API changes from the 0.3.x version of PyGeoprocessing.\n* Fixing an issue with aggregate_raster_values that caused a crash if feature\n  IDs were not in increasing order starting with 0.\n* Removed \"create_rat/create_rat_uri\" and migrated it to\n  natcap.invest.wind_energy; the only InVEST model that uses that function.\n* Fixing an issue with aggregate_raster_values that caused a crash if feature\n  IDs were not in increasing order starting with 0.\n* Removed \"create_rat/create_rat_uri\" and migrated it to\n  natcap.invest.wind_energy; the only InVEST model that uses that function.\n\n0.3.3 (2017-02-09)\n------------------\n* Fixing a memory leak with large polygons when calculating disjoint set.\n\n0.3.2 (2017-01-24)\n------------------\n* Hotfix to patch an issue with watershed delineation packing that causes some\n  field values to lose precision due to default field widths being set.\n\n0.3.1 (2017-01-18)\n------------------\n* Hotfix patch to address an issue in watershed delineation that doesn't pack\n  the target watershed output file.  Half the shapefile consists of features\n  polygonalized around nodata values that are flagged for deletion, but not\n  removed from the file.  This patch packs those features and returns a clean\n  watershed.\n\n0.3.0 (2016-10-21)\n------------------\n* Added ``rel_tol`` and ``abs_tol`` parameters to ``testing.assertions`` to be\n  consistent with PEP485 and deal with real world testing situations that\n  required an absolute tolerance.\n* Removed calls to ``logging.basicConfig`` throughout pygeoprocessing.  Client\n  applications may need to adjust their logging if pygeoprocessing's log\n  messages are desired.\n* Added a flag  to ``aggregate_raster_values_uri`` that can be used to\n  indicate incoming polygons do not overlap, or the user does not care about\n  overlap. This can be used in cases where there is a computational or memory\n  bottleneck in calculating the polygon disjoint sets that would ultimately be\n  unnecessary if it is known a priori that such a check is unnecessary.\n* Fixed an issue where in some cases different nodata values for 'signal' and\n  'kernel' would cause incorrect convolution results in ``convolve_2d_uri``.\n* Added functionality to ``pygeoprocessing.iterblocks`` to iterate over\n  largest memory aligned block that fits into the number of elements provided\n  by the parameter.  With default parameters, this uses a ceiling around 16MB\n  of memory per band.\n* Added functionality to ``pygeoprocessing.iterblocks`` to return only the\n  offset dictionary.  This functionality would be used in cases where memory\n  aligned writes are desired without first reading arrays from the band.\n* Refactored ``pygeoprocessing.convolve_2d_uri`` to use ``iterblocks`` to take\n  advantage of large block sizes for FFT summing window method.\n* Refactoring source side to migrate source files from [REPO]/pygeoprocessing\n  to [REPO]/src/pygeoprocessing.\n* Adding a pavement script with routines to fetch SVN test data, build a\n  virtual environment, and clean the environment in a Windows based operating\n  system.\n* Adding ``transform_bounding_box`` to calculate the largest projected\n  bounding box given the four corners on a local coordinate system.\n* Removing GDAL, Shapely from the hard requirements in setup.py.  This will\n  allow pygeoprocessing to be built by package managers like pip without these\n  two packages being installed.  GDAL and Shapely will still need to be\n  installed for pygeoprocessing to run as expected.\n* Fixed a defect in ``pygeoprocessing.testing.assert_checksums_equal``\n  preventing BSD-style checksum files from being analyzed correctly.\n* Fixed an issue in reclassify_dataset_uri that would cause an exception if\n  the incoming raster didn't have a nodata value defined.\n* Fixed a defect in ``pygeoprocessing.geoprocessing.get_lookup_from_csv``\n  where the dialect was unable to be detected when analyzing a CSV that was\n  larger than 1K in size.  This fix enables the correct detection of comma or\n  semicolon delimited CSV files, so long as the header row by itself is not\n  larger than 1K.\n* Intra-package imports are now relative.  Addresses an import issue for users\n  with multiple copies of pygeoprocessing installed across multiple Python\n  installations.\n* Exposed cython routing functions so they may be imported from C modules.\n* ``get_lookup_from_csv`` attempts to determine the dialect of the CSV instead\n  of assuming comma delimited.\n* Added relative numerical tolerance parameters to the PyGeoprocessing raster\n  and csv tests with in the same API style as ``numpy.testing.allclose``.\n* Fixed an incomparability with GDAL 1.11.3 bindings that expects a boolean\n  type in ``band.ComputeStatistics``.  Before this fix PyGeoprocessing would\n  crash with a TypeError on many operations.\n* Fixed a defect in pygeoprocessing.routing.calculate_transport where the\n  nodata types were cast as int even though the base type of the routing\n  rasters were floats.  In extreme cases this could cause a crash on a type\n  that could not be converted to an int, like an ``inf``, and in subtle cases\n  this would result in nodata values in the raster being ignored during\n  routing.\n* Added functions to construct raster and vectors on disk from reasonable\n  datatypes (numpy matrices for rasters, lists of Shapely geometries for\n  vectors).\n* Fixed an issue where reproject_datasource_uri would add geometry that\n  couldn't be projected directly into the output datasource.  Function now\n  only adds geometries that transformed without error and reports if any\n  features failed to transform.\n* Added file flushing and dataset swig deletion in reproject_datasource_uri to\n  handle a race condition that might have been occurring.\n* Fixed an issue when \"None\" was passed in on new raster creation that would\n  attempt to directly set that value as the nodata value in the raster.\n* Added basic filetype-specific assertions for many geospatial filetypes, and\n  tests for these assertions.  These assertions are exposed in\n  ``pygeoprocessing.testing``.\n* Pygeoprocessing package tests can be run by invoking\n  ``python setup.py nosetests``.  A subset of tests may also be run from an\n  installed pygeoprocessing distribution by calling\n  ``pygeoprocessing.test()``.\n* Fixed an issue with reclassify dataset that would occur when small rasters\n  whose first memory block would extend beyond the size of the raster thus\n  passing in \"0\" values in the out of bounds area. Reclassify dataset\n  identified these as valid pixels, even though vectorize_datsets would mask\n  them out later.  Now vectorize_datasets only passes memory blocks that\n  contain valid pixel data to its kernel op.\n* Added support for very small AOIs that result in rasters less than a pixel\n  wide.  Additionally an ``all_touched`` flag was added to allow the\n  ALL_TOUCHED=TRUE option to be passed to RasterizeLayer in the AOI mask\n  calculation.\n* Added watershed delineation routine to\n  pygeoprocessing.routing.delineate_watershed.  Operates on a DEM and point\n  shapefile, optionally snaps outlet points to nearest stream as defined by a\n  thresholded flow accumulation raster and copies the outlet point fields into\n  the constructed watershed shapefile.\n* Fixing a memory leak in block caches that held on to dataset, band, and\n  block references even after the object was destroyed.\n* Add an option to route_flux that lets the current pixel's source be included\n  in the flux, or not.  Previous version would include on the source no matter\n  what.\n* Now using natcap.versioner for versioning instead of local versioning logic.\n\n0.2.2 (2015-05-07)\n------------------\n* Adding MinGW-specific compiler flags for statically linking pygeoprocessing\n  binaries against libstdc++ and libgcc.  Fixes an issue on many user's\n  computers when installing from a wheel on the Python Package Index without\n  having two needed DLLs on the PATH, resulting in an ImportError on pygeoprocessing.geoprocessing_core.pyd.\n* Fixing an issue with versioning where 'dev' was displayed instead of the\n  version recorded in pygeoprocessing/__init__.py.\n* Adding all pygeoprocessing.geoprocessing functions to\n  pygeoprocessing.__all__, which allows those functions to appear when\n  calling help(pygeoprocessing).\n* Adding routing_core.pxd to the manifest.  This fixes an issue where some\n  users were unable to compiler pygeoprocessing from source.\n\n0.2.1 (2015-04-23)\n------------------\n* Fixed a bug on the test that determines if a raster should be memory\n  blocked.  Rasters were not getting square blocked if the memory block was\n  row aligned.  Now creates 256x256 blocks on rasters larger than 256x256.\n* Updates to reclassify_dataset_uri to use numpy.digitize rather than Python\n  loops across the number of keys.\n* More informative error messages raised on incorrect bounding box mode.\n* Updated docstring on get_lookup_from_table to indicate the headers are case\n  insensitive.\n* Added updates to align dataset list that report which dataset is being\n  aligned.  This is helpful for logging feedback when many datasets are passed\n  in that don't take long enough to get a report from the underlying reproject\n  dataset function.\n* pygeoprocessing.routing.routing_core includes pxd to be ``cimport``\\able\n  from a Cython module.\n\n0.2.0 (2015-04-14)\n------------------\n* Fixed a library wide issue relating to the underlying numpy types of\n  GDT_Byte Datasets.  Now correctly identify the signed and unsigned versions\n  and removed all instances where code used to mod byte data to unsigned data\n  and correctly creates signed/unsigned byte datasets during resampling.\n* Removed extract_band_and_nodata function since it exposes the underlying\n  GDAL types.\n* Removed reclassify_by_dictionary since reclassify_dataset_uri provided\n  almost the same functionality and was widely used.\n* Removed the class OrderedDict that was not used.\n* Removed the function calculate_value_not_in_dataset since it loaded the\n  entire dataset into memory and was not useful.\n\n0.1.8 (2015-04-13)\n------------------\n* Fixed an issue on reclassifying signed byte rasters that had negative nodata\n  values but the internal type stored for vectorize datasets was unsigned.\n\n0.1.7 (2015-04-02)\n------------------\n* Package logger objects are now identified by python hierarchical package\n  paths (e.g. pygeoprocessing.routing)\n* Fixed an issue where rasters that had undefined nodata values caused\n  striping in the reclassify_dataset_uri function.\n\n0.1.6 (2015-03-24)\n------------------\n* Fixing LICENSE.TXT to .txt issue that keeps reoccurring.\n\n0.1.5 (2015-03-16)\n------------------\n* Fixed an issue where int32 dems with INT_MIN as the nodata value were being\n  treated as real DEM values because of an internal cast to a float for the\n  nodata type, but a cast to double for the DEM values.\n* Fixed an issue where flat regions, such as reservoirs, that could only drain\n  off the edge of the DEM now correctly drain as opposed to having undefined\n  flow directions.\n\n0.1.4 (2015-03-13)\n------------------\n* Fixed a memory issue for DEMs on the order of 25k X 25k, still may have\n  issues with larger DEMs.\n\n0.1.3 (2015-03-08)\n------------------\n* Fixed an issue so tox correctly executes on the repository.\n* Created a history file to document current and previous releases.\n* Created an informative README.rst.\n\n0.1.2 (2015-03-04)\n------------------\n* Fixing issue that caused \"LICENSE.TXT not found\" during pip install.\n\n0.1.1 (2015-03-04)\n------------------\n* Fixing issue with automatic versioning scheme.\n\n0.1.0 (2015-02-26)\n------------------\n* First release on PyPI.\n\n",
    "bugtrack_url": null,
    "license": "In this license, \"Natural Capital Project\" is defined as the parties of the Board of Trustees of the Leland Stanford Junior University, The Nature Conservancy, World Wildlife Fund Inc., and the Regents of the University of Minnesota.  This tool has an open license. All people are invited to use the tool under the following conditions and terms:  Copyright (c) 2021, Natural Capital Project  All rights reserved.  Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:  * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.  * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.  * Neither the name of Natural Capital Project nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission.  THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS \"AS IS\" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. ",
    "summary": "PyGeoprocessing: Geoprocessing routines for GIS",
    "version": "2.4.6",
    "project_urls": {
        "Homepage": "https://github.com/natcap/pygeoprocessing"
    },
    "split_keywords": [
        "gis",
        " geoprocessing",
        " pygeoprocessing"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "8487a956dcf2f217cdf678915946e23be004980448ed80b9cd9786767a79b072",
                "md5": "51b560119b0e6b7cb232e75c6a5e4b89",
                "sha256": "69d35cfc9dc057006ffa68d1f4af25944e22bdf569b97f54254f57ec3e17a32b"
            },
            "downloads": -1,
            "filename": "pygeoprocessing-2.4.6-cp310-cp310-macosx_10_9_universal2.whl",
            "has_sig": false,
            "md5_digest": "51b560119b0e6b7cb232e75c6a5e4b89",
            "packagetype": "bdist_wheel",
            "python_version": "cp310",
            "requires_python": ">=3.6",
            "size": 2372845,
            "upload_time": "2024-10-15T19:09:49",
            "upload_time_iso_8601": "2024-10-15T19:09:49.591411Z",
            "url": "https://files.pythonhosted.org/packages/84/87/a956dcf2f217cdf678915946e23be004980448ed80b9cd9786767a79b072/pygeoprocessing-2.4.6-cp310-cp310-macosx_10_9_universal2.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "2f8c0e9a1ed2859e4e2b5b5b96468ba968fc90c04a26917b6607f4119c636f01",
                "md5": "6857496dbc2133c4fc9a9cf7986923be",
                "sha256": "4c0e85c0693db39dc474d009df066eb0180deff2d17a17ab2dfef6bab87c2506"
            },
            "downloads": -1,
            "filename": "pygeoprocessing-2.4.6-cp310-cp310-win_amd64.whl",
            "has_sig": false,
            "md5_digest": "6857496dbc2133c4fc9a9cf7986923be",
            "packagetype": "bdist_wheel",
            "python_version": "cp310",
            "requires_python": ">=3.6",
            "size": 1523550,
            "upload_time": "2024-10-15T19:09:56",
            "upload_time_iso_8601": "2024-10-15T19:09:56.326760Z",
            "url": "https://files.pythonhosted.org/packages/2f/8c/0e9a1ed2859e4e2b5b5b96468ba968fc90c04a26917b6607f4119c636f01/pygeoprocessing-2.4.6-cp310-cp310-win_amd64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "1c0ab2820eae2510f46dab15d47251a983c2f097ea662b463afb4625824637ad",
                "md5": "305bee1e5a8d8b3211b4362d88729307",
                "sha256": "4b3ff119c7029ed5de1609f57b4522e61a68ec1e5b156e7f79c587a196aae138"
            },
            "downloads": -1,
            "filename": "pygeoprocessing-2.4.6-cp311-cp311-macosx_10_9_universal2.whl",
            "has_sig": false,
            "md5_digest": "305bee1e5a8d8b3211b4362d88729307",
            "packagetype": "bdist_wheel",
            "python_version": "cp311",
            "requires_python": ">=3.6",
            "size": 2370379,
            "upload_time": "2024-10-15T19:10:02",
            "upload_time_iso_8601": "2024-10-15T19:10:02.795164Z",
            "url": "https://files.pythonhosted.org/packages/1c/0a/b2820eae2510f46dab15d47251a983c2f097ea662b463afb4625824637ad/pygeoprocessing-2.4.6-cp311-cp311-macosx_10_9_universal2.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "dfc783d2052eb58e4891c287fc16edbf01cf60905c460c1ad8a4212e73e25ece",
                "md5": "a29def121ae52952688b38e2354ec4aa",
                "sha256": "7a65cb8594ce1f669dc182d6ead2ebc16ae7e1abd52c30e6f24c597e4dc9a00d"
            },
            "downloads": -1,
            "filename": "pygeoprocessing-2.4.6-cp311-cp311-win_amd64.whl",
            "has_sig": false,
            "md5_digest": "a29def121ae52952688b38e2354ec4aa",
            "packagetype": "bdist_wheel",
            "python_version": "cp311",
            "requires_python": ">=3.6",
            "size": 1526540,
            "upload_time": "2024-10-15T19:10:07",
            "upload_time_iso_8601": "2024-10-15T19:10:07.948280Z",
            "url": "https://files.pythonhosted.org/packages/df/c7/83d2052eb58e4891c287fc16edbf01cf60905c460c1ad8a4212e73e25ece/pygeoprocessing-2.4.6-cp311-cp311-win_amd64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "792385406646d6747eb5d3c65e33b6390f6b3747a7fc96d4f5afcf1d1843a0aa",
                "md5": "af0d28fdb5264618dba60e8fc76ace90",
                "sha256": "13e47dfd010c9889d8a86ab57223127d1277a99d795119a8b87254e976ec37f5"
            },
            "downloads": -1,
            "filename": "pygeoprocessing-2.4.6-cp312-cp312-macosx_10_13_universal2.whl",
            "has_sig": false,
            "md5_digest": "af0d28fdb5264618dba60e8fc76ace90",
            "packagetype": "bdist_wheel",
            "python_version": "cp312",
            "requires_python": ">=3.6",
            "size": 2337348,
            "upload_time": "2024-10-15T19:10:13",
            "upload_time_iso_8601": "2024-10-15T19:10:13.628124Z",
            "url": "https://files.pythonhosted.org/packages/79/23/85406646d6747eb5d3c65e33b6390f6b3747a7fc96d4f5afcf1d1843a0aa/pygeoprocessing-2.4.6-cp312-cp312-macosx_10_13_universal2.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "741b86b42aa0e83e03b824e8a6a4eccdb855ee10ec387bfe54ee7c68c80070a2",
                "md5": "0eb65a8ccbff11e9c7983bd6aa29b9f5",
                "sha256": "ff79db94834fd94ddd5e189341fa0000f309992bd7260138bead79219d55c981"
            },
            "downloads": -1,
            "filename": "pygeoprocessing-2.4.6-cp312-cp312-win_amd64.whl",
            "has_sig": false,
            "md5_digest": "0eb65a8ccbff11e9c7983bd6aa29b9f5",
            "packagetype": "bdist_wheel",
            "python_version": "cp312",
            "requires_python": ">=3.6",
            "size": 1504406,
            "upload_time": "2024-10-15T19:10:18",
            "upload_time_iso_8601": "2024-10-15T19:10:18.149713Z",
            "url": "https://files.pythonhosted.org/packages/74/1b/86b42aa0e83e03b824e8a6a4eccdb855ee10ec387bfe54ee7c68c80070a2/pygeoprocessing-2.4.6-cp312-cp312-win_amd64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "41ae0cbfeb6d83317965ac4bf1b15dadecc018a372ff36c01f70ad67e7ccbe9b",
                "md5": "5cdb4bbd113c408b210cc99102289ebe",
                "sha256": "efb38cecc65147960ea5a837ac9273ff7888403e754c8e56b63dda0713733c3b"
            },
            "downloads": -1,
            "filename": "pygeoprocessing-2.4.6-cp38-cp38-macosx_11_0_universal2.whl",
            "has_sig": false,
            "md5_digest": "5cdb4bbd113c408b210cc99102289ebe",
            "packagetype": "bdist_wheel",
            "python_version": "cp38",
            "requires_python": ">=3.6",
            "size": 2364430,
            "upload_time": "2024-10-15T19:10:24",
            "upload_time_iso_8601": "2024-10-15T19:10:24.918839Z",
            "url": "https://files.pythonhosted.org/packages/41/ae/0cbfeb6d83317965ac4bf1b15dadecc018a372ff36c01f70ad67e7ccbe9b/pygeoprocessing-2.4.6-cp38-cp38-macosx_11_0_universal2.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "8328aead6504867546fe74745a106aacd91c3ab8872d1f44dc50411ba5db5830",
                "md5": "8c67a9d14e134fe57fc8fe2d31fc1dcd",
                "sha256": "25e69dca1302815feefb7ed586a7da91fa115df7beb0d0a5afbc300d2d137c9e"
            },
            "downloads": -1,
            "filename": "pygeoprocessing-2.4.6-cp38-cp38-win_amd64.whl",
            "has_sig": false,
            "md5_digest": "8c67a9d14e134fe57fc8fe2d31fc1dcd",
            "packagetype": "bdist_wheel",
            "python_version": "cp38",
            "requires_python": ">=3.6",
            "size": 1537273,
            "upload_time": "2024-10-15T19:10:30",
            "upload_time_iso_8601": "2024-10-15T19:10:30.122365Z",
            "url": "https://files.pythonhosted.org/packages/83/28/aead6504867546fe74745a106aacd91c3ab8872d1f44dc50411ba5db5830/pygeoprocessing-2.4.6-cp38-cp38-win_amd64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "6ff6f95c08439e2b7cdab4872de3a009d2845d20ce88dd4417f563a0df5c154f",
                "md5": "8acfb039e3ec54e88da67d6b993d1d7f",
                "sha256": "7187e088ece3ded57084a1d659e0dfd765dabc98d2bb8c7f3f46b0f82fadacb2"
            },
            "downloads": -1,
            "filename": "pygeoprocessing-2.4.6-cp39-cp39-macosx_10_9_universal2.whl",
            "has_sig": false,
            "md5_digest": "8acfb039e3ec54e88da67d6b993d1d7f",
            "packagetype": "bdist_wheel",
            "python_version": "cp39",
            "requires_python": ">=3.6",
            "size": 2374075,
            "upload_time": "2024-10-15T19:10:36",
            "upload_time_iso_8601": "2024-10-15T19:10:36.491780Z",
            "url": "https://files.pythonhosted.org/packages/6f/f6/f95c08439e2b7cdab4872de3a009d2845d20ce88dd4417f563a0df5c154f/pygeoprocessing-2.4.6-cp39-cp39-macosx_10_9_universal2.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "908dd5cc0dc698dd017c132ccd0a6e80a8b6edb2ea6c85f27b670e2adfcd7a21",
                "md5": "de6e5763d9ac1365c5f7806f7007f0ee",
                "sha256": "f39e2f869ab82c1a2f285d44bde2cd1063b797fc1a15c67754c63962f1a8ddfd"
            },
            "downloads": -1,
            "filename": "pygeoprocessing-2.4.6-cp39-cp39-win_amd64.whl",
            "has_sig": false,
            "md5_digest": "de6e5763d9ac1365c5f7806f7007f0ee",
            "packagetype": "bdist_wheel",
            "python_version": "cp39",
            "requires_python": ">=3.6",
            "size": 1524188,
            "upload_time": "2024-10-15T19:10:40",
            "upload_time_iso_8601": "2024-10-15T19:10:40.370334Z",
            "url": "https://files.pythonhosted.org/packages/90/8d/d5cc0dc698dd017c132ccd0a6e80a8b6edb2ea6c85f27b670e2adfcd7a21/pygeoprocessing-2.4.6-cp39-cp39-win_amd64.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "9125760ce41928c4208a238efe37a183893e1bbf93c85437ff1938bb287e5d5c",
                "md5": "c95363f2ad6530396fbcea7a766a83cc",
                "sha256": "4a7440cfa7998b701313ef5026195d10737e9cc52b900b00a8ef54f62b0912dd"
            },
            "downloads": -1,
            "filename": "pygeoprocessing-2.4.6.tar.gz",
            "has_sig": false,
            "md5_digest": "c95363f2ad6530396fbcea7a766a83cc",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.6",
            "size": 1070615,
            "upload_time": "2024-10-15T19:10:44",
            "upload_time_iso_8601": "2024-10-15T19:10:44.119890Z",
            "url": "https://files.pythonhosted.org/packages/91/25/760ce41928c4208a238efe37a183893e1bbf93c85437ff1938bb287e5d5c/pygeoprocessing-2.4.6.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2024-10-15 19:10:44",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "natcap",
    "github_project": "pygeoprocessing",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "requirements": [],
    "tox": true,
    "lcname": "pygeoprocessing"
}
        
Elapsed time: 0.61165s