You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			207 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			Python
		
	
			
		
		
	
	
			207 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			Python
		
	
# Copyright (c) 2020 The Chromium Authors. All rights reserved.
 | 
						|
# Use of this source code is governed by a BSD-style license that can be
 | 
						|
# found in the LICENSE file.
 | 
						|
 | 
						|
import itertools
 | 
						|
import os
 | 
						|
import random
 | 
						|
import threading
 | 
						|
 | 
						|
import gerrit_util
 | 
						|
import git_common
 | 
						|
import owners as owners_db
 | 
						|
import scm
 | 
						|
 | 
						|
 | 
						|
class OwnersClient(object):
 | 
						|
  """Interact with OWNERS files in a repository.
 | 
						|
 | 
						|
  This class allows you to interact with OWNERS files in a repository both the
 | 
						|
  Gerrit Code-Owners plugin REST API, and the owners database implemented by
 | 
						|
  Depot Tools in owners.py:
 | 
						|
 | 
						|
   - List all the owners for a group of files.
 | 
						|
   - Check if files have been approved.
 | 
						|
   - Suggest owners for a group of files.
 | 
						|
 | 
						|
  All code should use this class to interact with OWNERS files instead of the
 | 
						|
  owners database in owners.py
 | 
						|
  """
 | 
						|
  # '*' means that everyone can approve.
 | 
						|
  EVERYONE = '*'
 | 
						|
 | 
						|
  # Possible status of a file.
 | 
						|
  # - INSUFFICIENT_REVIEWERS: The path needs owners approval, but none of its
 | 
						|
  #   owners is currently a reviewer of the change.
 | 
						|
  # - PENDING: An owner of this path has been added as reviewer, but approval
 | 
						|
  #   has not been given yet.
 | 
						|
  # - APPROVED: The path has been approved by an owner.
 | 
						|
  APPROVED = 'APPROVED'
 | 
						|
  PENDING = 'PENDING'
 | 
						|
  INSUFFICIENT_REVIEWERS = 'INSUFFICIENT_REVIEWERS'
 | 
						|
 | 
						|
  def ListOwners(self, path):
 | 
						|
    """List all owners for a file.
 | 
						|
 | 
						|
    The returned list is sorted so that better owners appear first.
 | 
						|
    """
 | 
						|
    raise Exception('Not implemented')
 | 
						|
 | 
						|
  def BatchListOwners(self, paths):
 | 
						|
    """List all owners for a group of files.
 | 
						|
 | 
						|
    Returns a dictionary {path: [owners]}.
 | 
						|
    """
 | 
						|
    with git_common.ScopedPool(kind='threads') as pool:
 | 
						|
      return dict(pool.imap_unordered(
 | 
						|
          lambda p: (p, self.ListOwners(p)), paths))
 | 
						|
 | 
						|
  def GetFilesApprovalStatus(self, paths, approvers, reviewers):
 | 
						|
    """Check the approval status for the given paths.
 | 
						|
 | 
						|
    Utility method to check for approval status when a change has not yet been
 | 
						|
    created, given reviewers and approvers.
 | 
						|
 | 
						|
    See GetChangeApprovalStatus for description of the returned value.
 | 
						|
    """
 | 
						|
    approvers = set(approvers)
 | 
						|
    if approvers:
 | 
						|
      approvers.add(self.EVERYONE)
 | 
						|
    reviewers = set(reviewers)
 | 
						|
    if reviewers:
 | 
						|
      reviewers.add(self.EVERYONE)
 | 
						|
    status = {}
 | 
						|
    owners_by_path = self.BatchListOwners(paths)
 | 
						|
    for path, owners in owners_by_path.items():
 | 
						|
      owners = set(owners)
 | 
						|
      if owners.intersection(approvers):
 | 
						|
        status[path] = self.APPROVED
 | 
						|
      elif owners.intersection(reviewers):
 | 
						|
        status[path] = self.PENDING
 | 
						|
      else:
 | 
						|
        status[path] = self.INSUFFICIENT_REVIEWERS
 | 
						|
    return status
 | 
						|
 | 
						|
  def ScoreOwners(self, paths, exclude=None):
 | 
						|
    """Get sorted list of owners for the given paths."""
 | 
						|
    if not paths:
 | 
						|
      return []
 | 
						|
    exclude = exclude or []
 | 
						|
    owners = []
 | 
						|
    queues = self.BatchListOwners(paths).values()
 | 
						|
    for i in range(max(len(q) for q in queues)):
 | 
						|
      for q in queues:
 | 
						|
        if i < len(q) and q[i] not in owners and q[i] not in exclude:
 | 
						|
          owners.append(q[i])
 | 
						|
    return owners
 | 
						|
 | 
						|
  def SuggestOwners(self, paths, exclude=None):
 | 
						|
    """Suggest a set of owners for the given paths."""
 | 
						|
    exclude = exclude or []
 | 
						|
 | 
						|
    paths_by_owner = {}
 | 
						|
    owners_by_path = self.BatchListOwners(paths)
 | 
						|
    for path, owners in owners_by_path.items():
 | 
						|
      for owner in owners:
 | 
						|
        paths_by_owner.setdefault(owner, set()).add(path)
 | 
						|
 | 
						|
    selected = []
 | 
						|
    missing = set(paths)
 | 
						|
    for owner in self.ScoreOwners(paths, exclude=exclude):
 | 
						|
      missing_len = len(missing)
 | 
						|
      missing.difference_update(paths_by_owner[owner])
 | 
						|
      if missing_len > len(missing):
 | 
						|
        selected.append(owner)
 | 
						|
      if not missing:
 | 
						|
        break
 | 
						|
 | 
						|
    return selected
 | 
						|
 | 
						|
 | 
						|
