Add CTS telemetry harness support

Migrates harness support files added in
https://chromium-review.googlesource.com/c/chromium/src/+/3541414
into Dawn's repo.

Tested in
https://chromium-review.googlesource.com/c/chromium/src/+/3537743

Bug: chromium:1306640
Change-Id: I3000b1223219a1da293af910bf442570b70b7c92
Reviewed-on: https://dawn-review.googlesource.com/c/dawn/+/84043
Reviewed-by: Loko Kung <lokokung@google.com>
Commit-Queue: Austin Eng <enga@chromium.org>
diff --git a/webgpu-cts/scripts/compile_src.py b/webgpu-cts/scripts/compile_src.py
new file mode 100755
index 0000000..b08c0a7
--- /dev/null
+++ b/webgpu-cts/scripts/compile_src.py
@@ -0,0 +1,101 @@
+#!/usr/bin/env python
+#
+# Copyright 2022 The Dawn Authors
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+#     http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+
+import os
+import shutil
+import sys
+
+from dir_paths import webgpu_cts_root_dir, node_dir
+from tsc_ignore_errors import run_tsc_ignore_errors
+
+try:
+    old_sys_path = sys.path
+    sys.path = [node_dir] + sys.path
+
+    from node import RunNode
+finally:
+    sys.path = old_sys_path
+
+
+def compile_src(out_dir):
+    # First, clean the output directory so deleted files are pruned from old builds.
+    shutil.rmtree(out_dir)
+
+    run_tsc_ignore_errors([
+        '--project',
+        os.path.join(webgpu_cts_root_dir, 'tsconfig.json'),
+        '--outDir',
+        out_dir,
+        '--noEmit',
+        'false',
+        '--noEmitOnError',
+        'false',
+        '--declaration',
+        'false',
+        '--sourceMap',
+        'false',
+        '--target',
+        'ES2017',
+    ])
+
+
+def compile_src_for_node(out_dir, additional_args=None, clean=True):
+    additional_args = additional_args or []
+    if clean:
+        # First, clean the output directory so deleted files are pruned from old builds.
+        shutil.rmtree(out_dir)
+
+    args = [
+        '--project',
+        os.path.join(webgpu_cts_root_dir, 'node.tsconfig.json'),
+        '--outDir',
+        out_dir,
+        '--noEmit',
+        'false',
+        '--noEmitOnError',
+        'false',
+        '--declaration',
+        'false',
+        '--sourceMap',
+        'false',
+        '--target',
+        'ES6',
+    ]
+    args.extend(additional_args)
+
+    run_tsc_ignore_errors(args)
+
+
+if __name__ == '__main__':
+    if len(sys.argv) != 2:
+        print('Usage: compile_src.py GEN_DIR')
+        sys.exit(1)
+
+    gen_dir = sys.argv[1]
+
+    # Compile the CTS src.
+    compile_src(os.path.join(gen_dir, 'src'))
+    compile_src_for_node(os.path.join(gen_dir, 'src-node'))
+
+    # Run gen_listings.js to overwrite the dummy src/webgpu/listings.js created
+    # from transpiling src/
+    RunNode([
+        os.path.join(gen_dir, 'src-node', 'common', 'tools',
+                     'gen_listings.js'),
+        '--no-validate',
+        os.path.join(gen_dir, 'src'),
+        os.path.join(gen_dir, 'src-node', 'webgpu'),
+    ])
diff --git a/webgpu-cts/scripts/dir_paths.py b/webgpu-cts/scripts/dir_paths.py
new file mode 100644
index 0000000..cc45758
--- /dev/null
+++ b/webgpu-cts/scripts/dir_paths.py
@@ -0,0 +1,32 @@
+#!/usr/bin/env python
+#
+# Copyright 2022 The Dawn Authors
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+#     http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+
+import os
+
+webgpu_cts_scripts_dir = os.path.dirname(os.path.abspath(__file__))
+dawn_third_party_dir = os.path.join(
+    os.path.dirname(os.path.dirname(webgpu_cts_scripts_dir)), 'third_party')
+gn_webgpu_cts_dir = os.path.join(dawn_third_party_dir, 'gn', 'webgpu-cts')
+webgpu_cts_root_dir = os.path.join(dawn_third_party_dir, 'webgpu-cts')
+chromium_third_party_dir = None
+node_dir = None
+
+_possible_chromium_third_party_dir = os.path.dirname(
+    os.path.dirname(dawn_third_party_dir))
+_possible_node_dir = os.path.join(_possible_chromium_third_party_dir, 'node')
+if os.path.exists(_possible_node_dir):
+    chromium_third_party_dir = _possible_chromium_third_party_dir
+    node_dir = _possible_node_dir
diff --git a/webgpu-cts/scripts/gen_ts_dep_lists.py b/webgpu-cts/scripts/gen_ts_dep_lists.py
new file mode 100755
index 0000000..5862ab2
--- /dev/null
+++ b/webgpu-cts/scripts/gen_ts_dep_lists.py
@@ -0,0 +1,86 @@
+#!/usr/bin/env python
+#
+# Copyright 2022 The Dawn Authors
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+#     http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+
+import argparse
+import os
+import sys
+
+from dir_paths import gn_webgpu_cts_dir, webgpu_cts_root_dir
+from tsc_ignore_errors import run_tsc_ignore_errors
+
+src_prefix = webgpu_cts_root_dir.replace('\\', '/') + '/'
+
+
+def get_ts_sources():
+    # This will output all the source files in the form:
+    # "/absolute/path/to/file.ts"
+    # The path is always Unix-style.
+    # It will also output many Typescript errors since the build doesn't download the .d.ts
+    # dependencies.
+    stdout = run_tsc_ignore_errors([
+        '--project',
+        os.path.join(webgpu_cts_root_dir, 'tsconfig.json'), '--listFiles',
+        '--declaration', 'false', '--sourceMap', 'false'
+    ])
+
+    lines = [l.decode() for l in stdout.splitlines()]
+    return [
+        line[len(src_prefix):] for line in lines
+        if line.startswith(src_prefix + 'src/')
+    ]
+
+
+def get_resource_files():
+    files = os.listdir(os.path.join(webgpu_cts_root_dir, 'src', 'resources'))
+    files.sort()
+    return files
+
+
+if __name__ == '__main__':
+    parser = argparse.ArgumentParser()
+    parser.add_argument('--check',
+                        action='store_true',
+                        help='Check that the output file is up to date.')
+    parser.add_argument('--stamp', help='Stamp file to write after success.')
+    args = parser.parse_args()
+
+    ts_sources = [x + '\n' for x in get_ts_sources()]
+    ts_sources_txt = os.path.join(gn_webgpu_cts_dir, 'ts_sources.txt')
+
+    resource_files = [x + '\n' for x in get_resource_files()]
+    resource_files_txt = os.path.join(gn_webgpu_cts_dir, 'resource_files.txt')
+
+    if args.check:
+        with open(ts_sources_txt, 'r') as f:
+            txt = f.readlines()
+            if (txt != ts_sources):
+                raise RuntimeError(
+                    '%s is out of date. Please re-run //third_party/dawn/third_party/webgpu-cts/scripts/gen_ts_dep_lists.py\n'
+                    % ts_sources_txt)
+        with open(resource_files_txt, 'r') as f:
+            if (f.readlines() != resource_files):
+                raise RuntimeError(
+                    '%s is out of date. Please re-run //third_party/dawn/third_party/webgpu-cts/scripts/gen_ts_dep_lists.py\n'
+                    % resource_files_txt)
+    else:
+        with open(ts_sources_txt, 'w') as f:
+            f.writelines(ts_sources)
+        with open(resource_files_txt, 'w') as f:
+            f.writelines(resource_files)
+
+    if args.stamp:
+        with open(args.stamp, 'w') as f:
+            f.write('')
diff --git a/webgpu-cts/scripts/list.py b/webgpu-cts/scripts/list.py
new file mode 100755
index 0000000..d63beb4
--- /dev/null
+++ b/webgpu-cts/scripts/list.py
@@ -0,0 +1,70 @@
+#!/usr/bin/env python
+#
+# Copyright 2022 The Dawn Authors
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+#     http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+
+import argparse
+import logging
+import os
+import shutil
+import sys
+import tempfile
+
+from dir_paths import node_dir
+
+from compile_src import compile_src_for_node
+
+
+def list_testcases(query, js_out_dir=None):
+    if js_out_dir is None:
+        js_out_dir = tempfile.mkdtemp()
+        delete_js_out_dir = True
+    else:
+        delete_js_out_dir = False
+
+    try:
+        logging.info('WebGPU CTS: Transpiling tools...')
+        compile_src_for_node(js_out_dir, [
+            '--incremental', '--tsBuildInfoFile',
+            os.path.join(js_out_dir, 'build.tsbuildinfo')
+        ],
+                             clean=False)
+
+        old_sys_path = sys.path
+        try:
+            sys.path = old_sys_path + [node_dir]
+            from node import RunNode
+        finally:
+            sys.path = old_sys_path
+
+        return RunNode([
+            os.path.join(js_out_dir, 'common', 'runtime', 'cmdline.js'), query,
+            '--list'
+        ])
+    finally:
+        if delete_js_out_dir:
+            shutil.rmtree(js_out_dir)
+
+
+# List all testcases matching a test query.
+if __name__ == '__main__':
+    parser = argparse.ArgumentParser()
+    parser.add_argument('--query', default='webgpu:*', help='WebGPU CTS Query')
+    parser.add_argument(
+        '--js-out-dir',
+        default=None,
+        help='Output directory for intermediate compiled JS sources')
+    args = parser.parse_args()
+
+    print(list_testcases(args.query, args.js_out_dir))
diff --git a/webgpu-cts/scripts/tsc_ignore_errors.py b/webgpu-cts/scripts/tsc_ignore_errors.py
new file mode 100755
index 0000000..14b8455
--- /dev/null
+++ b/webgpu-cts/scripts/tsc_ignore_errors.py
@@ -0,0 +1,51 @@
+#!/usr/bin/env python
+#
+# Copyright 2022 The Dawn Authors
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+#     http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+
+import subprocess
+import sys
+import os
+
+from dir_paths import node_dir
+
+try:
+    old_sys_path = sys.path
+    sys.path = [node_dir] + sys.path
+
+    from node import GetBinaryPath as get_node_binary_path
+finally:
+    sys.path = old_sys_path
+
+tsc = os.path.join(node_dir, 'node_modules', 'typescript', 'lib', 'tsc.js')
+
+
+def run_tsc_ignore_errors(args):
+    cmd = [get_node_binary_path(), tsc] + args
+    process = subprocess.Popen(cmd,
+                               cwd=os.getcwd(),
+                               stdout=subprocess.PIPE,
+                               stderr=subprocess.PIPE)
+
+    stdout, stderr = process.communicate()
+
+    # Typecheck errors go in stdout, not stderr. If we see something in stderr, raise an error.
+    if len(stderr):
+        raise RuntimeError('tsc \'%s\' failed\n%s' % (' '.join(cmd), stderr))
+
+    return stdout
+
+
+if __name__ == '__main__':
+    run_tsc_ignore_errors(sys.argv[1:])