1*cc02d7e2SAndroid Build Coastguard Worker#!/usr/bin/env python3 2*cc02d7e2SAndroid Build Coastguard Worker 3*cc02d7e2SAndroid Build Coastguard Worker# Copyright 2022 gRPC authors. 4*cc02d7e2SAndroid Build Coastguard Worker# 5*cc02d7e2SAndroid Build Coastguard Worker# Licensed under the Apache License, Version 2.0 (the "License"); 6*cc02d7e2SAndroid Build Coastguard Worker# you may not use this file except in compliance with the License. 7*cc02d7e2SAndroid Build Coastguard Worker# You may obtain a copy of the License at 8*cc02d7e2SAndroid Build Coastguard Worker# 9*cc02d7e2SAndroid Build Coastguard Worker# http://www.apache.org/licenses/LICENSE-2.0 10*cc02d7e2SAndroid Build Coastguard Worker# 11*cc02d7e2SAndroid Build Coastguard Worker# Unless required by applicable law or agreed to in writing, software 12*cc02d7e2SAndroid Build Coastguard Worker# distributed under the License is distributed on an "AS IS" BASIS, 13*cc02d7e2SAndroid Build Coastguard Worker# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 14*cc02d7e2SAndroid Build Coastguard Worker# See the License for the specific language governing permissions and 15*cc02d7e2SAndroid Build Coastguard Worker# limitations under the License. 16*cc02d7e2SAndroid Build Coastguard Worker 17*cc02d7e2SAndroid Build Coastguard Workerimport os 18*cc02d7e2SAndroid Build Coastguard Workerimport re 19*cc02d7e2SAndroid Build Coastguard Workerimport subprocess 20*cc02d7e2SAndroid Build Coastguard Workerimport sys 21*cc02d7e2SAndroid Build Coastguard Worker 22*cc02d7e2SAndroid Build Coastguard Workerfrom google.cloud import bigquery 23*cc02d7e2SAndroid Build Coastguard Workerimport run_buildozer 24*cc02d7e2SAndroid Build Coastguard Workerimport update_flakes_query 25*cc02d7e2SAndroid Build Coastguard Worker 26*cc02d7e2SAndroid Build Coastguard Workerlookback_hours = 24 * 7 * 4 27*cc02d7e2SAndroid Build Coastguard Worker 28*cc02d7e2SAndroid Build Coastguard Worker 29*cc02d7e2SAndroid Build Coastguard Workerdef include_test(test): 30*cc02d7e2SAndroid Build Coastguard Worker if "@" in test: 31*cc02d7e2SAndroid Build Coastguard Worker return False 32*cc02d7e2SAndroid Build Coastguard Worker if test.startswith("//test/cpp/qps:"): 33*cc02d7e2SAndroid Build Coastguard Worker return False 34*cc02d7e2SAndroid Build Coastguard Worker return True 35*cc02d7e2SAndroid Build Coastguard Worker 36*cc02d7e2SAndroid Build Coastguard Worker 37*cc02d7e2SAndroid Build Coastguard WorkerTEST_DIRS = ["test/core", "test/cpp"] 38*cc02d7e2SAndroid Build Coastguard Workertests = {} 39*cc02d7e2SAndroid Build Coastguard Workeralready_flaky = set() 40*cc02d7e2SAndroid Build Coastguard Workerfor test_dir in TEST_DIRS: 41*cc02d7e2SAndroid Build Coastguard Worker for line in subprocess.check_output( 42*cc02d7e2SAndroid Build Coastguard Worker ["bazel", "query", "tests({}/...)".format(test_dir)] 43*cc02d7e2SAndroid Build Coastguard Worker ).splitlines(): 44*cc02d7e2SAndroid Build Coastguard Worker test = line.strip().decode("utf-8") 45*cc02d7e2SAndroid Build Coastguard Worker if not include_test(test): 46*cc02d7e2SAndroid Build Coastguard Worker continue 47*cc02d7e2SAndroid Build Coastguard Worker tests[test] = False 48*cc02d7e2SAndroid Build Coastguard Workerfor test_dir in TEST_DIRS: 49*cc02d7e2SAndroid Build Coastguard Worker for line in subprocess.check_output( 50*cc02d7e2SAndroid Build Coastguard Worker ["bazel", "query", "attr(flaky, 1, tests({}/...))".format(test_dir)] 51*cc02d7e2SAndroid Build Coastguard Worker ).splitlines(): 52*cc02d7e2SAndroid Build Coastguard Worker test = line.strip().decode("utf-8") 53*cc02d7e2SAndroid Build Coastguard Worker if not include_test(test): 54*cc02d7e2SAndroid Build Coastguard Worker continue 55*cc02d7e2SAndroid Build Coastguard Worker already_flaky.add(test) 56*cc02d7e2SAndroid Build Coastguard Worker 57*cc02d7e2SAndroid Build Coastguard Workerflaky_e2e = set() 58*cc02d7e2SAndroid Build Coastguard Worker 59*cc02d7e2SAndroid Build Coastguard Workerclient = bigquery.Client() 60*cc02d7e2SAndroid Build Coastguard Workerfor row in client.query( 61*cc02d7e2SAndroid Build Coastguard Worker update_flakes_query.QUERY.format(lookback_hours=lookback_hours) 62*cc02d7e2SAndroid Build Coastguard Worker).result(): 63*cc02d7e2SAndroid Build Coastguard Worker if "/macos/" in row.job_name: 64*cc02d7e2SAndroid Build Coastguard Worker continue # we know mac stuff is flaky 65*cc02d7e2SAndroid Build Coastguard Worker if row.test_binary not in tests: 66*cc02d7e2SAndroid Build Coastguard Worker m = re.match( 67*cc02d7e2SAndroid Build Coastguard Worker r"^//test/core/end2end:([^@]*)@([^@]*)(.*)", row.test_binary 68*cc02d7e2SAndroid Build Coastguard Worker ) 69*cc02d7e2SAndroid Build Coastguard Worker if m: 70*cc02d7e2SAndroid Build Coastguard Worker flaky_e2e.add("{}@{}{}".format(m.group(1), m.group(2), m.group(3))) 71*cc02d7e2SAndroid Build Coastguard Worker print("will mark end2end test {} as flaky".format(row.test_binary)) 72*cc02d7e2SAndroid Build Coastguard Worker else: 73*cc02d7e2SAndroid Build Coastguard Worker print("skip obsolete test {}".format(row.test_binary)) 74*cc02d7e2SAndroid Build Coastguard Worker continue 75*cc02d7e2SAndroid Build Coastguard Worker print("will mark {} as flaky".format(row.test_binary)) 76*cc02d7e2SAndroid Build Coastguard Worker tests[row.test_binary] = True 77*cc02d7e2SAndroid Build Coastguard Worker 78*cc02d7e2SAndroid Build Coastguard Workerbuildozer_commands = [] 79*cc02d7e2SAndroid Build Coastguard Workerfor test, flaky in sorted(tests.items()): 80*cc02d7e2SAndroid Build Coastguard Worker if flaky: 81*cc02d7e2SAndroid Build Coastguard Worker buildozer_commands.append("set flaky True|{}".format(test)) 82*cc02d7e2SAndroid Build Coastguard Worker elif test in already_flaky: 83*cc02d7e2SAndroid Build Coastguard Worker buildozer_commands.append("remove flaky|{}".format(test)) 84*cc02d7e2SAndroid Build Coastguard Worker 85*cc02d7e2SAndroid Build Coastguard Workerwith open("test/core/end2end/flaky.bzl", "w") as f: 86*cc02d7e2SAndroid Build Coastguard Worker with open(sys.argv[0]) as my_source: 87*cc02d7e2SAndroid Build Coastguard Worker for line in my_source: 88*cc02d7e2SAndroid Build Coastguard Worker if line[0] != "#": 89*cc02d7e2SAndroid Build Coastguard Worker break 90*cc02d7e2SAndroid Build Coastguard Worker for line in my_source: 91*cc02d7e2SAndroid Build Coastguard Worker if line[0] == "#": 92*cc02d7e2SAndroid Build Coastguard Worker print(line.strip(), file=f) 93*cc02d7e2SAndroid Build Coastguard Worker break 94*cc02d7e2SAndroid Build Coastguard Worker for line in my_source: 95*cc02d7e2SAndroid Build Coastguard Worker if line[0] != "#": 96*cc02d7e2SAndroid Build Coastguard Worker break 97*cc02d7e2SAndroid Build Coastguard Worker print(line.strip(), file=f) 98*cc02d7e2SAndroid Build Coastguard Worker print( 99*cc02d7e2SAndroid Build Coastguard Worker ( 100*cc02d7e2SAndroid Build Coastguard Worker '"""A list of flaky tests, consumed by generate_tests.bzl to set' 101*cc02d7e2SAndroid Build Coastguard Worker ' flaky attrs."""' 102*cc02d7e2SAndroid Build Coastguard Worker ), 103*cc02d7e2SAndroid Build Coastguard Worker file=f, 104*cc02d7e2SAndroid Build Coastguard Worker ) 105*cc02d7e2SAndroid Build Coastguard Worker print("FLAKY_TESTS = [", file=f) 106*cc02d7e2SAndroid Build Coastguard Worker for line in sorted(list(flaky_e2e)): 107*cc02d7e2SAndroid Build Coastguard Worker print(' "{}",'.format(line), file=f) 108*cc02d7e2SAndroid Build Coastguard Worker print("]", file=f) 109*cc02d7e2SAndroid Build Coastguard Worker 110*cc02d7e2SAndroid Build Coastguard Workerrun_buildozer.run_buildozer(buildozer_commands) 111