comparison toolfactory/ToolFactory_tester.py @ 4:2a46da701dde draft

Uploaded
author fubar
date Mon, 26 Apr 2021 05:25:26 +0000
parents
children
comparison
equal deleted inserted replaced
3:c4f192ec521c 4:2a46da701dde
1 # see https://github.com/fubar2/toolfactory
2 #
3 # copyright ross lazarus (ross stop lazarus at gmail stop com) May 2012
4 #
5 # all rights reserved
6 # Licensed under the LGPL
7 # suggestions for improvement and bug fixes welcome at
8 # https://github.com/fubar2/toolfactory
9 #
10 # July 2020: BCC was fun and I feel like rip van winkle after 5 years.
11 # Decided to
12 # 1. Fix the toolfactory so it works - done for simplest case
13 # 2. Fix planemo so the toolfactory function works
14 # 3. Rewrite bits using galaxyxml functions where that makes sense - done
15
16 import argparse
17 import copy
18 import os
19 import subprocess
20 import shutil
21 import sys
22 import tarfile
23 import tempfile
24 import time
25
26 myversion = "V2.2 April 2021"
27 verbose = True
28 debug = True
29 toolFactoryURL = "https://github.com/fubar2/toolfactory"
30
31 def timenow():
32 """return current time as a string"""
33 return time.strftime("%d/%m/%Y %H:%M:%S", time.localtime(time.time()))
34
35 class ToolTester():
36 # requires highly insecure docker settings - like write to tool_conf.xml and to tools !
37 # if in a container possibly not so courageous.
38 # Fine on your own laptop but security red flag for most production instances
39 # uncompress passed tar, run planemo and rebuild a new tarball with tests
40
41 def __init__(self, args=None, in_tool_archive='/galaxy-central/tools/newtool/newtool_toolshed.gz', new_tool_archive=None):
42 self.args = args
43 self.new_tool_archive = new_tool_archive
44 assert tarfile.is_tarfile(in_tool_archive)
45 # this is not going to go well with arbitrary names. TODO introspect tool xml!
46 self.tooloutdir = "./tfout"
47 self.repdir = "./TF_run_report"
48 self.testdir = os.path.join(self.tooloutdir, "test-data")
49 if not os.path.exists(self.tooloutdir):
50 os.mkdir(self.tooloutdir)
51 if not os.path.exists(self.testdir):
52 os.mkdir(self.testdir)
53 if not os.path.exists(self.repdir):
54 os.mkdir(self.repdir)
55 tff = tarfile.open(in_tool_archive, "r:*")
56 flist = tff.getnames()
57 ourdir = os.path.commonpath(flist) # eg pyrevpos
58 self.tool_name = ourdir
59 ourxmls = [x for x in flist if x.lower().endswith('.xml') and os.path.split(x)[0] == ourdir]
60 assert len(ourxmls) > 0
61 self.ourxmls = ourxmls # [os.path.join(tool_path,x) for x in ourxmls]
62 res = tff.extractall()
63 tff.close()
64 self.update_tests(ourdir)
65 self.makeTool()
66 self.moveRunOutputs()
67 self.makeToolTar()
68
69 def call_planemo(self,xmlpath,ourdir):
70 penv = os.environ
71 penv['HOME'] = '/home/ross/galaxy-release_21.01'
72 toolfile = os.path.split(xmlpath)[1]
73 tool_name = self.tool_name
74 tool_test_output = f"{tool_name}_planemo_test_report.html"
75 cll = [
76 "planemo",
77 "test",
78 "--test_output",
79 os.path.abspath(tool_test_output),
80 "--galaxy_root",
81 self.args.galaxy_root,
82 "--update_test_data",
83 os.path.abspath(xmlpath),
84 ]
85 print(cll)
86 p = subprocess.run(
87 cll,
88 capture_output=True,
89 encoding='utf8',
90 env = penv,
91 shell=False,
92 )
93 return p
94
95 def makeTool(self):
96 """write xmls and input samples into place"""
97 for xreal in self.ourxmls:
98 x = os.path.split(xreal)[1]
99 xout = os.path.join(self.tooloutdir,x)
100 shutil.copyfile(xreal, xout)
101 # for p in self.infiles:
102 # pth = p["name"]
103 # dest = os.path.join(self.testdir, "%s_sample" % p["infilename"])
104 # shutil.copyfile(pth, dest)
105 # dest = os.path.join(self.repdir, "%s_sample" % p["infilename"])
106 # shutil.copyfile(pth, dest)
107
108 def makeToolTar(self):
109 """move outputs into test-data and prepare the tarball"""
110 excludeme = "_planemo_test_report.html"
111
112 def exclude_function(tarinfo):
113 filename = tarinfo.name
114 return None if filename.endswith(excludeme) else tarinfo
115
116 newtar = 'new_%s_toolshed.gz' % self.tool_name
117 ttf = tarfile.open(newtar, "w:gz")
118 ttf.add(name=self.tooloutdir,
119 arcname=self.tool_name,
120 filter=exclude_function)
121 ttf.close()
122 shutil.copyfile(newtar, self.new_tool_archive)
123
124 def moveRunOutputs(self):
125 """need to move planemo or run outputs into toolfactory collection"""
126 with os.scandir(self.tooloutdir) as outs:
127 for entry in outs:
128 if not entry.is_file():
129 continue
130 if "." in entry.name:
131 _, ext = os.path.splitext(entry.name)
132 if ext in [".tgz", ".json"]:
133 continue
134 if ext in [".yml", ".xml", ".yaml"]:
135 newname = f"{entry.name.replace('.','_')}.txt"
136 else:
137 newname = entry.name
138 else:
139 newname = f"{entry.name}.txt"
140 dest = os.path.join(self.repdir, newname)
141 src = os.path.join(self.tooloutdir, entry.name)
142 shutil.copyfile(src, dest)
143 with os.scandir('.') as outs:
144 for entry in outs:
145 if not entry.is_file():
146 continue
147 if "." in entry.name:
148 _, ext = os.path.splitext(entry.name)
149 if ext in [".yml", ".xml", ".yaml"]:
150 newname = f"{entry.name.replace('.','_')}.txt"
151 else:
152 newname = entry.name
153 else:
154 newname = f"{entry.name}.txt"
155 dest = os.path.join(self.repdir, newname)
156 src =entry.name
157 shutil.copyfile(src, dest)
158 if True or self.args.include_tests:
159 with os.scandir(self.testdir) as outs:
160 for entry in outs:
161 if (not entry.is_file()) or entry.name.endswith(
162 "_planemo_test_report.html"
163 ):
164 continue
165 if "." in entry.name:
166 _, ext = os.path.splitext(entry.name)
167 if ext in [".tgz", ".json"]:
168 continue
169 if ext in [".yml", ".xml", ".yaml"]:
170 newname = f"{entry.name.replace('.','_')}.txt"
171 else:
172 newname = entry.name
173 else:
174 newname = f"{entry.name}.txt"
175 dest = os.path.join(self.repdir, newname)
176 src = os.path.join(self.testdir, entry.name)
177 shutil.copyfile(src, dest)
178
179
180 def update_tests(self,ourdir):
181 for xmlf in self.ourxmls:
182 capture = self.call_planemo(xmlf,ourdir)
183 #sys.stderr.write('%s, stdout=%s, stderr=%s' % (xmlf, capture.stdout, capture.stdout))
184 print('%s, stdout=%s, stderr=%s' % (capture.stdout, capture.stdout,xmlf))
185
186 def main():
187 """
188 This is a Galaxy wrapper.
189 It expects to be called by a special purpose tool.xml
190
191 """
192 parser = argparse.ArgumentParser()
193 a = parser.add_argument
194 a("--in_tool_archive", default=None)
195 a("--new_tested_tool_archive", default=None)
196 a("--galaxy_root", default="/home/ross/galaxy-release_21.01/")
197 args = parser.parse_args()
198 print('Hello from',os.getcwd())
199 tt = ToolTester(args=args, in_tool_archive=args.in_tool_archive, new_tool_archive=args.new_tested_tool_archive)
200
201 if __name__ == "__main__":
202 main()