django1/django/contrib/gis/gdal/libgdal.py

117 lines
3.2 KiB
Python
Raw Normal View History

import logging
import os
2020-05-11 17:57:22 +08:00
import re
from ctypes import CDLL, CFUNCTYPE, c_char_p, c_int
from ctypes.util import find_library
from django.contrib.gis.gdal.error import GDALException
from django.core.exceptions import ImproperlyConfigured
logger = logging.getLogger('django.contrib.gis')
# Custom library path set?
try:
from django.conf import settings
lib_path = settings.GDAL_LIBRARY_PATH
except (AttributeError, ImportError, ImproperlyConfigured, OSError):
lib_path = None
if lib_path:
lib_names = None
elif os.name == 'nt':
# Windows NT shared libraries
2019-02-09 21:38:49 +08:00
lib_names = ['gdal204', 'gdal203', 'gdal202', 'gdal201', 'gdal20']
elif os.name == 'posix':
# *NIX library names.
2019-02-09 21:38:49 +08:00
lib_names = ['gdal', 'GDAL', 'gdal2.4.0', 'gdal2.3.0', 'gdal2.2.0', 'gdal2.1.0', 'gdal2.0.0']
else:
raise ImproperlyConfigured('GDAL is unsupported on OS "%s".' % os.name)
2012-06-11 05:59:50 +08:00
# Using the ctypes `find_library` utility to find the
# path to the GDAL library from the list of library names.
if lib_names:
for lib_name in lib_names:
lib_path = find_library(lib_name)
if lib_path is not None:
break
2012-06-11 05:59:50 +08:00
if lib_path is None:
raise ImproperlyConfigured(
'Could not find the GDAL library (tried "%s"). Is GDAL installed? '
'If it is, try setting GDAL_LIBRARY_PATH in your settings.'
% '", "'.join(lib_names)
2016-03-29 06:33:29 +08:00
)
# This loads the GDAL/OGR C library
lgdal = CDLL(lib_path)
2012-06-11 05:59:50 +08:00
# On Windows, the GDAL binaries have some OSR routines exported with
# STDCALL, while others are not. Thus, the library will also need to
# be loaded up as WinDLL for said OSR functions that require the
# different calling convention.
if os.name == 'nt':
from ctypes import WinDLL
lwingdal = WinDLL(lib_path)
2013-11-03 04:12:09 +08:00
def std_call(func):
"""
Return the correct STDCALL function for certain OSR routines on Win32
platforms.
"""
if os.name == 'nt':
return lwingdal[func]
else:
return lgdal[func]
2016-11-13 01:11:23 +08:00
2015-02-06 02:25:34 +08:00
# #### Version-information functions. ####
# Return GDAL library version information with the given key.
_version_info = std_call('GDALVersionInfo')
_version_info.argtypes = [c_char_p]
_version_info.restype = c_char_p
2013-11-03 04:12:09 +08:00
def gdal_version():
"Return only the GDAL version number information."
2012-09-24 01:59:27 +08:00
return _version_info(b'RELEASE_NAME')
2013-11-03 04:12:09 +08:00
2012-06-11 05:59:50 +08:00
def gdal_full_version():
"Return the full GDAL version information."
return _version_info(b'')
2016-11-13 01:11:23 +08:00
def gdal_version_info():
2020-05-11 17:57:22 +08:00
ver = gdal_version()
m = re.match(br'^(?P<major>\d+)\.(?P<minor>\d+)(?:\.(?P<subminor>\d+))?', ver)
if not m:
raise GDALException('Could not parse GDAL version string "%s"' % ver)
2020-05-11 17:57:22 +08:00
major, minor, subminor = m.groups()
return (int(major), int(minor), subminor and int(subminor))
2016-11-13 01:11:23 +08:00
2020-05-11 17:57:22 +08:00
GDAL_VERSION = gdal_version_info()
# Set library error handling so as errors are logged
CPLErrorHandler = CFUNCTYPE(None, c_int, c_int, c_char_p)
2013-11-03 04:12:09 +08:00
def err_handler(error_class, error_number, message):
logger.error('GDAL_ERROR %d: %s', error_number, message)
2016-11-13 01:11:23 +08:00
err_handler = CPLErrorHandler(err_handler)
2013-11-03 04:12:09 +08:00
def function(name, args, restype):
func = std_call(name)
func.argtypes = args
func.restype = restype
return func
2016-11-13 01:11:23 +08:00
set_error_handler = function('CPLSetErrorHandler', [CPLErrorHandler], CPLErrorHandler)
set_error_handler(err_handler)