rspec/rule_coverage/coverage.py

122 lines
3.9 KiB
Python
Raw Normal View History

import os
2020-07-06 17:54:11 +02:00
import sys
import argparse
import json
2020-07-06 17:54:11 +02:00
from git import Repo
from git import Git
from pathlib import Path
repos=['sonar-abap','sonar-cpp','sonar-cobol','sonar-dotnet','sonar-css','sonar-flex','slang-enterprise','sonar-java','SonarJS','sonar-php','sonar-pli','sonar-plsql','sonar-python','sonar-rpg','sonar-swift','sonar-tsql','sonar-vb','sonar-html','sonar-xml','sonar-kotlin', 'sonar-secrets', 'sonar-security']
def load_json(file):
with open(file) as json_file:
return json.load(json_file)
# repoAndVersion uniquely identifies the analyzer and version implementing
# the rule for the given languages.
# Rule implementations for some langauges are spread across multiple repositories
# for example sonar-java and sonar-security for Java.
# We use repoAndVersion to avoid confusion between version of different analyzers.
def get_rules_json(path, languages, repoAndVersion):
print(f"Getting rules from {os.getcwd()} {path} for {repoAndVersion}")
for filename in os.listdir(path):
if filename.endswith(".json") and not filename.startswith("Sonar_way"):
rule=load_json(os.path.join(path, filename))
dump_rule(filename[:-5], rule, languages, repoAndVersion)
else:
continue
def dump_rule(name, rule, languages, repoAndVersion):
if "compatibleLanguages" in rule:
for language in rule['compatibleLanguages']:
store_rule(name, rule, language, repoAndVersion)
else:
for language in languages:
store_rule(name, rule, language, repoAndVersion)
def store_rule(name, rule, language, repoAndVersion):
2020-07-06 17:54:11 +02:00
if language not in rules:
print(f"create entry for {language}")
2020-07-06 17:54:11 +02:00
rules[language] = {}
2020-07-09 16:11:40 +02:00
if '_' in name:
name=name[:name.find('_')]
2020-07-06 17:54:11 +02:00
if name not in rules[language]:
rules[language][name] = repoAndVersion
def dump_rules(repo, version):
for sp_file in Path('.').rglob('sonarpedia.json'):
print(sp_file)
sonarpedia_path=sp_file.parents[0]
2020-07-06 17:54:11 +02:00
sonarpedia = load_json(sp_file)
path=str(sonarpedia_path) + '/' + sonarpedia['rules-metadata-path'].replace('\\', '/')
2020-07-06 17:54:11 +02:00
languages=sonarpedia['languages']
get_rules_json(path, languages, repo + ' ' + version)
2020-07-06 17:54:11 +02:00
with open(f"../{rules_filename}", 'w') as outfile:
json.dump(rules, outfile, indent=2, sort_keys=True)
def checkout_repo(repo):
token=os.getenv('GITHUB_TOKEN')
if not token:
git_url=f"git@github.com:SonarSource/{repo}"
else:
2021-05-31 17:15:15 +02:00
git_url=f"https://{token}@github.com/SonarSource/{repo}"
2020-07-06 17:54:11 +02:00
git_repo=None
g=Git(repo)
if not os.path.exists(repo):
return Repo.clone_from(git_url, repo)
2020-07-06 17:54:11 +02:00
else:
return Repo(repo)
def scan_all_versions(repo):
git_repo = checkout_repo(repo)
versions = [tag.name for tag in git_repo.tags]
for version in versions:
if not '-' in version:
print(f"{repo} {version}")
scan_version(repo, version)
scan_version(repo, 'master')
def scan_version(repo, version):
r = checkout_repo(repo)
g = Git(repo)
os.chdir(repo)
try:
r.head.reference = r.commit(version)
r.head.reset(index=True, working_tree=True)
2020-07-07 10:12:56 +02:00
g.checkout(version)
dump_rules(repo, version)
except Exception as e:
print(f"{repo} {version} checkout failed, resetting and cleaning: {e}")
os.chdir('..')
def main():
parser = argparse.ArgumentParser(description='rules coverage')
parser.add_argument('command', nargs='+', help='see code for help')
args = parser.parse_args()
global rules
global rules_filename
rules_filename='covered_rules.json'
if os.path.exists(rules_filename):
rules=load_json(rules_filename)
else:
rules={}
2020-07-07 11:00:33 +02:00
if args.command[0] == "batchall":
print(f"batch mode for {repos}")
for repo in repos:
scan_all_versions(repo)
2020-07-07 11:00:33 +02:00
elif args.command[0] == "batch":
repo=args.command[1]
print(f"batch mode for {repo}")
scan_all_versions(repo)
else:
2020-07-07 11:00:33 +02:00
repo=args.command[0]
2020-07-07 10:12:56 +02:00
version=args.command[1]
2020-07-07 11:00:33 +02:00
print(f"checking {repo} version {version}")
scan_version(repo, version)
if __name__ == '__main__':
main()