# This Source Code Form is subject to the terms of the Mozilla Public # License, v. 2.0. If a copy of the MPL was not distributed with this # file, You can obtain one at http://mozilla.org/MPL/2.0/. import json import os import re import signal import six from mozboot.util import get_tools_dir from mozfile import which from mozlint import result from mozlint.pathutils import get_ancestors_by_name from mozprocess import ProcessHandler from packaging.version import Version CLIPPY_WRONG_VERSION = """ Clippy is not installed or an older version was detected. Please make sure clippy is installed and up to date. The minimum required version is {version}. To install: $ rustup component add clippy To update: $ rustup update Also ensure 'cargo' is on your $PATH. """.strip() CARGO_NOT_FOUND = """ Could not find cargo! Install cargo. And make sure that it is in the PATH """.strip() def parse_issues(log, config, issues, base_path, onlyIn): results = [] if onlyIn: onlyIn = os.path.normcase(os.path.normpath(onlyIn)) for issue in issues: try: detail = json.loads(six.ensure_text(issue)) if "message" in detail: p = detail["target"]["src_path"] detail = detail["message"] if "level" in detail: if ( detail["level"] == "error" or detail["level"] == "failure-note" ) and not detail["code"]: log.debug( "Error outside of clippy." "This means that the build failed. Therefore, skipping this" ) log.debug("File = {} / Detail = {}".format(p, detail)) continue # We are in a clippy warning if len(detail["spans"]) == 0: # For some reason, at the end of the summary, we can # get the following line # {'rendered': 'warning: 5 warnings emitted\n\n', 'children': # [], 'code': None, 'level': 'warning', 'message': # '5 warnings emitted', 'spans': []} # if this is the case, skip it log.debug( "Skipping the summary line {} for file {}".format(detail, p) ) continue l = detail["spans"][0] p = os.path.join(base_path, l["file_name"]) if onlyIn and onlyIn not in os.path.normcase(os.path.normpath(p)): # Case when we have a .rs in the include list in the yaml file log.debug( "{} is not part of the list of files '{}'".format(p, onlyIn) ) continue res = { "path": p, "level": detail["level"], "lineno": l["line_start"], "column": l["column_start"], "message": detail["message"], "hint": detail["rendered"], "rule": detail["code"]["code"], "lineoffset": l["line_end"] - l["line_start"], } results.append(result.from_config(config, **res)) except json.decoder.JSONDecodeError: log.debug("Could not parse the output:") log.debug("clippy output: {}".format(issue)) continue return results def get_cargo_binary(log): """ Returns the path of the first rustfmt binary available if not found returns None """ cargo_home = os.environ.get("CARGO_HOME") if cargo_home: log.debug("Found CARGO_HOME in {}".format(cargo_home)) cargo_bin = os.path.join(cargo_home, "bin", "cargo") if os.path.exists(cargo_bin): return cargo_bin log.debug("Did not find {} in CARGO_HOME".format(cargo_bin)) return None rust_path = os.path.join(get_tools_dir(), "rustc", "bin") return which("cargo", path=os.pathsep.join([rust_path, os.environ["PATH"]])) def get_clippy_version(log, cargo): """ Check if we are running the deprecated clippy """ output = run_cargo_command( log, [cargo, "clippy", "--version"], universal_newlines=True ) version = re.findall(r"(\d+-\d+-\d+)", output[0]) if not version: return False version = Version(version[0].replace("-", ".")) log.debug("Found version: {}".format(version)) return version class clippyProcess(ProcessHandler): def __init__(self, *args, **kwargs): kwargs["stream"] = False ProcessHandler.__init__(self, *args, **kwargs) def run(self, *args, **kwargs): orig = signal.signal(signal.SIGINT, signal.SIG_IGN) ProcessHandler.run(self, *args, **kwargs) signal.signal(signal.SIGINT, orig) def run_cargo_command(log, cmd, **kwargs): log.debug("Command: {}".format(cmd)) env = os.environ.copy() # Cargo doesn't find cargo-clippy on its own if it's not in `$PATH` when # `$CARGO_HOME` is not set and cargo is not in `~/.cargo`. env["PATH"] = os.pathsep.join([os.path.dirname(cmd[0]), os.environ["PATH"]]) proc = clippyProcess(cmd, env=env, **kwargs) proc.run() try: proc.wait() except KeyboardInterrupt: proc.kill() return proc.output def lint(paths, config, fix=None, **lintargs): log = lintargs["log"] cargo = get_cargo_binary(log) if not cargo: print(CARGO_NOT_FOUND) if "MOZ_AUTOMATION" in os.environ: return 1 return [] min_version_str = config.get("min_clippy_version") min_version = Version(min_version_str) actual_version = get_clippy_version(log, cargo) log.debug( "Found version: {}. Minimum expected version: {}".format( actual_version, min_version ) ) if not actual_version or actual_version < min_version: print(CLIPPY_WRONG_VERSION.format(version=min_version_str)) return 1 cmd_args_clean = [cargo] cmd_args_clean.append("clean") cmd_args_common = ["--manifest-path"] cmd_args_clippy = [cargo] cmd_args_clippy += [ "clippy", "--message-format=json", ] if fix: cmd_args_clippy += ["--fix"] lock_files_to_delete = [] for p in paths: lock_file = os.path.join(p, "Cargo.lock") if not os.path.exists(lock_file): lock_files_to_delete.append(lock_file) results = [] for p in paths: # Quick sanity check of the paths if p.endswith("Cargo.toml"): print("Error: expects a directory or a rs file") print("Found {}".format(p)) return 1 for p in paths: onlyIn = [] path_conf = p log.debug("Path = {}".format(p)) if os.path.isfile(p): # We are dealing with a file. We remove the filename from the path # to find the closest Cargo file # We also store the name of the file to be able to filter out other # files built by the cargo p = os.path.dirname(p) onlyIn = path_conf elif os.path.isdir(p): # Sometimes, clippy reports issues from other crates # Make sure that we don't display that either onlyIn = p cargo_files = get_ancestors_by_name("Cargo.toml", p, lintargs["root"]) p = cargo_files[0] log.debug("Path translated to = {}".format(p)) # Needs clean because of https://github.com/rust-lang/rust-clippy/issues/2604 clean_command = cmd_args_clean + cmd_args_common + [p] run_cargo_command(log, clean_command) # Create the actual clippy command base_command = cmd_args_clippy + cmd_args_common + [p] output = run_cargo_command(log, base_command) # Remove build artifacts created by clippy run_cargo_command(log, clean_command) # Source paths in clippy spans, when they are relative, are relative to the # workspace if there is one. for cargo_toml in cargo_files: with open(cargo_toml) as fh: if "[workspace]" in fh.read(): p = cargo_toml break results += parse_issues(log, config, output, os.path.dirname(p), onlyIn) # Remove Cargo.lock files created by clippy for lock_file in lock_files_to_delete: if os.path.exists(lock_file): os.remove(lock_file) return sorted(results, key=lambda issue: issue.path)