Skip to content
This repository was archived by the owner on Jun 6, 2025. It is now read-only.

A Django app for patching `django.contrib.gis` to lazily load the GDAL library

License

Notifications You must be signed in to change notification settings

joshuadavidthomas/django-lazy-gdal

Repository files navigation

django-lazy-gdal

PyPI PyPI - Python Version Django Version

A Django app that patches django.contrib.gis to lazily load the GDAL library, functioning as a drop-in replacement that follows the same pattern used for the GEOS library.

Note

Instead of fighting both making GDAL lazily loaded and which parts to monkeypatch to make this able to drop-in to a project, I decided to bite the bullet and upstream the changes needed to Django itself. Hopefully this will make it in a future release of Django.

Link to PR here.

Requirements

  • Python 3.9, 3.10, 3.11, 3.12, 3.13
  • Django 4.2, 5.1, 5.2

Installation

  1. Install the package from PyPI:

    python -m pip install django-lazy-gdal
    
    # or if you like the new hotness
    
    uv add django-lazy-gdal
    uv sync
  2. Import and call the monkeypatch function at the top of your Django project's settings module:

    # settings.py - add these lines at the TOP of the file
    import django_lazy_gdal
    
    
    django_lazy_gdal.monkeypatch()
    
    # ... rest of your settings file

Important

Timing matters! It's crucial to call django_lazy_gdal.monkeypatch() before any GeoDjango modules are imported. This is because Django imports models, which in turn imports the existing eager GDAL loading via imports in django.contrib.gis.models, before running app ready() methods. Calling the monkeypatch function at the top of your settings ensures the patching occurs before any other imports that might access django.contrib.gis.gdal.libgdal.

  1. That's it! The library will patch Django's django.contrib.gis.gdal.libgdal module to use lazy loading.

Why?

By default, Django's django.contrib.gis.gdal.libgdal module loads the GDAL library immediately upon import, which can cause issues if GDAL isn't installed. In contrast, the GEOS library in django.contrib.gis.geos.libgeos is lazily loaded using SimpleLazyObject, meaning errors are deferred until GEOS is actually used.

This provides several benefits:

  • Enables calling django.setup() in projects using django.contrib.gis without requiring GDAL installation, allowing for runtime interaction with non-GIS parts of the project
  • Prevents immediate import errors when GDAL isn't available, deferring them until actual GDAL use
  • Potentially reduces initial overhead by delaying GDAL library loading

This project originated from the need to work with GeoDjango projects without requiring GDAL to be installed locally. While developing django-language-server, I needed to set up Django to access information such as the template tags of the apps in INSTALLED_APPS.

However, I encountered errors when testing the language server on a project that uses GeoDjango but relies on Docker containers for development, thus bypassing the need to install Geospatial libraries locally.

This library uses monkeypatching as a temporary solution to modify Django's behavior regarding GDAL loading. The goal is to test the viability of lazily loading GDAL before proposing these changes upstream to Django. Hopefully, in the future, this library can be deprecated if/when this functionality is integrated into Django itself.

License

django-lazy-gdal is licensed under the MIT license. See the LICENSE file for more information.

About

A Django app for patching `django.contrib.gis` to lazily load the GDAL library

Resources

License

Code of conduct

Stars

Watchers

Forks

Sponsor this project

 

Packages

No packages published