-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathfwutil_job_run_local.py
152 lines (122 loc) · 4.63 KB
/
fwutil_job_run_local.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
#! /usr/bin/env python
#
# Given a Flywheel job id, this script will generate a local testing directory
# within which you can run the job locally, using Docker, as it would run on Flywheel.
#
# This code generates a directory structure that mimics exactly what the Gear would
# get when running in Flywheel. Importantly, this code will generate a "config.json"
# file, which contains all of the metadata the Gear received when running in Flywheel.
#
# For a given job this script will:
# 1. Make directory structure (gear_job base dir, inputs, outputs)
# 2. Write a valid config.json file
# 3. Download required data to inputs directories
# 4. Generate a shell script that will actually run the job with docker
# locally (run.sh)
#
# Usage:
# Positional inputs:
# [1] - Api Key
# [2] - Job ID
# [3] - (Optional) Directory to save job contents. Defaults to cwd.
#
# fwutil_job_run_local.py <api_key> <job_id> <output_base_directory>
#
# Example:
# fwutil_job_run_local.py $API_KEY 298e73lacbde98273lkad
#
#
import os
import sys
import json
import stat
import flywheel
def build_local_test(job, test_path_root, api_key):
'''
Build a local testing instance for a given Flywheel job
'''
# Make directories
test_path = os.path.join(
test_path_root, job.gear_info.name + '-' + job.gear_info.version + '_' + job.id)
input_dir = os.path.join(test_path, 'input')
output_dir = os.path.join(test_path, 'output')
if not os.path.isdir(input_dir):
print('Creating directory: %s' % (input_dir))
os.makedirs(input_dir)
if not os.path.isdir(output_dir):
print('Creating directory: %s' % (output_dir))
os.mkdir(output_dir)
# If the job requires an api_key, then add it from the env to the config
gear = fw.get_gear(job.gear_id).gear
if gear.inputs.get('api_key', ""):
job['config']['inputs']['api_key'] = {"key": api_key}
# Write the config file
config_file = os.path.join(test_path, 'config.json')
with open(config_file, 'w') as cf:
json.dump(job['config'], cf, indent=4)
# For each key in input, make the directory and download the data
input_data = job.config.get('inputs')
for k in input_data:
if k == 'api_key':
continue
_input = input_data[k]
# Make the directory
ipath = os.path.join(input_dir, k)
if os.path.exists(ipath):
print('Exists: %s' % ipath)
else:
os.mkdir(ipath)
print('Created directory: %s' % ipath)
# Download the file to the directory
ifilename = _input['location']['name']
ifilepath = os.path.join(ipath, ifilename)
iparentid = _input['hierarchy']['id']
if os.path.isfile(ifilepath):
print('Exists: %s' % ifilename)
else:
print('Downloading: %s' % ifilename)
fw.download_file_from_container(iparentid, ifilename, ifilepath)
print('Done!')
# Generate docker run command and save to script
gear_descriptor = gear['custom'].get('docker-image', '')
if not gear_descriptor:
gear_descriptor = gear['custom'].get('gear-builder').get('image')
run_string = (
'docker run --rm -ti --entrypoint=/bin/bash -v %s:/flywheel/v0/input -v %s:/flywheel/v0/output -v %s:/flywheel/v0/config.json %s'
% (input_dir, output_dir, config_file, gear_descriptor))
run_script = os.path.join(test_path, 'run.sh')
with open(run_script, 'w') as rs:
rs.write('#! /bin/bash \n\n')
rs.write(run_string)
rs.write('\n')
# Make the file executable
st = os.stat(run_script)
os.chmod(run_script, st.st_mode | stat.S_IEXEC)
print(run_string)
return test_path, run_script
if __name__ == '__main__':
"""
Given a Flywheel job id, this script will generate a local testing directory
within which you can run the job locally, using Docker, as it ran in Flywheel.
Positional inputs:
[1] - Api Key
[2] - Job ID
[3] - (Optional) Directory to save job contents. Defaults to cwd.
"""
if not sys.argv[1]:
raise ValueError('API KEY required')
else:
api_key = sys.argv[1]
fw = flywheel.Client(api_key)
if fw.get_current_user().root:
fw = flywheel.Client(api_key, root=True)
else:
raise ValueError('This process requires site-admin privileges!')
# Get the job
job = fw.get_job(sys.argv[2])
# Build the local test
if len(sys.argv) == 4:
test_path_root = sys.argv[3]
else:
test_path_root = os.getcwd()
build_local_test(job, test_path_root, api_key)