OLD | NEW |
---|---|
1 #!/usr/bin/env python | 1 #!/usr/bin/env python |
2 # Copyright (c) 2017 The WebRTC project authors. All Rights Reserved. | 2 # Copyright (c) 2017 The WebRTC project authors. All Rights Reserved. |
3 # | 3 # |
4 # Use of this source code is governed by a BSD-style license | 4 # Use of this source code is governed by a BSD-style license |
5 # that can be found in the LICENSE file in the root of the source | 5 # that can be found in the LICENSE file in the root of the source |
6 # tree. An additional intellectual property rights grant can be found | 6 # tree. An additional intellectual property rights grant can be found |
7 # in the file PATENTS. All contributing project authors may | 7 # in the file PATENTS. All contributing project authors may |
8 # be found in the AUTHORS file in the root of the source tree. | 8 # be found in the AUTHORS file in the root of the source tree. |
9 | 9 |
10 """ | 10 """ |
11 This script is the wrapper that runs the low-bandwidth audio test. | 11 This script is the wrapper that runs the low-bandwidth audio test. |
12 | 12 |
13 After running the test, post-process steps for calculating audio quality of the | 13 After running the test, post-process steps for calculating audio quality of the |
14 output files will be performed. | 14 output files will be performed. |
15 """ | 15 """ |
16 | 16 |
17 import argparse | 17 import argparse |
18 import logging | 18 import logging |
19 import os | 19 import os |
20 import re | 20 import re |
21 import shutil | |
21 import subprocess | 22 import subprocess |
22 import sys | 23 import sys |
23 | 24 |
24 | 25 |
25 SCRIPT_DIR = os.path.dirname(os.path.abspath(__file__)) | 26 SCRIPT_DIR = os.path.dirname(os.path.abspath(__file__)) |
26 SRC_DIR = os.path.normpath(os.path.join(SCRIPT_DIR, os.pardir, os.pardir, | 27 SRC_DIR = os.path.normpath(os.path.join(SCRIPT_DIR, os.pardir, os.pardir, |
27 os.pardir)) | 28 os.pardir)) |
28 | 29 |
29 | 30 |
30 def _LogCommand(command): | 31 def _LogCommand(command): |
31 logging.info('Running %r', command) | 32 logging.info('Running %r', command) |
32 return command | 33 return command |
33 | 34 |
34 | 35 |
35 def _ParseArgs(): | 36 def _ParseArgs(): |
36 parser = argparse.ArgumentParser(description='Run low-bandwidth audio tests.') | 37 parser = argparse.ArgumentParser(description='Run low-bandwidth audio tests.') |
37 parser.add_argument('build_dir', | 38 parser.add_argument('build_dir', |
38 help='Path to the build directory (e.g. out/Release).') | 39 help='Path to the build directory (e.g. out/Release).') |
39 parser.add_argument('--remove', action='store_true', | 40 parser.add_argument('--remove', action='store_true', |
40 help='Remove output audio files after testing.') | 41 help='Remove output audio files after testing.') |
42 parser.add_argument('--android', action='store_true', | |
43 help='Perform the test on a connected Android device instead.') | |
44 parser.add_argument('--adb-path', help='Path to adb binary.', default='adb') | |
41 args = parser.parse_args() | 45 args = parser.parse_args() |
42 return args | 46 return args |
43 | 47 |
44 | 48 |
45 def _GetPlatform(): | 49 def _GetPlatform(): |
46 if sys.platform == 'win32': | 50 if sys.platform == 'win32': |
47 return 'win' | 51 return 'win' |
48 elif sys.platform == 'darwin': | 52 elif sys.platform == 'darwin': |
49 return 'mac' | 53 return 'mac' |
50 elif sys.platform.startswith('linux'): | 54 elif sys.platform.startswith('linux'): |
51 return 'linux' | 55 return 'linux' |
52 | 56 |
53 | 57 |
54 def _GetExecutableExtension(): | |
55 if sys.platform == 'win32': | |
56 return '.exe' | |
57 else: | |
58 return '' | |
59 | |
60 | |
61 def _DownloadTools(): | 58 def _DownloadTools(): |
62 tools_dir = os.path.join(SRC_DIR, 'tools-webrtc') | 59 tools_dir = os.path.join(SRC_DIR, 'tools-webrtc') |
63 toolchain_dir = os.path.join(tools_dir, 'audio_quality') | 60 toolchain_dir = os.path.join(tools_dir, 'audio_quality') |
64 | 61 |
65 # Download pesq. | 62 # Download pesq. |
66 download_script = os.path.join(tools_dir, 'download_tools.py') | 63 download_script = os.path.join(tools_dir, 'download_tools.py') |
67 command = [sys.executable, download_script, toolchain_dir] | 64 command = [sys.executable, download_script, toolchain_dir] |
68 subprocess.check_call(_LogCommand(command)) | 65 subprocess.check_call(_LogCommand(command)) |
69 | 66 |
70 pesq_path = os.path.join(toolchain_dir, _GetPlatform(), | 67 pesq_path = os.path.join(toolchain_dir, _GetPlatform(), 'pesq') |
71 'pesq' + _GetExecutableExtension()) | |
72 return pesq_path | 68 return pesq_path |
73 | 69 |
74 | 70 |
71 def _GetFile(file_path, out_dir, android=False, adb_path='adb'): | |
kjellander_webrtc
2017/03/28 07:03:30
Let's have the default argument be None instead, s
| |
72 out_file_name = os.path.basename(file_path) | |
73 out_file_path = os.path.join(out_dir, out_file_name) | |
74 | |
75 if android: | |
76 # Pull the file from the connected Android device | |
77 adb_command = [adb_path, 'pull', file_path, out_dir] | |
78 subprocess.check_call(_LogCommand(adb_command)) | |
79 elif os.path.abspath(file_path) != os.path.abspath(out_file_path): | |
80 shutil.copy(file_path, out_file_path) | |
81 | |
82 return out_file_path | |
83 | |
84 | |
75 def main(): | 85 def main(): |
76 # pylint: disable=W0101 | 86 # pylint: disable=W0101 |
77 logging.basicConfig(level=logging.INFO) | 87 logging.basicConfig(level=logging.INFO) |
78 | 88 |
79 args = _ParseArgs() | 89 args = _ParseArgs() |
80 | 90 |
81 pesq_path = _DownloadTools() | 91 pesq_path = _DownloadTools() |
82 | 92 |
83 test_executable_path = os.path.join(args.build_dir, | 93 out_dir = os.path.join(args.build_dir, '..') |
84 'low_bandwidth_audio_test' + _GetExecutableExtension()) | 94 if args.android: |
95 test_command = [os.path.join(args.build_dir, 'bin', | |
96 'run_low_bandwidth_audio_test'), '-v'] | |
97 else: | |
98 test_command = [os.path.join(args.build_dir, 'low_bandwidth_audio_test')] | |
85 | 99 |
86 # Start the test executable that produces audio files. | 100 # Start the test executable that produces audio files. |
87 command = [test_executable_path] | 101 test_process = subprocess.Popen(_LogCommand(test_command), |
88 test_process = subprocess.Popen(_LogCommand(command), stdout=subprocess.PIPE) | 102 stdout=subprocess.PIPE) |
89 | 103 |
90 for line in iter(test_process.stdout.readline, ''): | 104 for line in iter(test_process.stdout.readline, ''): |
91 # Echo the output to screen. | 105 # Echo the output to screen. |
92 sys.stdout.write(line) | 106 sys.stdout.write(line) |
93 | 107 |
94 # Extract specific lines that contain information about produced files. | 108 # Extract specific lines that contain information about produced files. |
95 match = re.search(r'^TEST (\w+) ([^:]+?):([^:]+?)\n?$', line) | 109 # Output from Android has a prefix, need to skip it. |
110 match = re.search(r'^(?:I\b.+\b)?TEST (\w+) ([^ ]+?) ([^ ]+?)\s*$', line) | |
96 if not match: | 111 if not match: |
97 continue | 112 continue |
98 test_name, reference_file, degraded_file = match.groups() | 113 test_name = match.group(1) |
114 reference_file = _GetFile(match.group(2), out_dir, | |
115 args.android, args.adb_path) | |
116 degraded_file = _GetFile(match.group(3), out_dir, | |
117 args.android, args.adb_path) | |
99 | 118 |
100 # Analyze audio | 119 # Analyze audio. |
101 command = [pesq_path, '+16000', reference_file, degraded_file] | 120 pesq_command = [pesq_path, '+16000', |
102 pesq_output = subprocess.check_output(_LogCommand(command)) | 121 os.path.basename(reference_file), |
103 | 122 os.path.basename(degraded_file)] |
104 if args.remove: | 123 # Need to provide paths in the current directory due to a bug in PESQ: |
105 os.remove(degraded_file) | 124 # On Mac, for some 'path/to/file.wav', if 'file.wav' is longer than |
125 # 'path/to', PESQ crashes. | |
126 pesq_output = subprocess.check_output(_LogCommand(pesq_command), | |
127 cwd=out_dir) | |
106 | 128 |
107 # Find the scores in stdout of pesq. | 129 # Find the scores in stdout of pesq. |
108 match = re.search( | 130 match = re.search( |
109 r'Prediction \(Raw MOS, MOS-LQO\):\s+=\s+([\d.]+)\s+([\d.]+)', | 131 r'Prediction \(Raw MOS, MOS-LQO\):\s+=\s+([\d.]+)\s+([\d.]+)', |
110 pesq_output) | 132 pesq_output) |
111 if match: | 133 if match: |
112 raw_mos, _ = match.groups() | 134 raw_mos, _ = match.groups() |
113 | 135 |
114 # Output a result for the perf dashboard. | 136 # Output a result for the perf dashboard. |
115 print 'RESULT pesq_mos: %s= %s score' % (test_name, raw_mos) | 137 print 'RESULT pesq_mos: %s= %s score' % (test_name, raw_mos) |
116 else: | 138 else: |
117 logging.error('PESQ: %s', pesq_output.splitlines()[-1]) | 139 logging.error('PESQ: %s', pesq_output.splitlines()[-1]) |
118 | 140 |
141 if args.remove: | |
142 os.remove(reference_file) | |
143 os.remove(degraded_file) | |
144 | |
119 return test_process.wait() | 145 return test_process.wait() |
120 | 146 |
121 | 147 |
122 if __name__ == '__main__': | 148 if __name__ == '__main__': |
123 sys.exit(main()) | 149 sys.exit(main()) |
OLD | NEW |