blob: 0e2e126782cbe6863ad58ad612cdfc360afa1738 [file] [log] [blame] [edit]
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
# Copyright 2019 The Chromium OS Authors. All rights reserved.
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
"""Linter for various OWNERS files."""
from __future__ import division
from __future__ import print_function
from pathlib import Path
import sys
TOP_DIR = Path(__file__).resolve().parent.parent
# Find chromite!
sys.path.insert(0, str(TOP_DIR.parent.parent))
# pylint: disable=wrong-import-position
from chromite.lib import commandline
from chromite.lib import git
from chromite.lib import cros_logging as logging
# pylint: enable=wrong-import-position
def GetActiveProjects():
"""Return the list of active projects."""
# Look at all the paths (files & dirs) in the top of the git repo. This way
# we ignore local directories devs created that aren't actually committed.
cmd = ['ls-tree', '--name-only', '-z', 'HEAD']
result = git.RunGit(TOP_DIR, cmd)
# Split the output on NULs to avoid whitespace/etc... issues.
paths = result.stdout.split('\0')
# ls-tree -z will include a trailing NUL on all entries, not just seperation,
# so filter it out if found (in case ls-tree behavior changes on us).
for path in [Path(x) for x in paths if x]:
if (TOP_DIR / path).is_dir():
yield path
def CheckSubdirs():
"""Check the subdir OWNERS files exist."""
ret = 0
for proj in GetActiveProjects():
path = TOP_DIR / proj / 'OWNERS'
if not path.exists():
logging.error('*** Project "%s" needs an OWNERS file', proj)
ret = 1
continue
data = path.read_text()
for i, line in enumerate(data.splitlines(), start=1):
if line.strip().startswith('set noparent'):
logging.error('*** %s:%i: Do not use "noparent" in top level projects',
path, i)
ret = 1
if line.strip() != line:
logging.error('*** %s:%i: Trim leading/trailing whitespace', path, i)
ret = 1
if not data:
logging.error('*** %s: File is empty', path)
ret = 1
if not data.endswith('\n'):
logging.error('*** %s: Missing trailing newline', path)
ret = 1
if data.startswith('\n'):
logging.error('*** %s: Trim leading blanklines', path)
ret = 1
if data.endswith('\n\n'):
logging.error('*** %s: Trim trailing blanklines', path)
ret = 1
return ret
def GetParser():
"""Return an argument parser."""
parser = commandline.ArgumentParser(description=__doc__)
return parser
def main(argv):
"""The main func!"""
parser = GetParser()
opts = parser.parse_args(argv)
opts.Freeze()
return CheckSubdirs()
if __name__ == '__main__':
commandline.ScriptWrapperMain(lambda _: main)