2023-07-22 11:27:49 +03:00
|
|
|
import os
|
|
|
|
import sys
|
|
|
|
import argparse
|
|
|
|
from urllib.parse import urlparse
|
|
|
|
import xml.etree.ElementTree as ET
|
|
|
|
|
|
|
|
parser = argparse.ArgumentParser(description='Extracts release feeds from roles')
|
|
|
|
parser.add_argument('root_dir', help='Root dir which to traverse recursively for defaults/main.yml roles files')
|
|
|
|
parser.add_argument('action', help='Pass "check" to list roles with missing feeds or "dump" to dump an OPML file')
|
|
|
|
args = parser.parse_args()
|
2024-03-31 12:51:14 +03:00
|
|
|
if args.action not in ['check', 'dump']:
|
2023-07-22 11:27:49 +03:00
|
|
|
sys.exit('Error: possible arguments are "check" or "dump"')
|
|
|
|
|
|
|
|
excluded_paths = [
|
|
|
|
# appservice-kakaotalk defines a Project URL, but that Gitea repository does not have an Atom/RSS feed.
|
|
|
|
# It doesn't have any tags anyway.
|
|
|
|
'./upstream/roles/custom/matrix-bridge-appservice-kakaotalk/defaults',
|
|
|
|
]
|
|
|
|
project_source_url_str = '# Project source code URL:'
|
|
|
|
|
|
|
|
def get_roles_files_from_dir(root_dir):
|
|
|
|
file_paths = []
|
|
|
|
for dir_name, sub_dur_list, file_list in os.walk(root_dir):
|
|
|
|
for file_name in file_list:
|
|
|
|
if not dir_name.endswith('defaults') or file_name != 'main.yml':
|
|
|
|
continue
|
|
|
|
if dir_name in excluded_paths:
|
|
|
|
continue
|
|
|
|
file_paths.append(os.path.join(dir_name, file_name))
|
|
|
|
return file_paths
|
|
|
|
|
|
|
|
def get_git_repos_from_files(file_paths, break_on_missing_repos=False):
|
|
|
|
git_repos = {}
|
|
|
|
missing_repos = []
|
|
|
|
|
|
|
|
for file in file_paths:
|
|
|
|
file_lines = open(file, 'r').readlines()
|
|
|
|
found_project_repo = False
|
|
|
|
for line in file_lines:
|
|
|
|
project_repo_val = ''
|
|
|
|
if project_source_url_str in line:
|
|
|
|
# extract the value from a line like this:
|
|
|
|
# Project source code URL: https://github.com/mautrix/signal
|
|
|
|
project_repo_val = line.split(project_source_url_str)[1].strip()
|
|
|
|
if not validate_url(project_repo_val):
|
|
|
|
print('Invalid url for line ', line)
|
|
|
|
break
|
|
|
|
if project_repo_val != '':
|
|
|
|
if file not in git_repos:
|
|
|
|
git_repos[file] = []
|
|
|
|
|
|
|
|
git_repos[file].append(project_repo_val)
|
|
|
|
found_project_repo = True
|
|
|
|
|
|
|
|
if not found_project_repo:
|
|
|
|
missing_repos.append(file)
|
|
|
|
|
|
|
|
if break_on_missing_repos and len(missing_repos) > 0:
|
|
|
|
print('Missing `{0}` comment for:\n{1}'.format(project_source_url_str, '\n'.join(missing_repos)))
|
|
|
|
|
|
|
|
return git_repos
|
|
|
|
|
|
|
|
def validate_url(text):
|
|
|
|
if text == '':
|
|
|
|
return False
|
|
|
|
try:
|
|
|
|
result = urlparse(text)
|
|
|
|
return all([result.scheme, result.netloc])
|
|
|
|
except:
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
def format_feeds_from_git_repos(git_repos):
|
2024-03-31 12:51:14 +03:00
|
|
|
feeds = {}
|
2023-07-22 11:27:49 +03:00
|
|
|
for role, git_repos in git_repos.items():
|
|
|
|
for idx, git_repo in enumerate(git_repos):
|
|
|
|
if 'github' in git_repo:
|
|
|
|
atomFilePath = git_repo.replace('.git', '') + '/releases.atom'
|
|
|
|
elif ('gitlab' in git_repo or 'mau.dev' in git_repo):
|
|
|
|
atomFilePath = git_repo.replace('.git', '') + '/-/tags?format=atom'
|
2024-03-31 12:51:14 +03:00
|
|
|
elif 'git.zx2c4.com' in git_repo: # cgit
|
2023-07-22 11:27:49 +03:00
|
|
|
atomFilePath = git_repo + '/atom/'
|
2024-03-31 12:51:14 +03:00
|
|
|
elif 'framagit.org' in git_repo: # gitlab
|
|
|
|
atomFilePath = git_repo.replace('.git', '') + '/-/tags?format=atom'
|
|
|
|
elif 'git.osgeo.org' in git_repo: # gitea
|
|
|
|
atomFilePath = git_repo.replace('.git', '') + '.atom'
|
|
|
|
elif 'dev.funkwhale.audio' in git_repo: # gitlab
|
|
|
|
atomFilePath = git_repo.replace('.git', '') + '/-/tags?format=atom'
|
2023-07-22 11:27:49 +03:00
|
|
|
else:
|
|
|
|
print('Unrecognized git repository: %s' % git_repo)
|
|
|
|
continue
|
|
|
|
|
|
|
|
role_name = role.split('/')[4]
|
|
|
|
if role_name == 'defaults':
|
|
|
|
role_name = role.split('/')[3]
|
|
|
|
role_name = role_name.removeprefix('matrix-bot-').removeprefix('matrix-bridge-').removeprefix('matrix-client-').removeprefix('matrix-')
|
|
|
|
if idx > 0:
|
|
|
|
# there is more than 1 project source code for this role
|
|
|
|
role_name += '-' + str(idx+1)
|
|
|
|
|
|
|
|
feeds[role_name] = {
|
|
|
|
'text': role_name,
|
|
|
|
'title': role_name,
|
|
|
|
'type': 'rss',
|
|
|
|
'htmlUrl': git_repo,
|
|
|
|
'xmlUrl': atomFilePath
|
|
|
|
}
|
|
|
|
|
|
|
|
feeds = {key: val for key, val in sorted(feeds.items(), key = lambda item: item[0])}
|
|
|
|
return feeds
|
|
|
|
|
|
|
|
def dump_opml_file_from_feeds(feeds):
|
|
|
|
tree = ET.ElementTree('tree')
|
|
|
|
|
|
|
|
opml = ET.Element('opml', {'version': '1.0'})
|
|
|
|
head = ET.SubElement(opml, 'head')
|
|
|
|
|
|
|
|
title = ET.SubElement(head, 'title')
|
|
|
|
title.text = 'Release feeds for roles'
|
|
|
|
|
|
|
|
body = ET.SubElement(opml, 'body')
|
|
|
|
for role, feed_dict in feeds.items():
|
|
|
|
outline = ET.SubElement(body, 'outline', feed_dict)
|
|
|
|
|
|
|
|
ET.indent(opml)
|
|
|
|
tree._setroot(opml)
|
|
|
|
file_name = 'releases.opml'
|
|
|
|
tree.write(file_name, encoding = 'UTF-8', xml_declaration = True)
|
|
|
|
print('Generated %s' % file_name)
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
file_paths = get_roles_files_from_dir(root_dir=args.root_dir)
|
|
|
|
break_on_missing = args.action == 'check'
|
|
|
|
git_repos = get_git_repos_from_files(file_paths=file_paths, break_on_missing_repos=break_on_missing)
|
|
|
|
feeds = format_feeds_from_git_repos(git_repos)
|
|
|
|
|
|
|
|
if args.action == 'dump':
|
|
|
|
dump_opml_file_from_feeds(feeds)
|