"""Check that executable text files have a shebang.""" from __future__ import annotations import argparse import shlex import sys from collections.abc import Generator from collections.abc import Sequence from typing import NamedTuple from pre_commit_hooks.util import cmd_output from pre_commit_hooks.util import zsplit EXECUTABLE_VALUES = frozenset(('1', '3', '5', '7')) def check_executables(paths: list[str]) -> int: fs_tracks_executable_bit = cmd_output( 'git', 'config', 'core.fileMode', retcode=None, ).strip() if fs_tracks_executable_bit == 'false': # pragma: win32 cover return _check_git_filemode(paths) else: # pragma: win32 no cover retv = 0 for path in paths: if not has_shebang(path): _message(path) retv = 1 return retv class GitLsFile(NamedTuple): mode: str filename: str def git_ls_files(paths: Sequence[str]) -> Generator[GitLsFile]: outs = cmd_output('git', 'ls-files', '-z', '--stage', '--', *paths) for out in zsplit(outs): metadata, filename = out.split('\t') mode, _, _ = metadata.split() yield GitLsFile(mode, filename) def _check_git_filemode(paths: Sequence[str]) -> int: seen: set[str] = set() for ls_file in git_ls_files(paths): is_executable = any(b in EXECUTABLE_VALUES for b in ls_file.mode[-3:]) if is_executable and not has_shebang(ls_file.filename): _message(ls_file.filename) seen.add(ls_file.filename) return int(bool(seen)) def has_shebang(path: str) -> int: with open(path, 'rb') as f: first_bytes = f.read(2) return first_bytes == b'#!' def _message(path: str) -> None: print( f'{path}: marked executable but has no (or invalid) shebang!\n' f" If it isn't supposed to be executable, try: " f'`chmod -x {shlex.quote(path)}`\n' f' If on Windows, you may also need to: ' f'`git add --chmod=-x {shlex.quote(path)}`\n' f' If it is supposed to be executable, double-check its shebang.', file=sys.stderr, ) def main(argv: Sequence[str] | None = None) -> int: parser = argparse.ArgumentParser(description=__doc__) parser.add_argument('filenames', nargs='*') args = parser.parse_args(argv) return check_executables(args.filenames) if __name__ == '__main__': raise SystemExit(main())