File: setup.py

package info (click to toggle)
rasterio 1.5.0-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 22,732 kB
  • sloc: python: 23,119; sh: 947; makefile: 275; xml: 29
file content (399 lines) | stat: -rwxr-xr-x 14,966 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
#!/usr/bin/env python

# These following environmental variables influence this script:
#
# GDAL_CONFIG: the path to a gdal-config program that points to GDAL headers,
# libraries, and data files.
#
# PACKAGE_DATA: if defined, GDAL and PROJ4 data files will be copied into the
# source or binary distribution. This is essential when creating self-contained
# binary wheels.
#
# PATH: if we don't find `gdal-config`, we instead search for the `gdalinfo`
# executable using the `PATH` environment variable. We run it to get the GDAL
# version, and we compute the include, library, share, etc. directory paths
# from that executable's path.

import copy
import logging
import os
import platform
import pprint
import re
import shutil
import sys
from subprocess import check_output

from setuptools import setup
from setuptools.extension import Extension

logging.basicConfig(stream=sys.stderr, level=logging.INFO)
log = logging.getLogger()


def copy_data_tree(datadir, destdir):
    try:
        shutil.rmtree(destdir)
    except OSError:
        pass
    shutil.copytree(datadir, destdir)


# python -W all setup.py ...
if "all" in sys.warnoptions:
    log.level = logging.DEBUG

# Use Cython if available.
try:
    from Cython.Build import cythonize
except ImportError:
    raise SystemExit(
        "ERROR: Cython.Build.cythonize not found. "
        "Cython is required to build rasterio.")

# By default we'll try to get options via gdal-config. On systems without,
# options will need to be set in setup.cfg, on the setup command line, or through
# the GDAL_INSTALL_PREFIX environment variable.
include_dirs: list[str] = []
library_dirs: list[str] = []
libraries: list[str] = []
extra_link_args: list[str] = []
gdal2plus: bool = False
gdal_output: list[str | None] = [None] * 4
gdalversion: str | None = None
gdal_major_version: int = 0
gdal_minor_version: int = 0
gdal_patch_version: int = 0
gdal_data_dir: str | None = None

try:
    import numpy as np

    include_dirs.append(np.get_include())
except ImportError:
    raise SystemExit("ERROR: Numpy and its headers are required to run setup().")


def _find_executable(executable_name: str) -> str | None:
    """Find the given executable in PATH.

    Return the absolute path to the executable if found, otherwise None.
    """

    log.info("Searching for executable %s on sys.prefix")
    executable_path = shutil.which(executable_name, path=sys.prefix)
    if executable_path is None:
        log.info("Did not find executable %s on sys.prefix, searching on PATH")
        executable_path = shutil.which(executable_name, path=os.environ["PATH"])
    if executable_path is None:
        log.info("Did not find executable %s on sys.prefix or on PATH", executable_name)
        return None
    executable_path = os.path.abspath(executable_path)
    log.info("Found executable %s at path %s", executable_name, executable_path)
    return executable_path


def find_gdal_install_with_executable(
    executable_name: str = "gdalinfo",
) -> tuple[str, tuple[int, int, int]] | None:
    """Find the given GDAL executable and get the GDAL version and install directory from it.

    We first search on `sys.prefix`, then on `os.environ["PATH"]`.

    executable_name should be the name of a GDAL executable, like `"gdalinfo"`, with or without extension.
    It should be executable on the command line through this name.

    Return the absolute path to GDAL's install path (above bin, include, share, and lib) and the major, minor, and patch
    numbers of GDAL's version, if the executable was found. Otherwise, return None.

    This does not check if the install is complete.
    """

    executable_path = _find_executable(executable_name)
    # Run it to get the GDAL version
    result = check_output([executable_name, "--version"], text=True).strip()
    version_match = re.match(
        r"^GDAL (\d+)\.(\d+)\.(\d+) .*$", result, flags=re.MULTILINE
    )
    if version_match:
        gdal_version: tuple[int, int, int] = (
            int(version_match.group(1)),
            int(version_match.group(2)),
            int(version_match.group(3)),
        )
    else:
        raise RuntimeError(
            "Could not parse GDAL version from output of `%s --version`",
            executable_path,
        )
    # Move up two levels: parent of the `bin` folder
    gdal_install_prefix = os.path.dirname(os.path.dirname(executable_path))
    return gdal_install_prefix, gdal_version

