mirror of
				https://github.com/optim-enterprises-bv/kubernetes.git
				synced 2025-11-03 19:58:17 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			197 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			197 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
#!/usr/bin/env python
 | 
						|
 | 
						|
# Copyright 2016 The Kubernetes 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 collections
 | 
						|
import csv
 | 
						|
import re
 | 
						|
import json
 | 
						|
import os
 | 
						|
import random
 | 
						|
import subprocess
 | 
						|
import sys
 | 
						|
import time
 | 
						|
import urllib2
 | 
						|
import zlib
 | 
						|
 | 
						|
BASE_DIR = os.path.dirname(os.path.abspath(__file__))
 | 
						|
OWNERS_PATH = os.path.abspath(
 | 
						|
    os.path.join(BASE_DIR, '..', 'test', 'test_owners.csv'))
 | 
						|
GCS_URL_BASE = 'https://storage.googleapis.com/kubernetes-test-history/'
 | 
						|
SKIP_MAINTAINERS = {
 | 
						|
    'a-robinson', 'aronchick', 'bgrant0607-nocc', 'david-mcmahon',
 | 
						|
    'goltermann', 'sarahnovotny'}
 | 
						|
 | 
						|
 | 
						|
def normalize(name):
 | 
						|
    name = re.sub(r'\[.*?\]|\{.*?\}', '', name)
 | 
						|
    name = re.sub(r'\s+', ' ', name)
 | 
						|
    return name.strip()
 | 
						|
 | 
						|
 | 
						|
def get_test_history(days_ago):
 | 
						|
    url = time.strftime(GCS_URL_BASE + 'logs/%Y-%m-%d.json',
 | 
						|
                        time.gmtime(time.time() - days_ago * 24 * 60 * 60))
 | 
						|
    resp = urllib2.urlopen(url)
 | 
						|
    content = resp.read()
 | 
						|
    if resp.headers.get('content-encoding') == 'gzip':
 | 
						|
        content = zlib.decompress(content, 15 | 16)
 | 
						|
    return json.loads(content)
 | 
						|
 | 
						|
 | 
						|
def get_test_names_from_test_history():
 | 
						|
    test_names = set()
 | 
						|
    for days_ago in range(4):
 | 
						|
        test_history = get_test_history(days_ago)
 | 
						|
        test_names.update(normalize(name) for name in test_history['test_names'])
 | 
						|
    return test_names
 | 
						|
 | 
						|
 | 
						|
def get_test_names_from_local_files():
 | 
						|
    tests_json = subprocess.check_output(['go', 'run', 'test/list/main.go', '-json'])
 | 
						|
    tests = json.loads(tests_json)
 | 
						|
    return {normalize(t['Name'] + (' ' + t['TestName'] if 'k8s.io/' not in t['Name'] else ''))
 | 
						|
            for t in tests}
 | 
						|
 | 
						|
 | 
						|
def load_owners(fname):
 | 
						|
    owners = {}
 | 
						|
    with open(fname) as f:
 | 
						|
        for n, (name, owner, random_assignment) in enumerate(csv.reader(f)):
 | 
						|
            if n == 0:
 | 
						|
                continue  # header
 | 
						|
            owners[normalize(name)] = (owner, int(random_assignment))
 | 
						|
        return owners
 | 
						|
 | 
						|
 | 
						|
def write_owners(fname, owners):
 | 
						|
    with open(fname, 'w') as f:
 | 
						|
        out = csv.writer(f, lineterminator='\n')
 | 
						|
        out.writerow(['name', 'owner', 'auto-assigned'])
 | 
						|
        sort_key = lambda (k, v): (k != 'DEFAULT', k)  # put 'DEFAULT' first.
 | 
						|
        items = sorted(owners.items(), key=sort_key)
 | 
						|
        for name, (owner, random_assignment) in items:
 | 
						|
            out.writerow([name, owner, int(random_assignment)])
 | 
						|
 | 
						|
 | 
						|
def get_maintainers():
 | 
						|
    # Github doesn't seem to support team membership listing without a key with
 | 
						|
    # org admin privileges. Instead, we do it manually:
 | 
						|
    # Open https://github.com/orgs/kubernetes/teams/kubernetes-maintainers
 | 
						|
    # Run this in the js console:
 | 
						|
    # [].slice.call(document.querySelectorAll('.team-member-username a')).map(
 | 
						|
    #     e => e.textContent.trim())
 | 
						|
    ret = {"a-robinson", "alex-mohr", "amygdala", "andyzheng0831", "apelisse",
 | 
						|
           "aronchick", "bgrant0607", "bgrant0607-nocc", "bprashanth",
 | 
						|
           "brendandburns", "caesarxuchao", "childsb", "cjcullen",
 | 
						|
           "david-mcmahon", "davidopp", "dchen1107", "deads2k",
 | 
						|
           "derekwaynecarr", "dubstack", "eparis", "erictune", "fabioy",
 | 
						|
           "fejta", "fgrzadkowski", "freehan", "ghodss", "girishkalele",
 | 
						|
           "gmarek", "goltermann", "grodrigues3", "hurf", "ingvagabund", "ixdy",
 | 
						|
           "jackgr", "janetkuo", "jbeda", "jdef", "jfrazelle", "jingxu97",
 | 
						|
           "jlowdermilk", "jsafrane", "jszczepkowski", "justinsb", "kargakis",
 | 
						|
           "karlkfi", "kelseyhightower", "kevin-wangzefeng", "krousey",
 | 
						|
           "lavalamp", "liggitt", "luxas", "madhusudancs", "maisem", "mansoorj",
 | 
						|
           "matchstick", "mbohlool", "mikedanese", "mml", "mtaufen", "mwielgus",
 | 
						|
           "ncdc", "nikhiljindal", "piosz", "pmorie", "pwittrock", "Q-Lee",
 | 
						|
           "quinton-hoole", "Random-Liu", "rmmh", "roberthbailey", "ronnielai",
 | 
						|
           "saad-ali", "sarahnovotny", "smarterclayton", "soltysh", "spxtr",
 | 
						|
           "sttts", "thockin", "timothysc", "timstclair", "tmrts",
 | 
						|
           "vishh", "vulpecula", "wojtek-t", "xiang90", "yifan-gu", "yujuhong",
 | 
						|
           "zmerlynn"}
 | 
						|
    return sorted(ret - SKIP_MAINTAINERS)
 | 
						|
 | 
						|
 | 
						|
