Skip to content

Commit 10c86a1

Browse files
committed
Revise tools scripts to be python3 compatible on win32
We introduce setup_stdio function to setup stdout/stderr properly. For python <-> python pipe, we always use 'utf8'/'ignore' encoding for not lost characters. For tty <-> python, we using native encoding with xmlcharrefreplace to encode, to preserve maximal information. For python <-> native program, we use naive encoding with 'ignore' to not cause error update_exclude_list with binary mode so that on win32 would not generate \r\n run-test-suite.py: Handling skiplist properly on win32 Fixes jerryscript-project#4854 Fixes test262-harness.py complain cannot use a string pattern on a bytes-like object with running test262 with python3 For reading/writing to file, we use 'utf8' /'ignore' encoding for not lost characters. For decoding from process stdout, using native encoding with decoding error ignored for not lost data. Execute commands also ignore errors Fixes jerryscript-project#4853 Fixes running test262-esnext failed with installed python3.9 on win32 with space in path Fixes jerryscript-project#4852 ``` support both / \ in --test262-test-list arg On win32. python tools/run-tests.py --test262-es2015=update --test262-test-list=built-ins/decodeURI/ python tools/run-tests.py --test262-es2015=update --test262-test-list=built-ins\decodeURI\ should be both valid, currently only --test262-test-list=built-ins\decodeURI\ are valid. ``` ``` Support snapshot-tests-skiplist.txt on win32 by use os.path.normpath ``` Guard run-tests.py with timer. All run-tests.py are finished in 30 minutes in normal situation. May increase the timeout future. wait JERRY_CHECK_TIMEOUT ``` Move Windows CI to github actions Convert run-debugger-test.sh to run-debugger-test.py After this change, run-debugger-test.py could running on Win32 and OSX Define TERM colors for win32 properly ``` ``` flush stderr.write stdout.write On CI, the stderr are redirect to stdout, and if we don't flush stderr and stdout, The output from stderr/stdout would out of sync. ``` JerryScript-DCO-1.0-Signed-off-by: Yonggang Luo [email protected]
1 parent 8a271bc commit 10c86a1

12 files changed

+208
-137
lines changed

.github/workflows/gh-actions.yml

+32
Original file line numberDiff line numberDiff line change
@@ -56,6 +56,36 @@ jobs:
5656
- run: $RUNNER -q --jerry-tests --buildoptions=--compile-flag=-m32,--cpointer-32bit=on
5757
- run: $RUNNER -q --jerry-tests --buildoptions=--compile-flag=-m32,--cpointer-32bit=on --build-debug
5858

59+
Win_x86-64_Conformance_Tests_ESNext:
60+
runs-on: windows-latest
61+
steps:
62+
- uses: actions/checkout@v2
63+
- run: python $env:RUNNER --test262 update
64+
65+
Win_x86-64_Conformance_Tests_ESNext_Debug:
66+
runs-on: windows-latest
67+
steps:
68+
- uses: actions/checkout@v2
69+
- run: python $env:RUNNER --test262 update --build-debug
70+
71+
Win_x86-64_Tests:
72+
runs-on: windows-latest
73+
steps:
74+
- uses: actions/checkout@v2
75+
- run: python $env:RUNNER -q --jerry-tests
76+
- run: python $env:RUNNER -q --unittests
77+
- run: python $env:RUNNER -q --buildoption-test
78+
- run: python $env:RUNNER -q --jerry-debugger
79+
80+
Win_x86-64_Tests_Debug:
81+
runs-on: windows-latest
82+
steps:
83+
- uses: actions/checkout@v2
84+
- run: python $env:RUNNER -q --jerry-tests --build-debug
85+
- run: python $env:RUNNER -q --unittests --build-debug
86+
- run: python $env:RUNNER -q --buildoption-test --build-debug
87+
- run: python $env:RUNNER -q --jerry-debugger --build-debug
88+
5989
OSX_x86-64_Build_Correctness_Unit_Tests:
6090
runs-on: macos-13
6191
steps:
@@ -65,6 +95,7 @@ jobs:
6595
python-version: '>=3.6'
6696
- run: $RUNNER -q --jerry-tests
6797
- run: $RUNNER -q --unittests
98+
- run: $RUNNER -q --jerry-debugger
6899

