#!/usr/bin/env python """ Upload code coverage reports to codecov.io. Multiple coverage files from multiple languages are accepted and aggregated after upload. Python coverage, as well as PowerShell and Python stubs can all be uploaded. """ import argparse import dataclasses import pathlib import shutil import subprocess import tempfile import typing as t import urllib.request @dataclasses.dataclass(frozen=True) class CoverageFile: name: str path: pathlib.Path flags: t.List[str] @dataclasses.dataclass(frozen=True) class Args: dry_run: bool path: pathlib.Path def parse_args() -> Args: parser = argparse.ArgumentParser() parser.add_argument('-n', '--dry-run', action='store_true') parser.add_argument('path', type=pathlib.Path) args = parser.parse_args() # Store arguments in a typed dataclass fields = dataclasses.fields(Args) kwargs = {field.name: getattr(args, field.name) for field in fields} return Args(**kwargs) def process_files(directory: pathlib.Path) -> t.Tuple[CoverageFile, ...]: processed = [] for file in directory.joinpath('reports').glob('coverage*.xml'): name = file.stem.replace('coverage=', '') # Get flags from name flags = name.replace('-powershell', '').split('=') # Drop '-powershell' suffix flags = [flag if not flag.startswith('stub') else flag.split('-')[0] for flag in flags] # Remove "-01" from stub files processed.append(CoverageFile(name, file, flags)) return tuple(processed) def upload_files(codecov_bin: pathlib.Path, files: t.Tuple[CoverageFile, ...], dry_run: bool = False) -> None: for file in files: cmd = [ str(codecov_bin), '--name', file.name, '--file', str(file.path), ] for flag in file.flags: cmd.extend(['--flags', flag]) if dry_run: print(f'DRY-RUN: Would run command: {cmd}') continue subprocess.run(cmd, check=True) def download_file(url: str, dest: pathlib.Path, flags: int, dry_run: bool = False) -> None: if dry_run: print(f'DRY-RUN: Would download {url} to {dest} and set mode to {flags:o}') return with urllib.request.urlopen(url) as resp: with dest.open('w+b') as f: # Read data in chunks rather than all at once shutil.copyfileobj(resp, f, 64 * 1024) dest.chmod(flags) def main(): args = parse_args() url = 'https://ansible-ci-files.s3.amazonaws.com/codecov/linux/codecov' with tempfile.TemporaryDirectory(prefix='codecov-') as tmpdir: codecov_bin = pathlib.Path(tmpdir) / 'codecov' download_file(url, codecov_bin, 0o755, args.dry_run) files = process_files(args.path) upload_files(codecov_bin, files, args.dry_run) if __name__ == '__main__': main()