-
Notifications
You must be signed in to change notification settings - Fork 2
Expand file tree
/
Copy pathsetup_eb.py
More file actions
151 lines (118 loc) · 4.89 KB
/
setup_eb.py
File metadata and controls
151 lines (118 loc) · 4.89 KB
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
import os
import re
import toml.decoder
from setuptools import setup, find_packages
# from packaging.version import parse as version_parse
from sys import version_info as python_version_info
from semantic_version import SimpleSpec, Version
ROOT_DIR = os.path.abspath(os.path.dirname(__file__))
PYPROJECT_TOML = toml.decoder.load(os.path.join(ROOT_DIR, 'pyproject.toml'))
POETRY_DATA = PYPROJECT_TOML['tool']['poetry']
_CARET_MATCH = re.compile(r"[\^]([0-9]+)([.].*)?$")
_TILDE_MATCH = re.compile(r"[~]([0-9]+[.])([0-9]+)([.].*)?$")
def fix_requirement(requirement):
if isinstance(requirement, str):
return _fix_requirement_string(requirement)
elif isinstance(requirement, list):
return fix_requirement(_select_requirement(requirement))
elif isinstance(requirement, dict):
return _fix_requirement_dict(requirement)
else:
raise ValueError(f"Unrecognized requirement: {requirement!r}")
def _select_requirement(requirement):
if not isinstance(requirement, list):
raise ValueError(f"{requirement!r} is not a list.")
python_version = Version(f"{python_version_info.major}.{python_version_info.minor}.{python_version_info.micro}")
for clause in requirement:
if "python" not in clause:
raise ValueError(f"Missing 'python' in clause: {clause!r}")
if SimpleSpec(clause['python']).match(python_version):
return clause
raise ValueError(f"No clauses matched: {requirement!r}")
def _fix_requirement_string(requirement):
m = _CARET_MATCH.match(requirement)
if m:
return ">=%s%s,<%s" % (m.group(1), m.group(2), int(m.group(1)) + 1)
m = _TILDE_MATCH.match(requirement)
if m:
return ">=%s%s%s,<%s%s" % (m.group(1), m.group(2), m.group(3), m.group(1), int(m.group(2)) + 1)
if requirement[0].isdigit():
return "==" + requirement
else:
return requirement
def _fix_requirement_dict(requirement):
extras = requirement.get("extras", [])
extras_str = f"[{','.join(extras)}]" if extras else ""
if "version" in requirement:
version = _fix_requirement_string(requirement["version"])
return f"{extras_str}{version}" if extras_str else version
# Handle VCS (Git) style requirements
vcs_keys = {"git", "hg", "svn", "bzr"}
vcs_type = next((key for key in vcs_keys if key in requirement), None)
if vcs_type:
vcs_url = requirement[vcs_type]
ref = ""
if "branch" in requirement:
ref = f"@{requirement['branch']}"
elif "rev" in requirement:
ref = f"@{requirement['rev']}"
elif "tag" in requirement:
ref = f"@{requirement['tag']}"
return f"{extras_str} {vcs_type}+{vcs_url}{ref}".strip()
raise ValueError(f"Unsupported requirement format: {requirement!r}")
_EMAIL_MATCH = re.compile(r"^([^<]*)[<]([^>]*)[>]$")
def author_and_email(authorship_spec):
m = _EMAIL_MATCH.match(authorship_spec)
if m:
return m.group(1), m.group(2)
else:
raise ValueError("Expect authorship in format 'human_name <email_name@email_host>': %s" % authorship_spec)
def get_requirements(kind='dependencies'):
return [
f"{pkg} @ {fix_requirement(requirement)}" if isinstance(requirement, dict) and any(vcs in requirement for vcs in {"git", "hg", "svn", "bzr"})
else pkg + fix_requirement(requirement)
for pkg, requirement in POETRY_DATA[kind].items()
if pkg != "python"
]
def flatten_config_data(key, dictionary):
return "%s\n%s\n\n" % (key, "\n".join([
key + " = " + val
for key, val in dictionary.items()
]))
def entry_points():
result = flatten_config_data("[console_scripts]", POETRY_DATA['scripts'])
paste_dict = PYPROJECT_TOML['paste']
for subkey in paste_dict:
result += flatten_config_data('[paste.%s]' % subkey, paste_dict[subkey])
return result
ENTRY_POINTS = entry_points()
PACKAGE_NAME = POETRY_DATA['name']
README = open(os.path.join(ROOT_DIR, 'README.rst')).read()
# CHANGES = open(os.path.join(ROOT_DIR, 'CHANGES.rst')).read())
DESCRIPTION = POETRY_DATA['description']
LONG_DESCRIPTION = README
AUTHOR, AUTHOR_EMAIL = author_and_email(POETRY_DATA['authors'][0])
URL = 'http://data.4dnucleome.org'
LICENSE = 'MIT'
INSTALL_REQUIRES = get_requirements()
TESTS_REQUIRE = get_requirements('dev-dependencies')
VERSION = POETRY_DATA['version']
if __name__ == '__main__':
setup(
name=PACKAGE_NAME,
version=VERSION,
description=DESCRIPTION,
long_description=LONG_DESCRIPTION,
packages=find_packages('src'),
package_dir={'': 'src'},
include_package_data=True,
zip_safe=False,
author=AUTHOR,
author_email=AUTHOR_EMAIL,
url=URL,
license=LICENSE,
install_requires=INSTALL_REQUIRES,
tests_require=TESTS_REQUIRE,
extras_require={'test': TESTS_REQUIRE},
entry_points=ENTRY_POINTS,
)