69100
OSX_x86-64_Build_Correctness_Unit_Tests_Debug:
70101
runs-on: macos-13
@@ -75,6 +106,7 @@ jobs:
75106
python-version: '>=3.6'
76107
- run: $RUNNER -q --jerry-tests --build-debug
77108
- run: $RUNNER -q --unittests --build-debug
109+
- run: $RUNNER -q --jerry-debugger --build-debug
78110

79111
Linux_x86-64_Build_Option_Tests:
80112
runs-on: ubuntu-latest

appveyor.yml

-27
This file was deleted.

jerry-debugger/jerry_client.py

+2
Original file line numberDiff line numberDiff line change
@@ -326,6 +326,8 @@ def main():
326326
break
327327
if res_type == result.PROMPT:
328328
prompt.cmdloop()
329+
sys.stdout.flush()
330+
sys.stderr.flush()
329331
elif res_type == result.TEXT:
330332
write(result.get_text())
331333
continue

tools/run-tests.py

+24-19
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,9 @@
3131

3232
OUTPUT_DIR = os.path.join(settings.PROJECT_DIR, 'build', 'tests')
3333

34+
# All run_check proc must finished in 30 minutes, may increase in future
35+
JERRY_CHECK_TIMEOUT = 30 * 60
36+
3437
Options = collections.namedtuple('Options', ['name', 'build_args', 'test_args', 'skip'])
3538
Options.__new__.__defaults__ = ([], [], False)
3639

@@ -197,25 +200,22 @@ def get_arguments():
197200

198201
BINARY_CACHE = {}
199202

