-
Notifications
You must be signed in to change notification settings - Fork 56
/
Copy pathsetup.py
407 lines (361 loc) · 18.6 KB
/
setup.py
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
400
401
402
403
404
405
406
407
#!/usr/bin/python
# -*- coding: utf-8 -*-
# This program is free software; you can redistribute it and/or modify it under
# the terms of the (LGPL) GNU Lesser General Public License as published by the
# Free Software Foundation; either version 3 of the License, or (at your
# option) any later version.
#
# This program is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
# FOR A PARTICULAR PURPOSE. See the GNU Library Lesser General Public License
# for more details at ( http://www.gnu.org/licenses/lgpl.html ).
#
# You should have received a copy of the GNU Lesser General Public License
# along with this program; if not, write to the Free Software Foundation, Inc.,
# 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
# written by: Jurko Gospodnetić ( jurko.gospodnetic@pke.hr )
"""
Main installation and project management script for this project.
Requires setuptools. Having setuptools available provides us with the following
benefits:
- setup.py 'egg_info' command constructing the project's metadata
- setup.py 'develop' command deploying the project in 'development mode',
thus making it available on sys.path, yet still editable directly in its
source checkout - equivalent to running a pip based installation using
'easy_install -e' or 'pip install -e'
- setup.py 'test' command as a standard way to run the project's test suite
- when running the installation directly from the source tree setup.py
'install' command:
- automatically installs the project's metadata so package management
tools like pip recognize the installation correctly, e.g. this allows
using 'pip uninstall' to undo the installation
- package installed as a zipped egg by default
"""
import sys
import os
import os.path
import re
# -----------------------------------------------------------------------------
# Global variables.
# -----------------------------------------------------------------------------
distutils_cmdclass = {}
# -----------------------------------------------------------------------------
# Detect the setup.py environment - current & script folder.
# -----------------------------------------------------------------------------
# Setup documentation incorrectly states that it will search for packages
# relative to the setup script folder by default when in fact it will search
# for them relative to the current working folder. It seems avoiding this
# problem cleanly and making the setup script runnable with any current working
# folder would require better distutils and/or setuptools support.
# Attempted alternatives:
# * Changing the current working folder internally makes any passed path
# parameters be interpreted relative to the setup script folder when they
# should be interpreted relative to the initial current working folder.
# * Passing the script folder to setup() using the parameter
# package_dir={"": script_folder}
# makes the setup 'build' command work from any folder but 'egg-info'
# (together with any related commands) still fails.
script_folder = os.path.realpath(os.path.dirname(__file__))
current_folder = os.path.realpath(os.getcwd())
if script_folder != current_folder:
print("ERROR: Suds library setup script needs to be run from the folder "
"containing it.")
print("")
print("Current folder: %s" % current_folder)
print("Script folder: %s" % script_folder)
sys.exit(-2)
# -----------------------------------------------------------------------------
# Import suds_devel module shared between setup & development scripts.
# -----------------------------------------------------------------------------
tools_folder = os.path.join(script_folder, "tools")
sys.path.insert(0, tools_folder)
import suds_devel
sys.path.pop(0)
# -----------------------------------------------------------------------------
# Attempt to use setuptools for this installation.
# -----------------------------------------------------------------------------
# setuptools brings us several useful features (see the main setup script
# docstring) but if it causes problems for our users or even only
# inconveniences them, they tend to complain about why we are using setuptools
# at all. Therefore we try to use setuptools as silently as possible, with a
# clear error message displayed to the user, but only in cases when we
# absolutely need setuptools.
#
# Setuptools usage logic:
# 1. attempt to use a preinstalled setuptools version
# 2. if a preinstalled setuptools version is not available, attempt to install
# and use the most recent tested compatible setuptools release, possibly
# downloading the installation from PyPI into the current folder
# 3. if we still do not have setuptools available, fall back to using distutils
#
# Note that we have made a slight trade-off here and chose to reuse an existing
# setuptools installation if available instead of always downloading and
# installing the most recent compatible setuptools release.
#
# Alternative designs and rationale for selecting the current design:
#
# distutils/setuptools usage:
# * always use distutils
# - misses out on all the setuptools features
# * use setuptools if available, or fall back to using distutils
# - chosen design
# - gets us setuptools features if available, with clear end-user error
# messages in case an operation is triggered that absolutely requires
# setuptools to be available
# - see below for notes on different setuptools installation alternatives
# * always use setuptools
# - see below for notes on different setuptools installation alternatives
#
# setuptools installation:
# * expect setuptools to be preinstalled
# - if not available, burden the user with installing setuptools manually
# - see below for notes on using different setuptools versions
# * use preinstalled setuptools if possible, or fall back to installing it
# on-demand
# - see below for notes on using different setuptools versions
# - automated setuptools installations, and especially in-place upgrades,
# can fail for various reasons (see below)
# - reduces the risk of a stalled download stalling the whole setup
# operation, e.g. because of an unavailable or unresponsive DNS server
# * always install setuptools
# - chosen design
# - automated setuptools installations, and especially in-place upgrades,
# can fail for various reasons (see below)
# - user has no way to avoid setuptools installation issues by installing
# setuptools himself, which would force us to make our setuptools
# installation support universally applicable and that is just not
# possible, e.g. users might be wanting to use customized package
# management or their own package index instead of PyPI.
#
# setuptools version usage:
# - basic problem here is that our project can be and has been tested only
# with certain setuptools versions
# - using a different version may have issues causing our setup procedures to
# fail outside our control, either due to a setuptools bug or due to an
# incompatibility between our implementation and the used setuptools
# version
# - some setuptools releases have known regressions, e.g. setuptools 3.0
# which is documented to fail on Python 2.6 due to an accidental backward
# incompatibility corrected in the setuptools 3.1 release
# * allow using any setuptools version
# - chosen design
# - problems caused by incompatible setuptools version usage are
# considered highly unlikely and can therefore be patched up as needed
# when and if they appear
# - users will most likely not have a setuptools version preinstalled
# into their Python environment that is incompatible with that
# environment
# - if there is no setuptools version installed, our setup will attempt
# to install the most recent tested setuptools release
# * allow using only tested setuptools versions or possibly more recent ones
# - unless we can automatically install a suitable setuptools version, we
# will need to burden the user with installing it manually or fall back
# to using distutils
# - automated setuptools installations, and especially in-place upgrades,
# can fail for various reasons (see below)
# - extra implementation effort required compared to the chosen design,
# with no significant benefit
#
# Some known scenarios causing automated setuptools installation failures:
# * Download failures, e.g. because user has no access to PyPI.
# * In-place setuptool upgrades can fail due to a known setuptools issue (see
# 'https://bitbucket.org/pypa/setuptools/issue/168') when both the original
# and the new setuptools version is installed as a zipped egg distribution.
# Last seen using setuptools 3.4.4.
# * If the Python environment running our setup has already loaded setuptools
# packages, then upgrading that installation in-place will fail with an
# error message instructing the user to do the upgrade manually.
# * When installing our project using pip, pip will load setuptools
# internally, and it typically uses an older setuptools version which can
# trigger the in-place upgrade failure as described above. What is worse,
# we ignore this failure, we run into the following combination problem:
# * pip calls our setup twice - once to collect the package requirement
# information and once to perform the actual installation, and we do
# not want to display multiple potentially complex error messages to
# user for what is effectively the same error.
# * Since we can not affect how external installers call our setup, to
# avoid this we would need to either:
# * Somehow cache the information that we already attempted and
# failed to upgrade setuptools (complicated + possibly not robust).
# * Patch the setuptools installation script to not display those
# error messages (we would prefer to not be forced to maintain our
# own patches for this script and use it as is).
# * Avoid the issue by never upgrading an existing setuptools
# installation (chosen design).
def acquire_setuptools_setup():
try:
from setuptools import setup
except ImportError:
print("WARNING: setuptools is not detected. It is required, install via: "
"python -m pip install --upgrade pip setuptools wheel")
return
return setup
setup = acquire_setuptools_setup()
using_setuptools = bool(setup)
# -----------------------------------------------------------------------------
# Support functions.
# -----------------------------------------------------------------------------
def read_python_code(filename):
"Returns the given Python source file's compiled content."
file = open(filename, "rt")
try:
source = file.read()
finally:
file.close()
# Python 2.6 and below did not support passing strings to exec() &
# compile() functions containing line separators other than '\n'. To
# support them we need to manually make sure such line endings get
# converted even on platforms where this is not handled by native text file
# read operations.
source = source.replace("\r\n", "\n").replace("\r", "\n")
return compile(source, filename, "exec")
def recursive_package_list(*packages):
"""
Returns a list of all the given packages and all their subpackages.
Given packages are expected to be found relative to this script's location.
Subpackages are detected by scanning the given packages' subfolder
hierarchy for any folders containing the '__init__.py' module. As a
consequence, namespace packages are not supported.
This is our own specialized setuptools.find_packages() replacement so we
can avoid the setuptools dependency.
"""
result = set()
todo = []
for package in packages:
folder = os.path.join(script_folder, *package.split("."))
if not os.path.isdir(folder):
raise Exception("Folder not found for package '%s'." % (package,))
todo.append((package, folder))
while todo:
package, folder = todo.pop()
if package in result:
continue
result.add(package)
for subitem in os.listdir(folder):
subpackage = ".".join((package, subitem))
subfolder = os.path.join(folder, subitem)
if not os.path.isfile(os.path.join(subfolder, "__init__.py")):
continue
todo.append((subpackage, subfolder))
return list(result)
# Shamelessly stolen from setuptools project's pkg_resources module.
def safe_version(version_string):
"""
Convert an arbitrary string to a standard version string
Spaces become dots, and all other non-alphanumeric characters become
dashes, with runs of multiple dashes condensed to a single dash.
"""
version_string = version_string.replace(" ", ".")
return re.sub("[^A-Za-z0-9.]+", "-", version_string)
def unicode2ascii(unicode):
"""Convert a unicode string to its approximate ASCII equivalent."""
return unicode.encode("ascii", 'xmlcharrefreplace').decode("ascii")
# -----------------------------------------------------------------------------
# Load the suds library version information.
# -----------------------------------------------------------------------------
# Load the suds library version information directly into this module without
# having to import the whole suds library itself. Importing the suds package
# would have caused problems like the following:
# * Forcing the imported package module to be Python 3 compatible without any
# lib2to3 fixers first being run on it (since such fixers get run only
# later as a part of the setup procedure).
# * Making the setup module depend on the package module's dependencies, thus
# forcing the user to install them manually (since the setup procedure that
# is supposed to install them automatically will not be able to run unless
# they are already installed).
# We execute explicitly compiled source code instead of having the exec()
# function compile it to get a better error messages. If we used exec() on the
# source code directly, the source file would have been listed as just
# '<string>'.
exec(read_python_code(os.path.join(script_folder, "suds", "version.py")))
# -----------------------------------------------------------------------------
# Avoid setup warnings when constructing a list of all project sources.
# -----------------------------------------------------------------------------
# Part of this workaround implemented and part in the project's MANIFEST.in
# template. See a related comment in MANIFEST.in for more detailed information.
dummy_tools_folder = os.path.join(tools_folder, "__dummy__")
dummy_tools_file = os.path.join(dummy_tools_folder, "readme.txt")
try:
if not os.path.isdir(dummy_tools_folder):
os.mkdir(dummy_tools_folder)
if not os.path.isfile(dummy_tools_file):
f = open(dummy_tools_file, "w")
try:
f.write("""\
Dummy empty folder added as a part of a patch to silence setup.py warnings when
determining which files belong to the project. See a related comment in the
project's MANIFEST.in template for more detailed information.
Both the folder and this file have been generated by the project's setup.py
script and should not be placed under version control.
""")
finally:
f.close()
except EnvironmentError:
# Something went wrong attempting to construct the dummy file. Ah well, we
# gave it our best. Continue on with possible spurious warnings.
pass
# -----------------------------------------------------------------------------
# Set up project metadata and run the actual setup.
# -----------------------------------------------------------------------------
# Package meta-data needs may be specified as:
# * Python 3.2.2+ - unicode string
# - unicode strings containing non-ASCII characters supported since Python
# commit fb4d2e6d393e96baac13c4efc216e361bf12c293
# Wrap long_description at 72 characters since the PKG-INFO package
# distribution metadata file stores this text with an 8 space indentation.
long_description = """
---------------------------------------
Lightweight SOAP client (Community fork).
---------------------------------------
Based on the original 'suds' project by Jeff Ortel (jortel at redhat
dot com) hosted at 'http://fedorahosted.org/suds'.
'Suds' is a lightweight SOAP-based web service client for Python
licensed under LGPL (see the LICENSE.txt file included in the
distribution).
"""
forked_package_name = 'suds-community'
package_name = os.environ.get('SUDS_PACKAGE', forked_package_name)
version_tag = safe_version(__version__)
project_url = "https://github.com/suds-community/suds"
base_download_url = project_url + "/archive"
download_distribution_name = "release-%s.tar.gz" % (version_tag)
download_url = "%s/%s" % (base_download_url, download_distribution_name)
maintainer="Jurko Gospodnetić"
setup(
name=package_name,
version=__version__,
description="Lightweight SOAP client (community fork)",
long_description=long_description,
long_description_content_type='text/markdown',
keywords=["SOAP", "web", "service", "client"],
url=project_url,
download_url=download_url,
packages=recursive_package_list("suds"),
author="Jeff Ortel",
author_email="jortel@redhat.com",
maintainer=maintainer,
maintainer_email="jurko.gospodnetic@pke.hr",
# See PEP-301 for the classifier specification. For a complete list of
# available classifiers see
# 'http://pypi.python.org/pypi?%3Aaction=list_classifiers'.
classifiers=["Development Status :: 5 - Production/Stable",
"Intended Audience :: Developers",
"License :: OSI Approved :: "
"GNU Library or Lesser General Public License (LGPL)",
"Natural Language :: English",
"Operating System :: OS Independent",
"Programming Language :: Python",
"Programming Language :: Python :: 3",
"Programming Language :: Python :: 3 :: Only",
"Programming Language :: Python :: 3.7",
"Programming Language :: Python :: 3.8",
"Programming Language :: Python :: 3.9",
"Programming Language :: Python :: 3.10",
"Topic :: Internet"],
# PEP-314 states that, if possible, license & platform should be specified
# using 'classifiers'.
license="(specified using classifiers)",
platforms=["(specified using classifiers)"],
python_requires=">=3.7",
obsoletes=["suds"] if package_name == forked_package_name else [],
)