forked from project-chip/connectedhomeip
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Enforce that files in the
src
directory are referenced from BUILD.gn (
project-chip#31960) * Start creating a script * Have much more functionality * Restyle * Add some doc comments ... this starts being usable * Add workflow to validate that gn knows about files * Remove controller from known exceptions: we fixed that one * Fix flake8 * Add more known failures * Better error reporting for gn reachability * Remove the platform specific orphan file listing * Force the "not failures anymore" to be fatal --------- Co-authored-by: Andrei Litvin <[email protected]>
- Loading branch information
Showing
2 changed files
with
342 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,182 @@ | ||
#!/usr/bin/env python3 | ||
# | ||
# Copyright (c) 2024 Project CHIP 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. | ||
# | ||
""" | ||
Lists files specific files from a source tree and ensures | ||
they are covered by GN in some way. | ||
'Covered' is very loosely and it just tries to see if the GN text | ||
contains that word without trying to validate if this is a | ||
comment or some actual 'source' element. | ||
It is intended as a failsafe to not foget adding source files | ||
to gn. | ||
""" | ||
import logging | ||
import os | ||
import sys | ||
from pathlib import Path, PurePath | ||
from typing import Dict, Set | ||
|
||
import click | ||
import coloredlogs | ||
|
||
__LOG_LEVELS__ = { | ||
'debug': logging.DEBUG, | ||
'info': logging.INFO, | ||
'warn': logging.WARN, | ||
'fatal': logging.FATAL, | ||
} | ||
|
||
|
||
class OrphanChecker: | ||
def __init__(self): | ||
self.gn_data: Dict[str, str] = {} | ||
self.known_failures: Set[str] = set() | ||
self.fatal_failures = 0 | ||
self.failures = 0 | ||
self.found_failures: Set[str] = set() | ||
|
||
def AppendGnData(self, gn: PurePath): | ||
"""Adds a GN file to the list of internally known GN data. | ||
Will read the entire content of the GN file in memory for future reference. | ||
""" | ||
logging.debug(f'Adding GN {gn!s} for {gn.parent!s}') | ||
self.gn_data[str(gn.parent)] = gn.read_text('utf-8') | ||
|
||
def AddKnownFailure(self, k: str): | ||
self.known_failures.add(k) | ||
|
||
def _IsKnownFailure(self, path: str) -> bool: | ||
"""check if failing on the given path is a known/acceptable failure""" | ||
for k in self.known_failures: | ||
if path == k or path.endswith(os.path.sep + k): | ||
# mark some found failures to report if something is supposed | ||
# to be known but it is not | ||
self.found_failures.add(k) | ||
return True | ||
return False | ||
|
||
def Check(self, top_dir: str, file: PurePath): | ||
""" | ||
Validates that the given path is somehow referenced in GN files in any | ||
of the parent sub-directories of the file. | ||
`file` must be relative to `top_dir`. Top_dir is used to resolve relative | ||
paths in error reports and known failure checks. | ||
""" | ||
# Check logic: | ||
# - ensure the file name is included in some GN file inside this or | ||
# upper directory (although upper directory is not ideal) | ||
for p in file.parents: | ||
data = self.gn_data.get(str(p), None) | ||
if not data: | ||
continue | ||
|
||
if file.name in data: | ||
logging.debug("%s found in BUILD.gn for %s", file, p) | ||
return | ||
|
||
path = str(file.relative_to(top_dir)) | ||
if not self._IsKnownFailure(path): | ||
logging.error("UNKNOWN to gn: %s", path) | ||
self.fatal_failures += 1 | ||
else: | ||
logging.warning("UNKNOWN to gn: %s (known error)", path) | ||
|
||
self.failures += 1 | ||
|
||
|
||
@click.command() | ||
@click.option( | ||
'--log-level', | ||
default='INFO', | ||
type=click.Choice(list(__LOG_LEVELS__.keys()), case_sensitive=False), | ||
help='Determines the verbosity of script output', | ||
) | ||
@click.option( | ||
'--extensions', | ||
default=["cpp", "cc", "c", "h", "hpp"], | ||
type=str, multiple=True, | ||
help='What file extensions to consider', | ||
) | ||
@click.option( | ||
'--known-failure', | ||
type=str, multiple=True, | ||
help='What paths are known to fail', | ||
) | ||
@click.option( | ||
'--skip-dir', | ||
type=str, | ||
multiple=True, | ||
help='Skip a specific sub-directory from checks', | ||
) | ||
@click.argument('dirs', | ||
type=click.Path(exists=True, file_okay=False, resolve_path=True), nargs=-1) | ||
def main(log_level, extensions, dirs, known_failure, skip_dir): | ||
coloredlogs.install(level=__LOG_LEVELS__[log_level], | ||
fmt='%(asctime)s %(levelname)-7s %(message)s') | ||
|
||
if not dirs: | ||
logging.error("Please provide at least one directory to scan") | ||
sys.exit(1) | ||
|
||
if not extensions: | ||
logging.error("Need at least one extension") | ||
sys.exit(1) | ||
|
||
checker = OrphanChecker() | ||
for k in known_failure: | ||
checker.AddKnownFailure(k) | ||
|
||
# ensure all GN data is loaded | ||
for directory in dirs: | ||
for name in Path(directory).rglob("BUILD.gn"): | ||
checker.AppendGnData(name) | ||
|
||
skip_dir = set(skip_dir) | ||
|
||
# Go through all files and check for orphaned (if any) | ||
extensions = set(extensions) | ||
for directory in dirs: | ||
for path, dirnames, filenames in os.walk(directory): | ||
if any([s in path for s in skip_dir]): | ||
continue | ||
for f in filenames: | ||
full_path = Path(os.path.join(path, f)) | ||
if not full_path.suffix or full_path.suffix[1:] not in extensions: | ||
continue | ||
checker.Check(directory, full_path) | ||
|
||
if checker.failures: | ||
logging.warning("%d files not known to GN (%d fatal)", checker.failures, checker.fatal_failures) | ||
|
||
if checker.known_failures != checker.found_failures: | ||
not_failing = checker.known_failures - checker.found_failures | ||
logging.warning("NOTE: %d failures are not found anymore:", len(not_failing)) | ||
for name in not_failing: | ||
logging.warning(" - %s", name) | ||
# Assume this is fatal - remove some of the "known-failing" should be easy. | ||
# This forces scripts to always be correct and not accumulate bad input. | ||
sys.exit(1) | ||
|
||
if checker.fatal_failures > 0: | ||
sys.exit(1) | ||
|
||
|
||
if __name__ == '__main__': | ||
main(auto_envvar_prefix='CHIP') |