200-
TERM_NORMAL = '\033[0m'
201-
TERM_YELLOW = '\033[1;33m'
202-
TERM_BLUE = '\033[1;34m'
203-
TERM_RED = '\033[1;31m'
204-
205203
def report_command(cmd_type, cmd, env=None):
206-
sys.stderr.write(f'{TERM_BLUE}{cmd_type}{TERM_NORMAL}\n')
204+
sys.stderr.write(f'{util.TERM_BLUE}{cmd_type}{util.TERM_NORMAL}\n')
207205
if env is not None:
208-
sys.stderr.write(''.join(f'{TERM_BLUE}{var}={val!r} \\{TERM_NORMAL}\n'
206+
sys.stderr.write(''.join(f'{util.TERM_BLUE}{var}={val!r} \\{util.TERM_NORMAL}\n'
209207
for var, val in sorted(env.items())))
210-
sys.stderr.write(f"{TERM_BLUE}" +
211-
f" \\{TERM_NORMAL}\n\t{TERM_BLUE}".join(cmd) +
212-
f"{TERM_NORMAL}\n")
208+
sys.stderr.write(f"{util.TERM_BLUE}" +
209+
f" \\{util.TERM_NORMAL}\n\t{util.TERM_BLUE}".join(cmd) +
210+
f"{util.TERM_NORMAL}\n")
211+
sys.stderr.flush()
213212

214213
def report_skip(job):
215-
sys.stderr.write(f'{TERM_YELLOW}Skipping: {job.name}')
214+
sys.stderr.write(f'{util.TERM_YELLOW}Skipping: {job.name}')
216215
if job.skip:
217216
sys.stderr.write(f' ({job.skip})')
218-
sys.stderr.write(f'{TERM_NORMAL}\n')
217+
sys.stderr.write(f'{util.TERM_NORMAL}\n')
218+
sys.stderr.flush()
219219

220220
def create_binary(job, options):
221221
build_args = job.build_args[:]
@@ -246,13 +246,15 @@ def create_binary(job, options):
246246
if binary_key in BINARY_CACHE:
247247
ret, build_dir_path = BINARY_CACHE[binary_key]
248248
sys.stderr.write(f'(skipping: already built at {build_dir_path} with returncode {ret})\n')
249+
sys.stderr.flush()
249250
return ret, build_dir_path
250251

251252
try:
252253
subprocess.check_output(build_cmd)
253254
ret = 0
254255
except subprocess.CalledProcessError as err:
255-
print(err.output.decode("utf8"))
256+
# For python <-> native program, we use default encoding with error='ignore' to not lost data
257+
print(err.output.decode(errors="ignore"))
256258
ret = err.returncode
257259

258260
BINARY_CACHE[binary_key] = (ret, build_dir_path)
@@ -283,6 +285,7 @@ def iterate_test_runner_jobs(jobs, options):
283285

284286
if build_dir_path in tested_paths:
285287
sys.stderr.write(f'(skipping: already tested with {build_dir_path})\n')
288+
sys.stderr.flush()
286289
continue
287290
tested_paths.add(build_dir_path)
288291

@@ -291,6 +294,7 @@ def iterate_test_runner_jobs(jobs, options):
291294

292295
if bin_hash in tested_hashes:
293296
sys.stderr.write(f'(skipping: already tested with equivalent {tested_hashes[bin_hash]})\n')
297+
sys.stderr.flush()
294298
continue
295299
tested_hashes[bin_hash] = build_dir_path
296300

@@ -308,23 +312,23 @@ def run_check(runnable, env=None):
308312
env = full_env
309313

310314
with subprocess.Popen(runnable, env=env) as proc:
311-
proc.wait()
315+
proc.wait(timeout=JERRY_CHECK_TIMEOUT)
312316
return proc.returncode
313317

314318
def run_jerry_debugger_tests(options):
315319
ret_build = ret_test = 0
316320
for job in DEBUGGER_TEST_OPTIONS:
317321
ret_build, build_dir_path = create_binary(job, options)
318322
if ret_build:
319-
print(f"\n{TERM_RED}Build failed{TERM_NORMAL}\n")
323+
print(f"\n{util.TERM_RED}Build failed{util.TERM_NORMAL}\n")
320324
break
321325

322326
for channel in ["websocket", "rawpacket"]:
323327
for test_file in os.listdir(settings.DEBUGGER_TESTS_DIR):
324328
if test_file.endswith(".cmd"):
325329
test_case, _ = os.path.splitext(test_file)
326330
test_case_path = os.path.join(settings.DEBUGGER_TESTS_DIR, test_case)
327-
test_cmd = [
331+
test_cmd = util.get_python_cmd_prefix() + [
328332
settings.DEBUGGER_TEST_RUNNER_SCRIPT,
329333
get_binary_path(build_dir_path),
330334
channel,
@@ -379,7 +383,7 @@ def run_test262_test_suite(options):
379383
for job in jobs:
380384
ret_build, build_dir_path = create_binary(job, options)
381385
if ret_build:
382-
print(f"\n{TERM_RED}Build failed{TERM_NORMAL}\n")
386+
print(f"\n{util.TERM_RED}Build failed{util.TERM_NORMAL}\n")
383387
break
384388

385389
test_cmd = util.get_python_cmd_prefix() + [
@@ -409,7 +413,7 @@ def run_unittests(options):
409413
continue
410414
ret_build, build_dir_path = create_binary(job, options)
411415
if ret_build:
412-
print(f"\n{TERM_RED}Build failed{TERM_NORMAL}\n")
416+
print(f"\n{util.TERM_RED}Build failed{util.TERM_NORMAL}\n")
413417
break
414418

415419
if sys.platform == 'win32':
@@ -438,14 +442,15 @@ def run_buildoption_test(options):
438442

439443
ret, _ = create_binary(job, options)
440444
if ret:
441-
print(f"\n{TERM_RED}Build failed{TERM_NORMAL}\n")
445+
print(f"\n{util.TERM_RED}Build failed{util.TERM_NORMAL}\n")
442446
break
443447

444448
return ret
445449

446450
Check = collections.namedtuple('Check', ['enabled', 'runner', 'arg'])
447451

448452
def main(options):
453+
util.setup_stdio()
449454
checks = [
450455
Check(options.check_signed_off, run_check, [settings.SIGNED_OFF_SCRIPT]
451456
+ {'tolerant': ['--tolerant'], 'gh-actions': ['--gh-actions']}.get(options.check_signed_off, [])),

tools/runners/run-debugger-test.py

+99
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,99 @@
1+
#!/usr/bin/env python
2+
3+
# Copyright JS Foundation and other contributors, http://js.foundation
4+
#
5+
# Licensed under the Apache License, Version 2.0 (the "License");
6+
# you may not use this file except in compliance with the License.
7+
# You may obtain a copy of the License at
8+
#
9+
# http://www.apache.org/licenses/LICENSE-2.0
10+
#
11+
# Unless required by applicable law or agreed to in writing, software
12+
# distributed under the License is distributed on an "AS IS" BASIS
13+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14+
# See the License for the specific language governing permissions and
15+
# limitations under the License.
16+
17+
import os
18+
import subprocess
19+
import sys
20+
import time
21+
22+
import util
23+
TempFile = __import__("test262-harness").TempFile # pylint: disable=invalid-name
24+
25+
class DebuggerArgs:
26+
def __init__(self):
27+
self.jerry = sys.argv[1]
28+
self.channel = sys.argv[2]
29+
self.debugger_client = sys.argv[3]
30+
self.test_case = sys.argv[4]
31+
32+
33+
def check_output(command_args, stdin=None, encoding=None):
34+
try:
35+
out = subprocess.check_output(command_args, stdin=stdin, shell=False, stderr=subprocess.STDOUT)
36+
except subprocess.CalledProcessError as check_error:
37+
out = check_error.output
38+
return out.decode(encoding or 'utf-8', 'ignore')
39+
40+
41+
def execute_debug_client(out_tmp, cmd_file_name, debug_client_args):
42+
print(f'input debug cmd: {cmd_file_name}')
43+
with open(cmd_file_name, 'rb') as cmd_file:
44+
out = check_output(debug_client_args, cmd_file)
45+
out_tmp.write(out)
46+
47+
48+
def main(args):
49+
util.setup_stdio()
50+
jerry_debug_server_cmd = [args.jerry]
51+
client_args = []
52+
if 'client_source' in args.test_case:
53+
jerry_debug_server_cmd += ['--start-debug-server', '--debug-channel',
54+
args.channel, '--debugger-wait-source']
55+
client_args += ['--client-source']
56+
if 'client_source_multiple' in args.test_case:
57+
client_args += [args.test_case + '_2.js', args.test_case + '_1.js']
58+
else:
59+
client_args += [args.test_case + '.js']
60+
else:
61+
jerry_debug_server_cmd += [args.test_case + '.js', '--start-debug-server', '--debug-channel', args.channel]
62+
print(f'run debug server: {jerry_debug_server_cmd}')
63+
with subprocess.Popen(jerry_debug_server_cmd, stdin=subprocess.PIPE,
64+
stdout=subprocess.PIPE, stderr=subprocess.STDOUT) as proc:
65+
time.sleep(1)
66+
67+
out_tmp = TempFile(prefix=os.path.basename(args.test_case), suffix='out')
68+
git_failed = False
69+
try:
70+
debug_client_args = util.get_python_cmd_prefix()
71+
debug_client_args += [args.debugger_client, '--channel', args.channel, '--non-interactive']
72+
debug_client_args += client_args
73+
print(f"run debug client: {' '.join(debug_client_args)}")
74+
execute_debug_client(out_tmp, args.test_case + '.cmd', debug_client_args)
75+
if 'restart' in args.test_case:
76+
continue_case = args.test_case.replace('restart', 'continue')
77+
execute_debug_client(out_tmp, continue_case + '.cmd', debug_client_args)
78+
out_tmp.close()
79+
git_diff_cmd = ['git', '--no-pager', 'diff', '--ignore-space-at-eol',
80+
'--no-index', args.test_case + '.expected', out_tmp.name]
81+
git_out = check_output(git_diff_cmd)
82+
if '@@' in git_out:
83+
git_failed = True
84+
finally:
85+
proc.wait()
86+
print(f"jerry out:\n{proc.stdout.read().decode('utf-8')}\nEOF")
87+
print(f"git diff cmd: {' '.join(git_diff_cmd)}")
88+
if git_failed:
89+
print(f'git diff result:\n{git_out}\nEOF')
90+
print(f'{util.TERM_RED}FAIL: {args.test_case}{util.TERM_NORMAL}')
91+
sys.exit(1)
92+
else:
93+
out_tmp.dispose()
94+
print(f'{util.TERM_GREEN}PASS: {args.test_case}{util.TERM_NORMAL}')
95+
sys.exit(0)
96+
97+
98+
if __name__ == "__main__":
99+
sys.exit(main(DebuggerArgs()))

tools/runners/run-debugger-test.sh

-66
This file was deleted.

0 commit comments

Comments
 (0)