forked from rotki/rotki
-
Notifications
You must be signed in to change notification settings - Fork 0
/
rotkehlchen.spec
160 lines (141 loc) · 5.35 KB
/
rotkehlchen.spec
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
# -*- mode: python -*-
from __future__ import print_function # isort:skip
import os
import platform
import sys
from distutils.spawn import find_executable
from PyInstaller.utils.hooks import collect_submodules
from rotkehlchen.exchanges.constants import SUPPORTED_EXCHANGES
from rotkehlchen.types import Location
from rotkehlchen.utils.misc import get_system_spec
"""
PyInstaller spec file to build single file or dir distributions
Currently only tested on macOS
"""
# Set to false to produce an exploded single-dir
ONEFILE = int(os.environ.get('ONEFILE', True))
MACOS_BUILD_ARCH = os.environ.get('MACOS_BUILD_ARCH', None)
def Entrypoint(dist, group, name, scripts=None, pathex=None, hiddenimports=None, hookspath=None,
excludes=None, runtime_hooks=None, datas=None):
import pkg_resources
# get toplevel packages of distribution from metadata
def get_toplevel(dist):
distribution = pkg_resources.get_distribution(dist)
if distribution.has_metadata('top_level.txt'):
return list(distribution.get_metadata('top_level.txt').split())
else:
return []
hiddenimports = hiddenimports or []
packages = []
for distribution in hiddenimports:
try:
packages += get_toplevel(distribution)
except:
pass
scripts = scripts or []
pathex = pathex or []
# get the entry point
ep = pkg_resources.get_entry_info(dist, group, name)
# insert path of the egg at the verify front of the search path
pathex = [ep.dist.location] + pathex
# script name must not be a valid module name to avoid name clashes on import
script_path = os.path.join(workpath, name + '-script.py')
print("creating script for entry point", dist, group, name)
with open(script_path, 'w') as fh:
print("import", ep.module_name, file=fh)
print("%s.%s()" % (ep.module_name, '.'.join(ep.attrs)), file=fh)
for package in packages:
print("import", package, file=fh)
return Analysis(
[script_path] + scripts,
pathex=pathex,
hiddenimports=hiddenimports,
hookspath=hookspath,
excludes=excludes,
runtime_hooks=runtime_hooks,
datas=datas,
)
# We don't need Tk and friends
sys.modules['FixTk'] = None
executable_name = 'rotki-core-{}-{}'.format(
get_system_spec()['rotkehlchen'],
'macos' if platform.system() == 'Darwin' else platform.system().lower())
hiddenimports = ['cytoolz.utils', 'cytoolz._signatures']
# Since the exchanges are loaded dynamically and some of them may not be detected
# by pyinstaller (https://github.com/rotki/rotki/issues/602) make sure they are
# all included as imports in the created executable
for exchange_name in SUPPORTED_EXCHANGES:
if exchange_name == Location.BINANCEUS:
continue
hiddenimports.append(f'rotkehlchen.exchanges.{exchange_name}')
# TODO: Make this dynamic you dummy
ethereum_modules = collect_submodules('rotkehlchen.chain.ethereum.modules')
optimism_modules = collect_submodules('rotkehlchen.chain.optimism.modules')
polygon_pos_modules = collect_submodules('rotkehlchen.chain.polygon_pos.modules')
dynamic_modules = ethereum_modules + optimism_modules + polygon_pos_modules
hiddenimports.extend(dynamic_modules)
a = Entrypoint(
'rotkehlchen',
'console_scripts',
'rotkehlchen',
hookspath=['tools/pyinstaller_hooks'],
hiddenimports=hiddenimports,
datas=[
# This list should be kept in sync with setup.py (package_data)
('rotkehlchen/data/eth_abi.json', 'rotkehlchen/data'),
('rotkehlchen/data/eth_contracts.json', 'rotkehlchen/data'),
('rotkehlchen/data/uniswapv2_lp_tokens.json', 'rotkehlchen/data'),
('rotkehlchen/data/uniswapv2_lp_tokens.meta', 'rotkehlchen/data'),
('rotkehlchen/data/global.db', 'rotkehlchen/data'),
('rotkehlchen/data/globaldb_v2_v3_assets.sql', 'rotkehlchen/data'),
('rotkehlchen/data/nodes.json', 'rotkehlchen/data'),
('rotkehlchen/data/nodes_as_of_1-26-1.json', 'rotkehlchen/data'),
('rotkehlchen/data/populate_asset_collections.sql', 'rotkehlchen/data'),
('rotkehlchen/data/populate_multiasset_mappings.sql', 'rotkehlchen/data'),
# TODO
# We probably should have a better way to specify some data should be loaded
# by a module in pyinstaller. Should be loaded dynamically by rotki and not
# by pyinstaller if we want it to be truly modular
(
'rotkehlchen/chain/ethereum/modules/dxdaomesa/data/contracts.json',
'rotkehlchen/chain/ethereum/modules/dxdaomesa/data',
),
],
excludes=['FixTk', 'tcl', 'tk', '_tkinter', 'tkinter', 'Tkinter', 'debugimporter'],
)
pyz = PYZ(a.pure, a.zipped_data, cipher=None)
if ONEFILE:
exe = EXE(
pyz,
a.scripts,
a.binaries,
a.zipfiles,
a.datas,
name=executable_name,
debug=False,
strip=False,
upx=False,
runtime_tmpdir=None,
console=True,
)
else:
exe = EXE(
pyz,
a.scripts,
exclude_binaries=True,
name=executable_name,
debug=False,
strip=False,
upx=False,
console=True,
target_arch=MACOS_BUILD_ARCH
)
coll = COLLECT(
exe,
a.binaries,
a.zipfiles,
a.datas,
strip=False,
upx=False,
name='rotki-core',
)