def fill_gdal_build_options_using_executable(executable_name: str) -> None:
    """Fill the global GDAL build options using information from the provided install path.

    executable_name should be the name of a GDAL executable, like `"gdalinfo"`, with or without extension.
    It should be executable on the command line through this name.

    Raise `RuntimeError` if we don't find a GDAL install.
    """

    global \
        gdal_data_dir, \
        gdalversion, \
        gdal_major_version, \
        gdal_minor_version, \
        gdal_patch_version

    prefix_search_result = find_gdal_install_with_executable(
        executable_name=executable_name
    )
    if prefix_search_result is None:
        raise RuntimeError(
            'Could not find the GDAL install directory using executable "%s"',
            executable_name,
        )

    gdal_install_prefix, gdal_version_numbers = prefix_search_result

    # Find library directory
    found_library_dir = os.path.join(gdal_install_prefix, "lib")
    if not os.path.isdir(found_library_dir):
        raise RuntimeError("Could not find library directory at %s", found_library_dir)

    # Does the GDAL library file exist?
    found_gdal_lib = os.path.join(
        found_library_dir, "gdal.lib" if os.name == "nt" else "gdal.so"
    )
    if not os.path.isfile(found_gdal_lib):
        raise RuntimeError("Could not find GDAL library file at %s", found_library_dir)
    found_gdal_lib_name = os.path.splitext(os.path.basename(found_gdal_lib))[0]

    # Find include directory
    found_include_dir = os.path.join(gdal_install_prefix, "include")
    if not os.path.isdir(found_include_dir):
        raise RuntimeError("Could not find include directory at %s", found_include_dir)
    # Check that at least one header exists
    found_gdal_header = os.path.join(found_include_dir, "gdal.h")
    if not os.path.isfile(found_gdal_header):
        raise RuntimeError("Could not find GDAL header at %s", found_gdal_header)

    # Find GDAL_DATA directory
    found_gdal_data_dir = os.path.join(gdal_install_prefix, "share", "gdal")
    if not os.path.isdir(found_gdal_data_dir):
        raise RuntimeError(
            "Could not find GDAL_DATA directory at %s", found_gdal_data_dir
        )

    log.info(
        "Found a GDAL install at %s that seems complete, using it", gdal_install_prefix
    )
    library_dirs.append(found_library_dir)
    include_dirs.append(found_include_dir)
    gdalversion = ".".join(map(str, gdal_version_numbers))
    gdal_major_version, gdal_minor_version, gdal_patch_version = gdal_version_numbers
    libraries.append(found_gdal_lib_name)
    gdal_data_dir = found_gdal_data_dir

def fill_gdal_build_options_using_gdal_config() -> None:
    """Run gdal-config and fill the global build options using its output.

    If gdal-config's path isn't found, raise `FileNotFoundError`.
    If it fails, raise `CalledProcessError`.
    """

    global gdalversion

    log.info("Using gdal-config to get GDAL build options")
    gdal_config = os.environ.get("GDAL_CONFIG", _find_executable("gdal-config"))
    for i, flag in enumerate(("--cflags", "--libs", "--datadir", "--version")):
        try:
            gdal_output[i] = check_output([gdal_config, flag]).decode("utf-8").strip()
        except FileNotFoundError as error:
            raise FileNotFoundError(
                'gdal-config not found under the name "%s"', gdal_config
            ) from error

    for item in gdal_output[0].split():
        if item.startswith("-I"):
            include_dirs.extend(item[2:].split(":"))
    for item in gdal_output[1].split():
        if item.startswith("-L"):
            library_dirs.extend(item[2:].split(":"))
        elif item.startswith("-l"):
            libraries.append(item[2:])
        else:
            # e.g. -framework GDAL
            extra_link_args.append(item)
    # datadir, gdal_output[2] handled below

    gdalversion = gdal_output[3]
    if gdalversion:
        log.info("GDAL API version obtained from gdal-config: %s", gdalversion)

if "clean" not in sys.argv:
    try:
        fill_gdal_build_options_using_gdal_config()
    except Exception as e:
        # Try to run gdalinfo and get information from that instead
        log.info(
            "Failed to use gdal-config, trying to run gdalinfo instead (gdal-config error of type %s: %s)",
            type(e).__name__,
            e,
        )
        try:
            fill_gdal_build_options_using_executable(executable_name="gdalinfo")
        except Exception as e:
            log.warning(
                "Failed to get options via both gdal-config and gdalinfo. (gdalinfo error of type %s: %s)",
                type(e).__name__,
                e,
            )

    # Get GDAL API version from environment variable.
    if 'GDAL_VERSION' in os.environ:
        gdalversion = os.environ['GDAL_VERSION']
        log.info("GDAL API version obtained from environment: %s", gdalversion)

    # Get GDAL API version from the command line if specified there.
    if '--gdalversion' in sys.argv:
        index = sys.argv.index('--gdalversion')
        sys.argv.pop(index)
        gdalversion = sys.argv.pop(index)
        log.info("GDAL API version obtained from command line option: %s",
                 gdalversion)

    if not gdalversion:
        raise SystemExit("ERROR: A GDAL API version must be specified. Provide a path "
                 "to gdal-config using a GDAL_CONFIG environment variable "
                 "or use a GDAL_VERSION environment variable.")

    gdal_major_version, gdal_minor_version, gdal_patch_version = map(
        int, re.findall("[0-9]+", gdalversion)[:3]
    )

    if (gdal_major_version, gdal_minor_version) < (3, 8):
        raise SystemExit("ERROR: GDAL >= 3.8 is required for rasterio. "
                 "Please upgrade GDAL.")

