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
|
# Copyright Cartopy Contributors
#
# This file is part of Cartopy and is released under the LGPL license.
# See COPYING and COPYING.LESSER in the root of the repository for full
# licensing details.
# NOTE: This file must remain Python 2 compatible for the foreseeable future,
# to ensure that we error out properly for people with outdated setuptools
# and/or pip.
import sys
PYTHON_MIN_VERSION = (3, 8)
if sys.version_info < PYTHON_MIN_VERSION:
error = """
Beginning with Cartopy 0.21, Python {} or above is required.
You are using Python {}.
This may be due to an out of date pip.
Make sure you have pip >= 9.0.1.
""".format('.'.join(str(n) for n in PYTHON_MIN_VERSION),
'.'.join(str(n) for n in sys.version_info[:3]))
sys.exit(error)
import os
import shutil
import subprocess
import warnings
from collections import defaultdict
from sysconfig import get_config_var
from setuptools import Extension, find_packages, setup
"""
Distribution definition for Cartopy.
"""
# The existence of a PKG-INFO directory is enough to tell us whether this is a
# source installation or not (sdist).
HERE = os.path.dirname(__file__)
IS_SDIST = os.path.exists(os.path.join(HERE, 'PKG-INFO'))
FORCE_CYTHON = os.environ.get('FORCE_CYTHON', False)
if not IS_SDIST or FORCE_CYTHON:
import Cython
if Cython.__version__ < '0.29':
raise ImportError(
"Cython 0.29+ is required to install cartopy from source.")
from Cython.Distutils import build_ext as cy_build_ext
try:
import numpy as np
except ImportError:
raise ImportError('NumPy 1.19+ is required to install cartopy.')
# Please keep in sync with INSTALL file.
GEOS_MIN_VERSION = (3, 7, 2)
def file_walk_relative(top, remove=''):
"""
Return a generator of files from the top of the tree, removing
the given prefix from the root/file result.
"""
top = top.replace('/', os.path.sep)
remove = remove.replace('/', os.path.sep)
for root, dirs, files in os.walk(top):
for file in files:
yield os.path.join(root, file).replace(remove, '')
# Dependency checks
# =================
# GEOS
try:
geos_version = subprocess.check_output(['geos-config', '--version'])
geos_version = tuple(int(v) for v in geos_version.split(b'.')
if 'dev' not in str(v))
geos_includes = subprocess.check_output(['geos-config', '--includes'])
geos_clibs = subprocess.check_output(['geos-config', '--clibs'])
except (OSError, ValueError, subprocess.CalledProcessError):
warnings.warn(
'Unable to determine GEOS version. Ensure you have %s or later '
'installed, or installation may fail.' % (
'.'.join(str(v) for v in GEOS_MIN_VERSION), ))
geos_includes = []
geos_library_dirs = []
geos_libraries = ['geos_c']
else:
if geos_version < GEOS_MIN_VERSION:
print('GEOS version %s is installed, but cartopy requires at least '
'version %s.' % ('.'.join(str(v) for v in geos_version),
'.'.join(str(v) for v in GEOS_MIN_VERSION)),
file=sys.stderr)
exit(1)
geos_includes = geos_includes.decode().split()
geos_libraries = []
geos_library_dirs = []
for entry in geos_clibs.decode().split():
if entry.startswith('-L'):
geos_library_dirs.append(entry[2:])
elif entry.startswith('-l'):
geos_libraries.append(entry[2:])
# Python dependencies
extras_require = {}
for name in os.listdir(os.path.join(HERE, 'requirements')):
with open(os.path.join(HERE, 'requirements', name)) as fh:
section, ext = os.path.splitext(name)
extras_require[section] = []
for line in fh:
if line.startswith('#'):
pass
elif line.startswith('-'):
pass
else:
extras_require[section].append(line.strip())
install_requires = extras_require.pop('default')
tests_require = extras_require.get('tests', [])
# General extension paths
if sys.platform.startswith('win'):
def get_config_var(name):
return '.'
include_dir = get_config_var('INCLUDEDIR')
library_dir = get_config_var('LIBDIR')
extra_extension_args = defaultdict(list)
if not sys.platform.startswith('win'):
extra_extension_args["runtime_library_dirs"].append(
get_config_var('LIBDIR')
)
# Description
# ===========
with open(os.path.join(HERE, 'README.md')) as fh:
description = ''.join(fh.readlines())
cython_coverage_enabled = os.environ.get('CYTHON_COVERAGE', None)
if cython_coverage_enabled:
extra_extension_args["define_macros"].append(
('CYTHON_TRACE_NOGIL', '1')
)
extensions = [
Extension(
'cartopy.trace',
['lib/cartopy/trace.pyx'],
include_dirs=([include_dir, './lib/cartopy', np.get_include()] +
geos_includes),
libraries=geos_libraries,
library_dirs=[library_dir] + geos_library_dirs,
language='c++',
**extra_extension_args),
]
if cython_coverage_enabled:
# We need to explicitly cythonize the extension in order
# to control the Cython compiler_directives.
from Cython.Build import cythonize
directives = {'linetrace': True,
'binding': True}
extensions = cythonize(extensions, compiler_directives=directives)
def decythonize(extensions, **_ignore):
# Remove pyx sources from extensions.
# Note: even if there are changes to the pyx files, they will be ignored.
for extension in extensions:
sources = []
for sfile in extension.sources:
path, ext = os.path.splitext(sfile)
if ext in ('.pyx',):
if extension.language == 'c++':
ext = '.cpp'
else:
ext = '.c'
sfile = path + ext
sources.append(sfile)
extension.sources[:] = sources
return extensions
if IS_SDIST and not FORCE_CYTHON:
extensions = decythonize(extensions)
cmdclass = {}
else:
cmdclass = {'build_ext': cy_build_ext}
# Main setup
# ==========
setup(
name='Cartopy',
url='https://scitools.org.uk/cartopy/docs/latest/',
download_url='https://github.com/SciTools/cartopy',
author='UK Met Office',
description='A cartographic python library with Matplotlib support for '
'visualisation',
long_description=description,
long_description_content_type='text/markdown',
license="LGPLv3",
keywords="cartography map transform projection proj proj.4 geos shapely "
"shapefile",
install_requires=install_requires,
extras_require=extras_require,
tests_require=tests_require,
use_scm_version={
'write_to': 'lib/cartopy/_version.py',
},
packages=find_packages("lib"),
package_dir={'': 'lib'},
package_data={'cartopy': list(file_walk_relative('lib/cartopy/tests/'
'mpl/baseline_images/',
remove='lib/cartopy/')) +
list(file_walk_relative('lib/cartopy/data/raster',
remove='lib/cartopy/')) +
list(file_walk_relative('lib/cartopy/data/netcdf',
remove='lib/cartopy/')) +
list(file_walk_relative('lib/cartopy/data/'
'shapefiles/gshhs',
remove='lib/cartopy/')) +
list(file_walk_relative('lib/cartopy/tests/lakes_shapefile',
remove='lib/cartopy/')) +
['io/srtm.npz']},
scripts=['tools/cartopy_feature_download.py'],
ext_modules=extensions,
cmdclass=cmdclass,
python_requires='>=' + '.'.join(str(n) for n in PYTHON_MIN_VERSION),
classifiers=[
'Development Status :: 4 - Beta',
'Framework :: Matplotlib',
'License :: OSI Approved :: GNU Lesser General Public License v3 '
'or later (LGPLv3+)',
'Operating System :: MacOS :: MacOS X',
'Operating System :: Microsoft :: Windows',
'Operating System :: POSIX',
'Operating System :: POSIX :: AIX',
'Operating System :: POSIX :: Linux',
'Programming Language :: C++',
'Programming Language :: Python',
'Programming Language :: Python :: 3',
'Programming Language :: Python :: 3.8',
'Programming Language :: Python :: 3.9',
'Programming Language :: Python :: 3.10',
'Programming Language :: Python :: 3 :: Only',
'Topic :: Scientific/Engineering',
'Topic :: Scientific/Engineering :: GIS',
'Topic :: Scientific/Engineering :: Visualization',
],
)
|