2021-03-14 17:01:17 +00:00
|
|
|
#!/usr/bin/env nix-shell
|
2021-07-06 17:28:04 +01:00
|
|
|
#! nix-shell -i python3 -p bundix bundler nix-update nix-universal-prefetch python3 python3Packages.requests python3Packages.click python3Packages.click-log
|
2021-03-14 17:01:17 +00:00
|
|
|
|
|
|
|
import click
|
|
|
|
import click_log
|
|
|
|
import shutil
|
|
|
|
import tempfile
|
|
|
|
import re
|
|
|
|
import logging
|
|
|
|
import subprocess
|
2021-07-06 17:28:04 +01:00
|
|
|
import os
|
|
|
|
import stat
|
|
|
|
import json
|
|
|
|
import requests
|
2021-08-13 17:42:56 +01:00
|
|
|
import textwrap
|
2021-03-14 17:01:17 +00:00
|
|
|
from distutils.version import LooseVersion
|
2021-07-06 17:28:04 +01:00
|
|
|
from pathlib import Path
|
2021-03-14 17:01:17 +00:00
|
|
|
from typing import Iterable
|
|
|
|
|
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
class DiscourseRepo:
|
|
|
|
version_regex = re.compile(r'^v\d+\.\d+\.\d+$')
|
2021-07-06 17:28:04 +01:00
|
|
|
_latest_commit_sha = None
|
|
|
|
|
2021-03-14 17:01:17 +00:00
|
|
|
def __init__(self, owner: str = 'discourse', repo: str = 'discourse'):
|
|
|
|
self.owner = owner
|
|
|
|
self.repo = repo
|
|
|
|
|
|
|
|
@property
|
|
|
|
def tags(self) -> Iterable[str]:
|
|
|
|
r = requests.get(f'https://api.github.com/repos/{self.owner}/{self.repo}/git/refs/tags').json()
|
|
|
|
tags = [x['ref'].replace('refs/tags/', '') for x in r]
|
|
|
|
|
|
|
|
# filter out versions not matching version_regex
|
|
|
|
versions = list(filter(self.version_regex.match, tags))
|
|
|
|
|
|
|
|
# sort, but ignore v for sorting comparisons
|
|
|
|
versions.sort(key=lambda x: LooseVersion(x.replace('v', '')), reverse=True)
|
|
|
|
return versions
|
|
|
|
|
2021-07-06 17:28:04 +01:00
|
|
|
@property
|
|
|
|
def latest_commit_sha(self) -> str:
|
|
|
|
if self._latest_commit_sha is None:
|
|
|
|
r = requests.get(f'https://api.github.com/repos/{self.owner}/{self.repo}/commits?per_page=1')
|
|
|
|
r.raise_for_status()
|
|
|
|
self._latest_commit_sha = r.json()[0]['sha']
|
|
|
|
|
|
|
|
return self._latest_commit_sha
|
|
|
|
|
2021-03-14 17:01:17 +00:00
|
|
|
@staticmethod
|
|
|
|
def rev2version(tag: str) -> str:
|
|
|
|
"""
|
|
|
|
normalize a tag to a version number.
|
|
|
|
This obviously isn't very smart if we don't pass something that looks like a tag
|
|
|
|
:param tag: the tag to normalize
|
|
|
|
:return: a normalized version number
|
|
|
|
"""
|
|
|
|
# strip v prefix
|
|
|
|
return re.sub(r'^v', '', tag)
|
|
|
|
|
|
|
|
def get_file(self, filepath, rev):
|
|
|
|
"""returns file contents at a given rev :param filepath: the path to
|
|
|
|
the file, relative to the repo root :param rev: the rev to
|
|
|
|
fetch at :return:
|
|
|
|
|
|
|
|
"""
|
|
|
|
return requests.get(f'https://raw.githubusercontent.com/{self.owner}/{self.repo}/{rev}/{filepath}').text
|
|
|
|
|
|
|
|
|
|
|
|
def _call_nix_update(pkg, version):
|
|
|
|
"""calls nix-update from nixpkgs root dir"""
|
2021-07-06 17:28:04 +01:00
|
|
|
nixpkgs_path = Path(__file__).parent / '../../../../'
|
2021-03-14 17:01:17 +00:00
|
|
|
return subprocess.check_output(['nix-update', pkg, '--version', version], cwd=nixpkgs_path)
|
|
|
|
|
|
|
|
|
2021-07-06 17:28:04 +01:00
|
|
|
def _nix_eval(expr: str):
|
|
|
|
nixpkgs_path = Path(__file__).parent / '../../../../'
|
2021-08-13 17:42:56 +01:00
|
|
|
try:
|
|
|
|
output = subprocess.check_output(['nix', 'eval', '--json', f'(with import {nixpkgs_path} {{}}; {expr})'], text=True)
|
|
|
|
except subprocess.CalledProcessError:
|
|
|
|
return None
|
|
|
|
return json.loads(output)
|
2021-07-06 17:28:04 +01:00
|
|
|
|
|
|
|
|
2021-03-14 17:01:17 +00:00
|
|
|
def _get_current_package_version(pkg: str):
|
2021-07-06 17:28:04 +01:00
|
|
|
return _nix_eval(f'{pkg}.version')
|
2021-03-14 17:01:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
def _diff_file(filepath: str, old_version: str, new_version: str):
|
|
|
|
repo = DiscourseRepo()
|
|
|
|
|
2021-07-06 17:28:04 +01:00
|
|
|
current_dir = Path(__file__).parent
|
2021-03-14 17:01:17 +00:00
|
|
|
|
|
|
|
old = repo.get_file(filepath, 'v' + old_version)
|
|
|
|
new = repo.get_file(filepath, 'v' + new_version)
|
|
|
|
|
|
|
|
if old == new:
|
|
|
|
click.secho(f'{filepath} is unchanged', fg='green')
|
|
|
|
return
|
|
|
|
|
|
|
|
with tempfile.NamedTemporaryFile(mode='w') as o, tempfile.NamedTemporaryFile(mode='w') as n:
|
|
|
|
o.write(old), n.write(new)
|
|
|
|
width = shutil.get_terminal_size((80, 20)).columns
|
|
|
|
diff_proc = subprocess.run(
|
|
|
|
['diff', '--color=always', f'--width={width}', '-y', o.name, n.name],
|
|
|
|
stdout=subprocess.PIPE,
|
|
|
|
cwd=current_dir,
|
|
|
|
text=True
|
|
|
|
)
|
|
|
|
|
|
|
|
click.secho(f'Diff for {filepath} ({old_version} -> {new_version}):', fg='bright_blue', bold=True)
|
|
|
|
click.echo(diff_proc.stdout + '\n')
|
|
|
|
return
|
|
|
|
|
|
|
|
|
2021-08-13 18:00:23 +01:00
|
|
|
def _remove_platforms(rubyenv_dir: Path):
|
|
|
|
for platform in ['arm64-darwin-20', 'x86_64-darwin-18',
|
|
|
|
'x86_64-darwin-19', 'x86_64-darwin-20',
|
|
|
|
'x86_64-linux']:
|
|
|
|
with open(rubyenv_dir / 'Gemfile.lock', 'r') as f:
|
|
|
|
for line in f:
|
|
|
|
if platform in line:
|
|
|
|
subprocess.check_output(
|
|
|
|
['bundle', 'lock', '--remove-platform', platform], cwd=rubyenv_dir)
|
|
|
|
break
|
|
|
|
|
|
|
|
|
2021-03-14 17:01:17 +00:00
|
|
|
@click_log.simple_verbosity_option(logger)
|
|
|
|
|
|
|
|
|
|
|
|
@click.group()
|
|
|
|
def cli():
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
@cli.command()
|
|
|
|
@click.argument('rev', default='latest')
|
|
|
|
@click.option('--reverse/--no-reverse', default=False, help='Print diffs from REV to current.')
|
|
|
|
def print_diffs(rev, reverse):
|
|
|
|
"""Print out diffs for files used as templates for the NixOS module.
|
|
|
|
|
|
|
|
The current package version found in the nixpkgs worktree the
|
|
|
|
script is run from will be used to download the "from" file and
|
|
|
|
REV used to download the "to" file for the diff, unless the
|
|
|
|
'--reverse' flag is specified.
|
|
|
|
|
|
|
|
REV should be the git rev to find changes in ('vX.Y.Z') or
|
|
|
|
'latest'; defaults to 'latest'.
|
|
|
|
|
|
|
|
"""
|
|
|
|
if rev == 'latest':
|
|
|
|
repo = DiscourseRepo()
|
|
|
|
rev = repo.tags[0]
|
|
|
|
|
|
|
|
old_version = _get_current_package_version('discourse')
|
|
|
|
new_version = DiscourseRepo.rev2version(rev)
|
|
|
|
|
|
|
|
if reverse:
|
|
|
|
old_version, new_version = new_version, old_version
|
|
|
|
|
|
|
|
for f in ['config/nginx.sample.conf', 'config/discourse_defaults.conf']:
|
|
|
|
_diff_file(f, old_version, new_version)
|
|
|
|
|
|
|
|
|
|
|
|
@cli.command()
|
|
|
|
@click.argument('rev', default='latest')
|
|
|
|
def update(rev):
|
|
|
|
"""Update gem files and version.
|
|
|
|
|
|
|
|
REV should be the git rev to update to ('vX.Y.Z') or 'latest';
|
|
|
|
defaults to 'latest'.
|
|
|
|
|
|
|
|
"""
|
|
|
|
repo = DiscourseRepo()
|
|
|
|
|
|
|
|
if rev == 'latest':
|
|
|
|
rev = repo.tags[0]
|
|
|
|
logger.debug(f"Using rev {rev}")
|
|
|
|
|
|
|
|
version = repo.rev2version(rev)
|
|
|
|
logger.debug(f"Using version {version}")
|
|
|
|
|
2021-07-06 17:28:04 +01:00
|
|
|
rubyenv_dir = Path(__file__).parent / "rubyEnv"
|
2021-03-14 17:01:17 +00:00
|
|
|
|
|
|
|
for fn in ['Gemfile.lock', 'Gemfile']:
|
|
|
|
with open(rubyenv_dir / fn, 'w') as f:
|
|
|
|
f.write(repo.get_file(fn, rev))
|
|
|
|
|
|
|
|
subprocess.check_output(['bundle', 'lock'], cwd=rubyenv_dir)
|
2021-08-13 18:00:23 +01:00
|
|
|
_remove_platforms(rubyenv_dir)
|
2021-03-14 17:01:17 +00:00
|
|
|
subprocess.check_output(['bundix'], cwd=rubyenv_dir)
|
|
|
|
|
|
|
|
_call_nix_update('discourse', repo.rev2version(rev))
|
|
|
|
|
2021-07-06 17:28:04 +01:00
|
|
|
@cli.command()
|
|
|
|
def update_plugins():
|
|
|
|
"""Update plugins to their latest revision.
|
|
|
|
|
|
|
|
"""
|
|
|
|
plugins = [
|
2021-09-28 23:07:12 +01:00
|
|
|
{'name': 'discourse-assign'},
|
2021-08-05 18:16:15 +01:00
|
|
|
{'name': 'discourse-calendar'},
|
2021-07-06 17:28:04 +01:00
|
|
|
{'name': 'discourse-canned-replies'},
|
2021-09-28 23:09:00 +01:00
|
|
|
{'name': 'discourse-chat-integration'},
|
2021-08-05 18:16:15 +01:00
|
|
|
{'name': 'discourse-checklist'},
|
|
|
|
{'name': 'discourse-data-explorer'},
|
2021-09-28 23:13:47 +01:00
|
|
|
{'name': 'discourse-docs'},
|
2021-07-06 17:28:04 +01:00
|
|
|
{'name': 'discourse-github'},
|
2021-08-03 00:07:31 +01:00
|
|
|
{'name': 'discourse-ldap-auth', 'owner': 'jonmbake'},
|
2021-07-06 17:28:04 +01:00
|
|
|
{'name': 'discourse-math'},
|
2021-08-05 18:16:15 +01:00
|
|
|
{'name': 'discourse-migratepassword', 'owner': 'discoursehosting'},
|
2021-09-28 23:16:02 +01:00
|
|
|
# We can't update this automatically at the moment because the plugin.rb
|
|
|
|
# tries to load a version number which breaks bundler called by this script.
|
|
|
|
# {'name': 'discourse-prometheus'},
|
2021-09-28 23:16:47 +01:00
|
|
|
{'name': 'discourse-saved-searches'},
|
2021-07-06 17:28:04 +01:00
|
|
|
{'name': 'discourse-solved'},
|
|
|
|
{'name': 'discourse-spoiler-alert'},
|
|
|
|
{'name': 'discourse-yearly-review'},
|
|
|
|
]
|
|
|
|
|
|
|
|
for plugin in plugins:
|
|
|
|
fetcher = plugin.get('fetcher') or "fetchFromGitHub"
|
|
|
|
owner = plugin.get('owner') or "discourse"
|
|
|
|
name = plugin.get('name')
|
|
|
|
repo_name = plugin.get('repo_name') or name
|
|
|
|
|
|
|
|
repo = DiscourseRepo(owner=owner, repo=repo_name)
|
2021-08-13 17:42:56 +01:00
|
|
|
|
|
|
|
filename = _nix_eval(f'builtins.unsafeGetAttrPos "src" discourse.plugins.{name}')
|
|
|
|
if filename is None:
|
|
|
|
filename = Path(__file__).parent / 'plugins' / name / 'default.nix'
|
|
|
|
filename.parent.mkdir()
|
|
|
|
|
|
|
|
has_ruby_deps = False
|
|
|
|
for line in repo.get_file('plugin.rb', repo.latest_commit_sha).splitlines():
|
|
|
|
if 'gem ' in line:
|
|
|
|
has_ruby_deps = True
|
|
|
|
break
|
|
|
|
|
|
|
|
with open(filename, 'w') as f:
|
|
|
|
f.write(textwrap.dedent(f"""
|
|
|
|
{{ lib, mkDiscoursePlugin, fetchFromGitHub }}:
|
|
|
|
|
|
|
|
mkDiscoursePlugin {{
|
|
|
|
name = "{name}";"""[1:] + ("""
|
|
|
|
bundlerEnvArgs.gemdir = ./.;""" if has_ruby_deps else "") + f"""
|
|
|
|
src = {fetcher} {{
|
|
|
|
owner = "{owner}";
|
|
|
|
repo = "{repo_name}";
|
|
|
|
rev = "replace-with-git-rev";
|
|
|
|
sha256 = "replace-with-sha256";
|
|
|
|
}};
|
|
|
|
meta = with lib; {{
|
|
|
|
homepage = "";
|
|
|
|
maintainers = with maintainers; [ ];
|
|
|
|
license = licenses.mit; # change to the correct license!
|
|
|
|
description = "";
|
|
|
|
}};
|
|
|
|
}}"""))
|
|
|
|
|
|
|
|
all_plugins_filename = Path(__file__).parent / 'plugins' / 'all-plugins.nix'
|
|
|
|
with open(all_plugins_filename, 'r+') as f:
|
|
|
|
content = f.read()
|
|
|
|
pos = -1
|
|
|
|
while content[pos] != '}':
|
|
|
|
pos -= 1
|
|
|
|
content = content[:pos] + f' {name} = callPackage ./{name} {{}};' + os.linesep + content[pos:]
|
|
|
|
f.seek(0)
|
|
|
|
f.write(content)
|
|
|
|
f.truncate()
|
|
|
|
|
|
|
|
else:
|
|
|
|
filename = filename['file']
|
|
|
|
|
2021-07-06 17:28:04 +01:00
|
|
|
prev_commit_sha = _nix_eval(f'discourse.plugins.{name}.src.rev')
|
|
|
|
|
|
|
|
if prev_commit_sha == repo.latest_commit_sha:
|
|
|
|
click.echo(f'Plugin {name} is already at the latest revision')
|
|
|
|
continue
|
|
|
|
|
|
|
|
prev_hash = _nix_eval(f'discourse.plugins.{name}.src.outputHash')
|
|
|
|
new_hash = subprocess.check_output([
|
|
|
|
'nix-universal-prefetch', fetcher,
|
|
|
|
'--owner', owner,
|
|
|
|
'--repo', repo_name,
|
|
|
|
'--rev', repo.latest_commit_sha,
|
|
|
|
], text=True).strip("\n")
|
|
|
|
|
|
|
|
click.echo(f"Update {name}, {prev_commit_sha} -> {repo.latest_commit_sha} in {filename}")
|
|
|
|
|
|
|
|
with open(filename, 'r+') as f:
|
|
|
|
content = f.read()
|
|
|
|
content = content.replace(prev_commit_sha, repo.latest_commit_sha)
|
|
|
|
content = content.replace(prev_hash, new_hash)
|
|
|
|
f.seek(0)
|
|
|
|
f.write(content)
|
|
|
|
f.truncate()
|
|
|
|
|
|
|
|
rubyenv_dir = Path(filename).parent
|
|
|
|
gemfile = rubyenv_dir / "Gemfile"
|
|
|
|
gemfile_text = ''
|
|
|
|
for line in repo.get_file('plugin.rb', repo.latest_commit_sha).splitlines():
|
|
|
|
if 'gem ' in line:
|
|
|
|
gemfile_text = gemfile_text + line + os.linesep
|
|
|
|
|
|
|
|
if len(gemfile_text) > 0:
|
|
|
|
if os.path.isfile(gemfile):
|
|
|
|
os.remove(gemfile)
|
|
|
|
|
|
|
|
subprocess.check_output(['bundle', 'init'], cwd=rubyenv_dir)
|
|
|
|
os.chmod(gemfile, stat.S_IREAD | stat.S_IWRITE | stat.S_IRGRP | stat.S_IROTH)
|
|
|
|
|
|
|
|
with open(gemfile, 'a') as f:
|
|
|
|
f.write(gemfile_text)
|
|
|
|
|
2021-08-13 18:00:23 +01:00
|
|
|
subprocess.check_output(['bundle', 'lock', '--add-platform', 'ruby'], cwd=rubyenv_dir)
|
2021-07-06 17:28:04 +01:00
|
|
|
subprocess.check_output(['bundle', 'lock', '--update'], cwd=rubyenv_dir)
|
2021-08-13 18:00:23 +01:00
|
|
|
_remove_platforms(rubyenv_dir)
|
2021-07-06 17:28:04 +01:00
|
|
|
subprocess.check_output(['bundix'], cwd=rubyenv_dir)
|
|
|
|
|
2021-03-14 17:01:17 +00:00
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
cli()
|