# Conditionally copy the GDAL data. To be used in conjunction with
# the bdist_wheel command to make self-contained binary wheels.
if os.environ.get("PACKAGE_DATA"):
    gdal_data_destination = "rasterio/gdal_data"
    if gdal_data_dir is None:
        if gdal_output[2]:
           gdal_data_dir = gdal_output[2]
        else:
            # check to see if GDAL_DATA is defined
            gdal_data_dir = os.environ.get('GDAL_DATA')
    if gdal_data_dir is not None and os.path.exists(gdal_data_dir):
        log.info("Copying GDAL_DATA from %s" % gdal_data_dir)
        copy_data_tree(gdal_data_dir, gdal_data_destination)
    else:
        raise SystemExit("GDAL_DATA directory not found. Unable to package data in wheels.")

    # Conditionally copy PROJ DATA.
    proj_data_dir = os.environ.get('PROJ_DATA', os.environ.get('PROJ_LIB', '/usr/local/share/proj'))
    if os.path.exists(proj_data_dir):
        log.info("Copying PROJ_DATA from %s" % proj_data_dir)
        copy_data_tree(proj_data_dir, 'rasterio/proj_data')
    else:
        raise SystemExit("PROJ_DATA directory not found. Unable to package data in wheels.")
else:
    log.info("Not copying GDAL data to the final package")

compile_time_env = {
    "CTE_GDAL_MAJOR_VERSION": gdal_major_version,
    "CTE_GDAL_MINOR_VERSION": gdal_minor_version,
    "CTE_GDAL_PATCH_VERSION": gdal_patch_version,
}

ext_options = {
    'include_dirs': include_dirs,
    'library_dirs': library_dirs,
    'libraries': libraries,
    'extra_link_args': extra_link_args,
    'define_macros': [],
    'cython_compile_time_env': compile_time_env
}

if not os.name == "nt":
    # These options fail on Windows if using Visual Studio
    ext_options['extra_compile_args'] = ['-Wno-unused-parameter',
                                         '-Wno-unused-function']

# Copy extension options for cpp extension modules.
cpp_ext_options = copy.deepcopy(ext_options)

# Remove -std=c++11 from C extension options.
try:
    ext_options['extra_link_args'].remove('-std=c++11')
    ext_options['extra_compile_args'].remove('-std=c++11')
except Exception:
    pass

cpp11_flag = '-std=c++11'

# 'extra_compile_args' may not be defined
eca = cpp_ext_options.get('extra_compile_args', [])

if platform.system() == 'Darwin':

    if cpp11_flag not in eca:
        eca.append(cpp11_flag)

    eca += [cpp11_flag, '-mmacosx-version-min=10.9', '-stdlib=libc++']

# TODO: Windows

elif cpp11_flag not in eca:
    eca.append(cpp11_flag)

cpp_ext_options['extra_compile_args'] = eca

# Configure optional Cython coverage.
cythonize_options = {"language_level": sys.version_info[0], "compiler_directives": {"freethreading_compatible": True}}
if os.environ.get('CYTHON_COVERAGE'):
    cythonize_options['compiler_directives'].update(linetrace=True)
    cythonize_options['annotate'] = True
    ext_options['define_macros'].extend(
        [('CYTHON_TRACE', '1'), ('CYTHON_TRACE_NOGIL', '1')])

log.debug('ext_options:\n%s', pprint.pformat(ext_options))

ext_modules = None
if "clean" not in sys.argv:
    extensions = [
        Extension("rasterio._base", ["rasterio/_base.pyx"], **ext_options),
        Extension("rasterio._io", ["rasterio/_io.pyx"], **ext_options),
        Extension("rasterio._features", ["rasterio/_features.pyx"], **ext_options),
        Extension("rasterio._env", ["rasterio/_env.pyx"], **ext_options),
        Extension("rasterio._warp", ["rasterio/_warp.pyx"], **cpp_ext_options),
        Extension("rasterio._fill", ["rasterio/_fill.pyx"], **cpp_ext_options),
        Extension("rasterio._err", ["rasterio/_err.pyx"], **ext_options),
        Extension("rasterio._example", ["rasterio/_example.pyx"], **ext_options),
        Extension("rasterio._version", ["rasterio/_version.pyx"], **ext_options),
        Extension("rasterio.cache", ["rasterio/cache.pyx"], **ext_options),
        Extension("rasterio.crs", ["rasterio/crs.pyx"], **ext_options),
        Extension("rasterio.shutil", ["rasterio/shutil.pyx"], **ext_options),
        Extension("rasterio._transform", ["rasterio/_transform.pyx"], **ext_options),
        Extension("rasterio._filepath", ["rasterio/_filepath.pyx"], **cpp_ext_options),
        Extension(
            "rasterio._vsiopener", ["rasterio/_vsiopener.pyx"], **ext_options
        ),
    ]
    ext_modules = cythonize(
        extensions, quiet=True, compile_time_env=compile_time_env, **cythonize_options
    )

setup_args = {}
if os.environ.get('PACKAGE_DATA'):
    setup_args['package_data'] = {'rasterio': ['gdal_data/*', 'proj_data/*']}

# See pyproject.toml for project metadata
setup(
    name="rasterio",  # need by GitHub dependency graph
    ext_modules=ext_modules,
    **setup_args,
)