class DepotToolsClient(OwnersClient):
 | 
						|
  """Implement OwnersClient using owners.py Database."""
 | 
						|
  def __init__(self, root, branch, fopen=open, os_path=os.path):
 | 
						|
    super(DepotToolsClient, self).__init__()
 | 
						|
 | 
						|
    self._root = root
 | 
						|
    self._branch = branch
 | 
						|
    self._fopen = fopen
 | 
						|
    self._os_path = os_path
 | 
						|
    self._db = None
 | 
						|
    self._db_lock = threading.Lock()
 | 
						|
 | 
						|
  def _ensure_db(self):
 | 
						|
    if self._db is not None:
 | 
						|
      return
 | 
						|
    self._db = owners_db.Database(self._root, self._fopen, self._os_path)
 | 
						|
    self._db.override_files = self._GetOriginalOwnersFiles()
 | 
						|
 | 
						|
  def _GetOriginalOwnersFiles(self):
 | 
						|
    return {
 | 
						|
      f: scm.GIT.GetOldContents(self._root, f, self._branch).splitlines()
 | 
						|
      for _, f in scm.GIT.CaptureStatus(self._root, self._branch)
 | 
						|
      if os.path.basename(f) == 'OWNERS'
 | 
						|
    }
 | 
						|
 | 
						|
  def ListOwners(self, path):
 | 
						|
    # all_possible_owners is not thread safe.
 | 
						|
    with self._db_lock:
 | 
						|
      self._ensure_db()
 | 
						|
      # all_possible_owners returns a dict {owner: [(path, distance)]}. We want
 | 
						|
      # to return a list of owners sorted by increasing distance.
 | 
						|
      distance_by_owner = self._db.all_possible_owners([path], None)
 | 
						|
      # We add a small random number to the distance, so that owners at the
 | 
						|
      # same distance are returned in random order to avoid overloading those
 | 
						|
      # who would appear first.
 | 
						|
      return sorted(
 | 
						|
          distance_by_owner,
 | 
						|
          key=lambda o: distance_by_owner[o][0][1] + random.random())
 | 
						|
 | 
						|
 | 
						|
class GerritClient(OwnersClient):
 | 
						|
  """Implement OwnersClient using OWNERS REST API."""
 | 
						|
  def __init__(self, host, project, branch):
 | 
						|
    super(GerritClient, self).__init__()
 | 
						|
 | 
						|
    self._host = host
 | 
						|
    self._project = project
 | 
						|
    self._branch = branch
 | 
						|
    self._owners_cache = {}
 | 
						|
 | 
						|
    # Seed used by Gerrit to shuffle code owners that have the same score. Can
 | 
						|
    # be used to make the sort order stable across several requests, e.g. to get
 | 
						|
    # the same set of random code owners for different file paths that have the
 | 
						|
    # same code owners.
 | 
						|
    self._seed = random.getrandbits(30)
 | 
						|
 | 
						|
  def ListOwners(self, path):
 | 
						|
    # Always use slashes as separators.
 | 
						|
    path = path.replace(os.sep, '/')
 | 
						|
    if path not in self._owners_cache:
 | 
						|
      # GetOwnersForFile returns a list of account details sorted by order of
 | 
						|
      # best reviewer for path. If owners have the same score, the order is
 | 
						|
      # random, seeded by `self._seed`.
 | 
						|
      data = gerrit_util.GetOwnersForFile(
 | 
						|
          self._host, self._project, self._branch, path,
 | 
						|
          resolve_all_users=False, seed=self._seed)
 | 
						|
      self._owners_cache[path] = [
 | 
						|
        d['account']['email']
 | 
						|
        for d in data['code_owners']
 | 
						|
        if 'account' in d and 'email' in d['account']
 | 
						|
      ]
 | 
						|
      # If owned_by_all_users is true, add everyone as an owner at the end of
 | 
						|
      # the owners list.
 | 
						|
      if data.get('owned_by_all_users', False):
 | 
						|
        self._owners_cache[path].append(self.EVERYONE)
 | 
						|
    return self._owners_cache[path]
 | 
						|
 | 
						|
 | 
						|
def GetCodeOwnersClient(root, upstream, host, project, branch):
 | 
						|
  """Get a new OwnersClient.
 | 
						|
 | 
						|
  Defaults to GerritClient, and falls back to DepotToolsClient if code-owners
 | 
						|
  plugin is not available."""
 | 
						|
  if gerrit_util.IsCodeOwnersEnabledOnHost(host):
 | 
						|
    return GerritClient(host, project, branch)
 | 
						|
  return DepotToolsClient(root, upstream)
 |