Source code for symforce.codegen.format_util

# ----------------------------------------------------------------------------
# SymForce - Copyright 2022, Skydio, Inc.
# This source code is under the Apache 2.0 license found in the LICENSE file.
# ----------------------------------------------------------------------------

import os
import shutil
import subprocess
from pathlib import Path

from ruff.__main__ import find_ruff_bin

from symforce import typing as T


[docs]def format_cpp(file_contents: str, filename: str) -> str: """ Autoformat a given C++ file using clang-format Args: file_contents: The unformatted contents of the file filename: A name that this file might have on disk; this does not have to be a real path, it's only used for clang-format to find the correct style file (by traversing upwards from this location) and to decide if an include is the corresponding .h file for a .cc file that's being formatted (this affects the include order) Returns: formatted_file_contents (str): The contents of the file after formatting """ try: import clang_format clang_format_path = str( Path(clang_format.__file__).parent / "data" / "bin" / "clang-format" ) except ImportError: clang_format_path = "clang-format" result = subprocess.run( [clang_format_path, f"-assume-filename={filename}"], input=file_contents, stdout=subprocess.PIPE, stderr=None, check=True, text=True, ) return result.stdout
_ruff_path: T.Optional[Path] = None def _find_ruff() -> Path: """ Find the ruff binary `find_ruff_bin` does not work in all environments, for example it does not work on debian when things are installed in `/usr/local/bin` and `sysconfig` only returns `/usr/bin`. Adding `shutil.which` should cover most cases, but not all, the better solution would require `ruff` putting the binary in `data` like `clang-format` does """ global _ruff_path # noqa: PLW0603 if _ruff_path is not None: return _ruff_path try: ruff = find_ruff_bin() except FileNotFoundError as ex: ruff = shutil.which("ruff") if ruff is None: raise FileNotFoundError("Could not find ruff") from ex _ruff_path = ruff return ruff
[docs]def format_py(file_contents: str, filename: str) -> str: """ Autoformat a given Python file using ruff Args: filename: A name that this file might have on disk; this does not have to be a real path, it's only used for ruff to find the correct style file (by traversing upwards from this location) """ result = subprocess.run( [_find_ruff(), "format", f"--stdin-filename={filename}", "-"], input=file_contents, stdout=subprocess.PIPE, check=True, # Disable the ruff cache. This is important for running in a hermetic context like a bazel # test, and shouldn't really hurt other use cases. If it does, we should work around this # differently. env=dict(os.environ, RUFF_NO_CACHE="true"), text=True, ) result = subprocess.run( [ _find_ruff(), "check", "--select=I", "--fix", "--quiet", f"--stdin-filename={filename}", "-", ], input=result.stdout, stdout=subprocess.PIPE, check=True, # Disable the ruff cache. This is important for running in a hermetic context like a bazel # test, and shouldn't really hurt other use cases. If it does, we should work around this # differently. env=dict(os.environ, RUFF_NO_CACHE="true"), text=True, ) return result.stdout
[docs]def format_py_dir(dirname: T.Openable) -> None: """ Autoformat python files in a directory (recursively) in-place """ subprocess.run( [_find_ruff(), "format", dirname], check=True, # Disable the ruff cache. This is important for running in a hermetic context like a bazel # test, and shouldn't really hurt other use cases. If it does, we should work around this # differently. env=dict(os.environ, RUFF_NO_CACHE="true"), text=True, )
_rustfmt_path: T.Optional[Path] = None def _find_rustfmt() -> Path: """ Find the rustfmt binary """ global _rustfmt_path # noqa: PLW0603 if _rustfmt_path is not None: return _rustfmt_path rustfmt = shutil.which("rustfmt") if rustfmt is None: raise FileNotFoundError("Could not find rustfmt") # Ignore the type because mypy can't reason about the fact that we just checked that rustfmt # is not None. _rustfmt_path = Path(rustfmt) return _rustfmt_path
[docs]def format_rust(file_contents: str, filename: str) -> str: """ Autoformat a given Rust file using rustfmt. Args: filename: A name that this file might have on disk; this does not have to be a real path, it's only used for ruff to find the correct style file (by traversing upwards from this location) """ result = subprocess.run( [_find_rustfmt()], input=file_contents, stdout=subprocess.PIPE, check=True, text=True, ) return result.stdout