-
Notifications
You must be signed in to change notification settings - Fork 87
/
setup.py
167 lines (146 loc) · 6.54 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
import glob
import os
import platform
import subprocess
import sys
from pathlib import Path
from setuptools import setup
from setuptools.command.build import build
from setuptools.command.build_py import build_py
from shutil import copyfile
include_java_stubs = os.environ.get('INCLUDE_JAVA_STUBS', 'false').lower() == 'true'
"""
If Java type stubs should be generated.
These are for our own type checker; users should never need
to use them (since they should be using the Python classes).
Not included in the default build since IDE's will see multiple
packages that define a class (the one from `ai.timefold...`
and the one from `timefold.solver...`), which can lead to the
wrong one being imported.
"""
class ChangeBuildDirectory(build):
def initialize_options(self):
super().initialize_options()
self.build_base = '.build'
class FetchDependencies(build_py):
"""
A command class that fetch Java Dependencies and
add them as files within a python package
"""
def create_stubs(self, project_root, command):
working_directory = project_root / 'python' / 'python-core'
subprocess.run([str((project_root / command).absolute()), 'dependency:copy-dependencies',
'-Dpython', '-Dexec.skip'],
cwd=working_directory, check=True)
subprocess.run([str((project_root / command).absolute()), 'dependency:copy-dependencies',
'-Dclassifier=javadoc', '-Dpython', '-Dexec.skip'], cwd=working_directory, check=True)
subprocess.run([sys.executable, str((project_root / 'create-stubs.py').absolute())],
cwd=working_directory, check=True)
target_dir = self.build_lib
for file_name in find_stub_files(str(working_directory / 'java-stubs')):
path = file_name[len(str(working_directory)) + 1:]
os.makedirs(os.path.dirname(os.path.join(target_dir, path)), exist_ok=True)
copyfile(os.path.join(str(working_directory), path), os.path.join(target_dir, path))
for file_name in find_stub_files(str(working_directory / 'jpype-stubs')):
path = file_name[len(str(working_directory)) + 1:]
os.makedirs(os.path.dirname(os.path.join(target_dir, path)), exist_ok=True)
copyfile(os.path.join(str(working_directory), path), os.path.join(target_dir, path))
for file_name in find_stub_files(str(working_directory / 'ai-stubs')):
path = file_name[len(str(working_directory)) + 1:]
os.makedirs(os.path.dirname(os.path.join(target_dir, path)), exist_ok=True)
copyfile(os.path.join(str(working_directory), path), os.path.join(target_dir, path))
def run(self):
if not self.dry_run:
project_root = Path(__file__).parent
# Do a mvn clean install
# which is configured to add dependency jars to 'target/dependency'
command = 'mvnw'
if platform.system() == 'Windows':
command = 'mvnw.cmd'
subprocess.run([str((project_root / command).absolute()), 'clean', 'install', '-Dpython', '-Dquickly',
'-Dexec.skip'],
cwd=project_root, check=True)
if include_java_stubs:
self.create_stubs(project_root, command)
classpath_jars = []
# Add the main artifact
classpath_jars.extend(glob.glob(os.path.join(project_root, 'python', 'python-core', 'target', '*.jar')))
# Add the main artifact's dependencies
classpath_jars.extend(glob.glob(os.path.join(project_root, 'python', 'python-core',
'target', 'dependency', '*.jar')))
self.mkpath(os.path.join(self.build_lib, 'timefold', 'solver', 'jars'))
# Copy classpath jars to timefold.solver.jars
for file in classpath_jars:
copyfile(file, os.path.join(self.build_lib, 'timefold', 'solver', 'jars', os.path.basename(file)))
# Make timefold.solver.jars a Python module
fp = open(os.path.join(self.build_lib, 'timefold', 'solver', 'jars', '__init__.py'), 'w')
fp.close()
build_py.run(self)
def find_stub_files(stub_root: str):
for root, dirs, files in os.walk(stub_root):
for file in files:
if file.endswith('.pyi'):
yield os.path.join(root, file)
packages = [
'timefold.solver',
'timefold.solver.config',
'timefold.solver.domain',
'timefold.solver.heuristic',
'timefold.solver.score',
'timefold.solver.test',
'_jpyinterpreter',
]
package_dir = {
'timefold.solver': 'python/python-core/src/main/python',
'_jpyinterpreter': 'python/jpyinterpreter/src/main/python',
}
if include_java_stubs:
packages += [
'java-stubs',
'jpype-stubs',
'ai-stubs',
]
package_dir.update({
'java-stubs': 'python/python-core/src/main/resources',
'jpype-stubs': 'python/python-core/src/main/resources',
'ai-stubs': 'python/python-core/src/main/resources',
})
this_directory = Path(__file__).parent
long_description = (this_directory / 'python' / 'README.md').read_text()
timefold_solver_python_version = '999-dev0'
setup(
name='timefold',
version=timefold_solver_python_version,
license='Apache License Version 2.0',
license_file='LICENSE.txt',
description='An AI constraint solver that optimizes planning and scheduling problems',
long_description=long_description,
long_description_content_type='text/markdown',
url='https://github.com/TimefoldAI/timefold-solver',
project_urls={
'Timefold Solver Documentation': 'https://timefold.ai/docs/timefold-solver/latest',
'Timefold Homepage': 'https://timefold.ai',
},
classifiers=[
'Development Status :: 4 - Beta',
'Programming Language :: Python :: 3',
'Topic :: Scientific/Engineering :: Artificial Intelligence',
'Topic :: Software Development :: Libraries :: Java Libraries',
'License :: OSI Approved :: Apache Software License',
'Operating System :: OS Independent'
],
packages=packages,
package_dir=package_dir,
test_suite='python/tests',
python_requires='>=3.10',
install_requires=[
'JPype1>=1.5.0'
],
cmdclass={
'build': ChangeBuildDirectory,
'build_py': FetchDependencies
},
package_data={
'timefold.solver.jars': ['*.jar'],
},
)