def detect_github_username():
 | 
						|
    origin_url = subprocess.check_output(['git', 'config', 'remote.origin.url'])
 | 
						|
    m = re.search(r'github.com[:/](.*)/', origin_url)
 | 
						|
    if m and m.group(1) != 'kubernetes':
 | 
						|
        return m.group(1)
 | 
						|
    raise ValueError('unable to determine GitHub user from '
 | 
						|
                     '`git config remote.origin.url` output, run with --user instead')
 | 
						|
 | 
						|
 | 
						|
def main():
 | 
						|
    parser = argparse.ArgumentParser()
 | 
						|
    parser.add_argument('--history', action='store_true', help='Generate test list from result history.')
 | 
						|
    parser.add_argument('--user', help='User to assign new tests to (or RANDOM, default: current GitHub user).')
 | 
						|
    parser.add_argument('--check', action='store_true', help='Exit with a nonzero status if the test list has changed.')
 | 
						|
    options = parser.parse_args()
 | 
						|
 | 
						|
    if options.history:
 | 
						|
        test_names = get_test_names_from_test_history()
 | 
						|
    else:
 | 
						|
        test_names = get_test_names_from_local_files()
 | 
						|
    test_names.add('DEFAULT')
 | 
						|
    test_names = sorted(test_names)
 | 
						|
    owners = load_owners(OWNERS_PATH)
 | 
						|
 | 
						|
    outdated_tests = sorted(set(owners) - set(test_names))
 | 
						|
    new_tests = sorted(set(test_names) - set(owners))
 | 
						|
    maintainers = get_maintainers()
 | 
						|
 | 
						|
    print '# OUTDATED TESTS (%d):' % len(outdated_tests)
 | 
						|
    print  '\n'.join('%s -- %s%s' %
 | 
						|
                     (t, owners[t][0], ['', ' (random)'][owners[t][1]])
 | 
						|
                      for t in outdated_tests)
 | 
						|
    print '# NEW TESTS (%d):' % len(new_tests)
 | 
						|
    print  '\n'.join(new_tests)
 | 
						|
 | 
						|
    if options.check:
 | 
						|
        if new_tests or outdated_tests:
 | 
						|
            print
 | 
						|
            print 'ERROR: the test list has changed'
 | 
						|
            sys.exit(1)
 | 
						|
        sys.exit(0)
 | 
						|
 | 
						|
    if not options.user:
 | 
						|
        options.user = detect_github_username()
 | 
						|
 | 
						|
    for name in outdated_tests:
 | 
						|
        owners.pop(name)
 | 
						|
 | 
						|
    print '# UNEXPECTED MAINTAINERS ',
 | 
						|
    print '(randomly assigned, but not in kubernetes-maintainers)'
 | 
						|
    for name, (owner, random_assignment) in sorted(owners.iteritems()):
 | 
						|
        if random_assignment and owner not in maintainers:
 | 
						|
            print '%-16s %s' % (owner, name)
 | 
						|
            owners.pop(name)
 | 
						|
    print
 | 
						|
 | 
						|
    owner_counts = collections.Counter(
 | 
						|
        owner for name, (owner, random) in owners.iteritems()
 | 
						|
        if owner in maintainers)
 | 
						|
    for test_name in set(test_names) - set(owners):
 | 
						|
        random_assignment = True
 | 
						|
        if options.user.lower() == 'random':
 | 
						|
            new_owner, _count = random.choice(owner_counts.most_common()[-4:])
 | 
						|
        else:
 | 
						|
            new_owner = options.user
 | 
						|
            random_assignment = False
 | 
						|
        owner_counts[new_owner] += 1
 | 
						|
        owners[test_name] = (new_owner, random_assignment)
 | 
						|
 | 
						|
    if options.user.lower() == 'random':
 | 
						|
        print '# Tests per maintainer:'
 | 
						|
        for owner, count in owner_counts.most_common():
 | 
						|
            print '%-20s %3d' % (owner, count)
 | 
						|
 | 
						|
    write_owners(OWNERS_PATH, owners)
 | 
						|
 | 
						|
 | 
						|
if __name__ == '__main__':
 | 
						|
    main()
 |