Initial working
This commit is contained in:
parent
64b80fa6a8
commit
764d0f9e3a
57
README.md
57
README.md
@ -5,3 +5,60 @@ Easily download releases from sites like Github and Gitea
|
||||
## Original repo
|
||||
|
||||
Originally hosted at https://git.iamthefij.com/iamthefij/release-gitter.git
|
||||
|
||||
## Usage
|
||||
|
||||
At minimum, release-gitter can be used to download the latest release file for a given repo using something like the following:
|
||||
|
||||
release-gitter --git-url https://github.com/coder/super-tool "super-tool-{version}-{system}-{arch}"
|
||||
|
||||
Originally created for downloading binary releases for [pre-commit](https://pre-commit.com) hooks, so it also has features
|
||||
to detect the remote repo automatically using `git remote get-url origin`, as well as detecting the currently checked out version
|
||||
by parsing metadata files (currently only `Cargo.toml`).
|
||||
|
||||
In practice, it means that for a project like [StyLua](https://github.com/JohnnyMorganz/StyLua), when run within the repo one would only need to provide:
|
||||
|
||||
release-gitter --extract-files "stylua" --exec "chmod +x stylua" \
|
||||
--map-system Windows=win64 --map-system Darwin=macos --map-system=linux=Linux \
|
||||
"stylua-{version}-{system}.zip"
|
||||
|
||||
And `release-gitter` will get the release version from the `Cargo.toml`, get the URL from the `git remote`, call the Github API and look for a release matching the templated file name, extract the `stylua` file from the archive, and then make it executable.
|
||||
|
||||
This allows a single command to be run from a checked out repo from pre-commit on any system to fetch the appropriate binary.
|
||||
|
||||
Additionally, it can be used to simplify install instructions for users by providing the `--git-url` option so it can be run from outside the repo.
|
||||
|
||||
Full usage is as follows:
|
||||
|
||||
usage: release-gitter [-h] [--hostname HOSTNAME] [--owner OWNER] [--repo REPO]
|
||||
[--git-url GIT_URL] [--version VERSION]
|
||||
[--map-system MAP_SYSTEM] [--map-arch MAP_ARCH]
|
||||
[--exec EXEC] [--extract-files EXTRACT_FILES]
|
||||
[--extract-all] [--url-only]
|
||||
format
|
||||
|
||||
positional arguments:
|
||||
format Format template to match assets. Eg
|
||||
`foo-{version}-{system}-{arch}.zip`
|
||||
|
||||
optional arguments:
|
||||
-h, --help show this help message and exit
|
||||
--hostname HOSTNAME Git repository hostname
|
||||
--owner OWNER Owner of the repo. If not provided, it will be
|
||||
retrieved from the git url
|
||||
--repo REPO Repo name. If not provided, it will be retrieved from
|
||||
the git url
|
||||
--git-url GIT_URL Git repository URL. Overrides `git remote` detection,
|
||||
but not command line options for hostname, owner, and
|
||||
repo
|
||||
--version VERSION Release version to download. If not provied, it will
|
||||
look for project metadata
|
||||
--map-system MAP_SYSTEM, -s MAP_SYSTEM
|
||||
Map a platform.system() value to a custom value
|
||||
--map-arch MAP_ARCH, -a MAP_ARCH
|
||||
Map a platform.machine() value to a custom value
|
||||
--exec EXEC, -c EXEC Shell commands to execute after download or extraction
|
||||
--extract-files EXTRACT_FILES, -e EXTRACT_FILES
|
||||
A list of file name to extract from downloaded archive
|
||||
--extract-all, -x Shell commands to execute after download or extraction
|
||||
--url-only Only print the URL and do not download
|
||||
|
107
main_test.py
Normal file
107
main_test.py
Normal file
@ -0,0 +1,107 @@
|
||||
import unittest
|
||||
from typing import Any
|
||||
from typing import Callable
|
||||
from typing import NamedTuple
|
||||
from typing import Optional
|
||||
from unittest.mock import MagicMock
|
||||
from unittest.mock import patch
|
||||
|
||||
import requests
|
||||
|
||||
import release_gitter
|
||||
|
||||
|
||||
class TestExpression(NamedTuple):
|
||||
t: unittest.TestCase
|
||||
args: list[Any]
|
||||
kwargs: dict[str, Any]
|
||||
expected: Any
|
||||
exception: Optional[type[Exception]]
|
||||
|
||||
def run(self, f: Callable):
|
||||
with self.t.subTest(f=f, args=self.args, kwargs=self.kwargs):
|
||||
try:
|
||||
result = f(*self.args, **self.kwargs)
|
||||
self.t.assertIsNone(
|
||||
self.exception,
|
||||
f"Expected an exception of type {self.exception}, but found none",
|
||||
)
|
||||
self.t.assertEqual(self.expected, result)
|
||||
return result
|
||||
except Exception as e:
|
||||
if self.exception and isinstance(e, self.exception):
|
||||
return e
|
||||
raise
|
||||
|
||||
|
||||
class TestRemoteInfo(unittest.TestCase):
|
||||
def test_parse_remote_info(self):
|
||||
for test_case in (
|
||||
TestExpression(
|
||||
self,
|
||||
["https://github.com/owner/repo"],
|
||||
{},
|
||||
release_gitter.GitRemoteInfo("github.com", "owner", "repo"),
|
||||
None,
|
||||
),
|
||||
TestExpression(
|
||||
self,
|
||||
["git@github.com:owner/repo"],
|
||||
{},
|
||||
release_gitter.GitRemoteInfo("github.com", "owner", "repo"),
|
||||
None,
|
||||
),
|
||||
TestExpression(
|
||||
self,
|
||||
["ssh://git@git.iamthefij.com/owner/repo"],
|
||||
{},
|
||||
release_gitter.GitRemoteInfo("git.iamthefij.com", "owner", "repo"),
|
||||
None,
|
||||
),
|
||||
TestExpression(
|
||||
self,
|
||||
["https://git@example.com/repo"],
|
||||
{},
|
||||
None,
|
||||
release_gitter.InvalidRemoteError,
|
||||
),
|
||||
):
|
||||
test_case.run(release_gitter.get_git_remote)
|
||||
|
||||
def test_generate_release_url(self):
|
||||
for subtest in (
|
||||
TestExpression(
|
||||
self,
|
||||
[release_gitter.GitRemoteInfo("github.com", "owner", "repo")],
|
||||
{},
|
||||
"https://api.github.com/repos/owner/repo/releases",
|
||||
None,
|
||||
),
|
||||
TestExpression(
|
||||
self,
|
||||
[release_gitter.GitRemoteInfo("git.iamthefij.com", "owner", "repo")],
|
||||
{},
|
||||
"https://git.iamthefij.com/api/v1/repos/owner/repo/releases",
|
||||
None,
|
||||
),
|
||||
TestExpression(
|
||||
self,
|
||||
[release_gitter.GitRemoteInfo("gitlab.com", "owner", "repo")],
|
||||
{},
|
||||
None,
|
||||
release_gitter.InvalidRemoteError,
|
||||
),
|
||||
):
|
||||
mock_response = MagicMock(spec=requests.Response)
|
||||
mock_response.json = MagicMock()
|
||||
if subtest.args[0].hostname == "git.iamthefij.com":
|
||||
mock_response.json.return_value = {
|
||||
"paths": {"/repos/{owner}/{repo}/releases": {}},
|
||||
"basePath": "/api/v1",
|
||||
}
|
||||
with patch("requests.get", return_value=mock_response):
|
||||
subtest.run(release_gitter.GitRemoteInfo.get_releases_url)
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
unittest.main()
|
441
release_gitter.py
Executable file
441
release_gitter.py
Executable file
@ -0,0 +1,441 @@
|
||||
#! /usr/bin/env python3
|
||||
import argparse
|
||||
import platform
|
||||
from collections.abc import Sequence
|
||||
from dataclasses import dataclass
|
||||
from io import BytesIO
|
||||
from mimetypes import guess_type
|
||||
from pathlib import Path
|
||||
from subprocess import check_call
|
||||
from subprocess import check_output
|
||||
from tarfile import TarFile
|
||||
from tarfile import TarInfo
|
||||
from typing import Any
|
||||
from typing import Optional
|
||||
from typing import Union
|
||||
from urllib.parse import urlparse
|
||||
from zipfile import ZipFile
|
||||
|
||||
import requests
|
||||
|
||||
|
||||
# Extract metadata from repo
|
||||
|
||||
class InvalidRemoteError(ValueError):
|
||||
pass
|
||||
|
||||
|
||||
@dataclass
|
||||
class GitRemoteInfo:
|
||||
hostname: str
|
||||
owner: str
|
||||
repo: str
|
||||
|
||||
def get_releases_url(self):
|
||||
"""Gets API url for releases based on hostname and repo info
|
||||
|
||||
Currently only supporting Github and Gitea APIs"""
|
||||
if self.hostname == "github.com":
|
||||
return f"https://api.{self.hostname}/repos/{self.owner}/{self.repo}/releases"
|
||||
|
||||
# Try to detect an api
|
||||
swagger_uri = f"https://{self.hostname}/swagger.v1.json"
|
||||
result = requests.get(swagger_uri)
|
||||
result.raise_for_status()
|
||||
swag = result.json()
|
||||
|
||||
# Look for releases API
|
||||
gitea_releases_template = "/repos/{owner}/{repo}/releases"
|
||||
if gitea_releases_template in swag["paths"]:
|
||||
# TODO: Might be helpful to validate fields that are referenced in responses too
|
||||
return "".join(
|
||||
(
|
||||
"https://",
|
||||
self.hostname,
|
||||
swag["basePath"],
|
||||
gitea_releases_template.format(owner=self.owner, repo=self.repo),
|
||||
)
|
||||
)
|
||||
|
||||
raise InvalidRemoteError(
|
||||
f"Could not find a valid API on host {self.hostname}. Only Github and Gitea APIs are supported"
|
||||
)
|
||||
|
||||
|
||||
def get_git_remote(git_url: Optional[str] = None) -> GitRemoteInfo:
|
||||
"""Extract Github repo info from git remote url"""
|
||||
if not git_url:
|
||||
git_url = (
|
||||
check_output(["git", "remote", "get-url", "origin"]).decode("UTF-8").strip()
|
||||
)
|
||||
|
||||
# Normalize Github ssh url as a proper URL
|
||||
if git_url.startswith("git@github.com:"):
|
||||
git_ssh_parts = git_url.partition(":")
|
||||
if not all(git_ssh_parts):
|
||||
raise InvalidRemoteError(f"Could not parse URL {git_url}. Is this an ssh url?")
|
||||
git_url = f"ssh://{git_ssh_parts[0]}/{git_ssh_parts[2]}"
|
||||
|
||||
u = urlparse(git_url)
|
||||
if not u.hostname:
|
||||
raise ValueError("Not an https url on origin")
|
||||
|
||||
path = u.path.split("/")
|
||||
if len(path) < 3 or not all(path[1:3]):
|
||||
raise InvalidRemoteError(f"{path[1:3]} Could not parse owner and repo from URL {git_url}")
|
||||
|
||||
return GitRemoteInfo(u.hostname, path[1], path[2].removesuffix(".git"))
|
||||
|
||||
|
||||
def get_cargo_version(p: Path) -> str:
|
||||
"""Extracts cargo version from a Cargo.toml file"""
|
||||
with p.open() as f:
|
||||
for line in f:
|
||||
if line.startswith("version"):
|
||||
return line.partition(" = ")[2].strip()[1:-1]
|
||||
|
||||
raise ValueError(f"No version found in {p}")
|
||||
|
||||
|
||||
def read_version() -> Optional[str]:
|
||||
matchers = {
|
||||
"Cargo.toml": get_cargo_version,
|
||||
}
|
||||
|
||||
for name, extractor in matchers.items():
|
||||
p = Path(name)
|
||||
if p.exists():
|
||||
return extractor(p)
|
||||
|
||||
# TODO: Log this out to stderr
|
||||
# raise ValueError(f"Unknown project type. Didn't find any of {matchers.keys()}")
|
||||
return None
|
||||
|
||||
|
||||
# Fetch release and assets from Github
|
||||
|
||||
|
||||
def get_release(
|
||||
remote: GitRemoteInfo,
|
||||
version: Optional[str] = None
|
||||
# TODO: Accept an argument for pre-release
|
||||
) -> dict[Any, Any]:
|
||||
"""Fetches a release object from a Github repo
|
||||
|
||||
If a version number is provided, that version will be retrieved. Otherwise, the latest
|
||||
will be returned.
|
||||
"""
|
||||
result = requests.get(
|
||||
remote.get_releases_url(),
|
||||
# headers={"Accept": "application/vnd.github.v3+json"},
|
||||
headers={"Accept": "application/json"},
|
||||
)
|
||||
result.raise_for_status()
|
||||
|
||||
# Return the latest if requested
|
||||
if version is None or version == "latest":
|
||||
return result.json()[0]
|
||||
|
||||
# Return matching version
|
||||
for release in result.json():
|
||||
if release["tag_name"].endswith(version):
|
||||
return release
|
||||
|
||||
raise ValueError(f"Could not find release version ending in {version}")
|
||||
|
||||
|
||||
def match_asset(
|
||||
release: dict[Any, Any],
|
||||
format: str,
|
||||
version: Optional[str] = None,
|
||||
system_mapping: Optional[dict[str, str]] = None,
|
||||
arch_mapping: Optional[dict[str, str]] = None,
|
||||
) -> dict[Any, Any]:
|
||||
"""Accepts a release and searches for an appropriate asset attached using
|
||||
a provided template and some alternative mappings for version, system, and machine info
|
||||
|
||||
Args
|
||||
`release`: A dict release value from the Github API
|
||||
`format`: is a python format string allowing for "{version}", "{system}", and "{arch}"
|
||||
`version`: the version to use when matching, default will be the name of the release
|
||||
`system_mapping`: alternative values for results returned from `platform.system()`
|
||||
`arch_mapping`: alternative values for results returned from `platform.machine()`
|
||||
|
||||
Note: Some fuzziness is built into the {version} template variable. We will try to match against
|
||||
the version as is, prefixed with a 'v' and have 'v' stripped from the beginning.
|
||||
|
||||
Eg. An example from an arm64 Mac:
|
||||
|
||||
match_asset({"name": "v1.0.0", ...}, `foo-{version}-{system}-{arch}.zip`)
|
||||
|
||||
Matches against:
|
||||
* "foo-v1.0.0-Darwin-arm64.zip"
|
||||
* "foo-1.0.0-Darwin-arm64.zip"
|
||||
|
||||
Now, instead of Darwin, maybe you want to use `macOS`. For that you'd provide a
|
||||
`system_mapping`.
|
||||
|
||||
match_asset({"name": "v1.0.0", ...}, `foo-{version}-{system}-{arch}.zip, system_mapping={"Darwin": "macOS"})
|
||||
|
||||
Matches against:
|
||||
* "foo-v1.0.0-macOS-arm64.zip"
|
||||
* "foo-1.0.0-macOS-arm64.zip"
|
||||
"""
|
||||
if version is None:
|
||||
version = release["tag_name"]
|
||||
|
||||
# This should never really happen
|
||||
if version is None:
|
||||
if "{version}" in format:
|
||||
raise ValueError(
|
||||
"No version provided or found in release name but is in format"
|
||||
)
|
||||
else:
|
||||
# This should never happen, but since version isn't used anywhere, we can make it an empty string
|
||||
version = ""
|
||||
|
||||
system = platform.system()
|
||||
if system_mapping:
|
||||
system = system_mapping.get(system, system)
|
||||
|
||||
arch = platform.machine()
|
||||
if arch_mapping:
|
||||
arch = arch_mapping.get(arch, arch)
|
||||
|
||||
expected_names = {
|
||||
format.format(
|
||||
version=normalized_version,
|
||||
system=system,
|
||||
arch=arch,
|
||||
)
|
||||
for normalized_version in (
|
||||
version.lstrip("v"),
|
||||
"v" + version if not version.startswith("v") else version,
|
||||
)
|
||||
}
|
||||
|
||||
for asset in release["assets"]:
|
||||
if asset["name"] in expected_names:
|
||||
return asset
|
||||
|
||||
raise ValueError(
|
||||
f"Could not find asset named {expected_names} on release {release['name']}"
|
||||
)
|
||||
|
||||
|
||||
class PackageAdapter:
|
||||
"""Adapts the names and extractall methods from ZipFile and TarFile classes"""
|
||||
|
||||
def __init__(self, content_type: str, response: requests.Response):
|
||||
self._package: Union[TarFile, ZipFile]
|
||||
if content_type == "application/zip":
|
||||
self._package = ZipFile(BytesIO(response.content))
|
||||
elif content_type == "application/x-tar":
|
||||
self._package = TarFile(fileobj=response.raw)
|
||||
elif content_type == "application/x-tar+gzip":
|
||||
self._package = TarFile.open(fileobj=BytesIO(response.content), mode="r:*")
|
||||
else:
|
||||
raise ValueError(f"Unknown or unsupported content type {content_type}")
|
||||
|
||||
def get_names(self) -> list[str]:
|
||||
"""Get list of all file names in package"""
|
||||
if isinstance(self._package, ZipFile):
|
||||
return self._package.namelist()
|
||||
if isinstance(self._package, TarFile):
|
||||
return self._package.getnames()
|
||||
|
||||
raise ValueError(
|
||||
f"Unknown package type, cannot extract from {type(self._package)}"
|
||||
)
|
||||
|
||||
def extractall(self, file_names: list[str]) -> list[str]:
|
||||
"""Extract all or a subset of files from the package
|
||||
|
||||
If the `file_names` list is empty, all files will be extracted"""
|
||||
if not file_names:
|
||||
self._package.extractall()
|
||||
return self.get_names()
|
||||
|
||||
if isinstance(self._package, ZipFile):
|
||||
self._package.extractall(members=file_names)
|
||||
if isinstance(self._package, TarFile):
|
||||
self._package.extractall(members=(TarInfo(name) for name in file_names))
|
||||
|
||||
return file_names
|
||||
|
||||
|
||||
def download_asset(
|
||||
asset: dict[Any, Any],
|
||||
extract_files: Optional[list[str]] = None,
|
||||
) -> list[Path]:
|
||||
result = requests.get(asset["browser_download_url"])
|
||||
|
||||
content_type = asset.get(
|
||||
"content_type",
|
||||
guess_type(asset["name"]),
|
||||
)
|
||||
if extract_files is not None:
|
||||
if isinstance(content_type, tuple):
|
||||
content_type = "+".join(t for t in content_type if t is not None)
|
||||
if not content_type:
|
||||
raise TypeError(
|
||||
"Cannot extract files from archive because we don't recognize the content type"
|
||||
)
|
||||
package = PackageAdapter(content_type, result)
|
||||
extract_files = package.extractall(extract_files)
|
||||
return [Path.cwd() / name for name in extract_files]
|
||||
|
||||
file_name = Path.cwd() / asset["name"]
|
||||
with open(file_name, "wb") as f:
|
||||
f.write(result.content)
|
||||
|
||||
return [file_name]
|
||||
|
||||
|
||||
class MapAddAction(argparse.Action):
|
||||
def __call__(
|
||||
self,
|
||||
_: argparse.ArgumentParser,
|
||||
namespace: argparse.Namespace,
|
||||
values: Union[str, Sequence[Any], None],
|
||||
option_string: Optional[str] = None,
|
||||
):
|
||||
# Validate that required value has something
|
||||
if self.required and not values:
|
||||
raise argparse.ArgumentError(
|
||||
self, f"Did not provide required argument {option_string}"
|
||||
)
|
||||
|
||||
# Get and initialize the destination
|
||||
dest = getattr(namespace, self.dest)
|
||||
if dest is None:
|
||||
dest = {}
|
||||
|
||||
# Parse values
|
||||
if values is not None:
|
||||
if isinstance(values, str):
|
||||
values = (values,)
|
||||
for value in values:
|
||||
if "=" not in value:
|
||||
raise argparse.ArgumentError(
|
||||
self,
|
||||
f"Value needs to be in the form `key=value` and received {value}",
|
||||
)
|
||||
parts = value.partition("=")
|
||||
dest[parts[0]] = parts[2]
|
||||
|
||||
# Set dest value
|
||||
setattr(namespace, self.dest, dest)
|
||||
|
||||
|
||||
def parse_args(args: Optional[list[str]] = None) -> argparse.Namespace:
|
||||
parser = argparse.ArgumentParser()
|
||||
parser.add_argument(
|
||||
"format",
|
||||
help="Format template to match assets. Eg `foo-{version}-{system}-{arch}.zip`",
|
||||
)
|
||||
parser.add_argument(
|
||||
"--hostname",
|
||||
help="Git repository hostname",
|
||||
)
|
||||
parser.add_argument(
|
||||
"--owner",
|
||||
help="Owner of the repo. If not provided, it will be retrieved from the git url",
|
||||
)
|
||||
parser.add_argument(
|
||||
"--repo",
|
||||
help="Repo name. If not provided, it will be retrieved from the git url",
|
||||
)
|
||||
parser.add_argument(
|
||||
"--git-url",
|
||||
help="Git repository URL. Overrides `git remote` detection, but not command line options for hostname, owner, and repo",
|
||||
)
|
||||
parser.add_argument(
|
||||
"--version",
|
||||
help="Release version to download. If not provied, it will look for project metadata",
|
||||
)
|
||||
parser.add_argument(
|
||||
"--map-system",
|
||||
"-s",
|
||||
action=MapAddAction,
|
||||
help="Map a platform.system() value to a custom value",
|
||||
)
|
||||
parser.add_argument(
|
||||
"--map-arch",
|
||||
"-a",
|
||||
action=MapAddAction,
|
||||
help="Map a platform.machine() value to a custom value",
|
||||
)
|
||||
parser.add_argument(
|
||||
"--exec", "-c", help="Shell commands to execute after download or extraction"
|
||||
)
|
||||
parser.add_argument(
|
||||
"--extract-files",
|
||||
"-e",
|
||||
action="append",
|
||||
help="A list of file name to extract from downloaded archive",
|
||||
)
|
||||
parser.add_argument(
|
||||
"--extract-all",
|
||||
"-x",
|
||||
action="store_true",
|
||||
help="Shell commands to execute after download or extraction",
|
||||
)
|
||||
parser.add_argument(
|
||||
"--url-only",
|
||||
action="store_true",
|
||||
help="Only print the URL and do not download",
|
||||
)
|
||||
|
||||
parsed_args = parser.parse_args(args)
|
||||
|
||||
# Merge in fields from args and git remote
|
||||
if not all((parsed_args.owner, parsed_args.repo, parsed_args.hostname)):
|
||||
remote_info = get_git_remote(parsed_args.git_url)
|
||||
|
||||
def merge_field(a, b, field):
|
||||
value = getattr(a, field)
|
||||
if value is None:
|
||||
setattr(a, field, getattr(b, field))
|
||||
|
||||
for field in ("owner", "repo", "hostname"):
|
||||
merge_field(parsed_args, remote_info, field)
|
||||
|
||||
if parsed_args.version is None:
|
||||
parsed_args.version = read_version()
|
||||
|
||||
if parsed_args.extract_all:
|
||||
parsed_args.extract_files = []
|
||||
|
||||
return parsed_args
|
||||
|
||||
|
||||
def main():
|
||||
args = parse_args()
|
||||
|
||||
release = get_release(
|
||||
GitRemoteInfo(args.hostname, args.owner, args.repo), args.version
|
||||
)
|
||||
asset = match_asset(
|
||||
release,
|
||||
args.format,
|
||||
version=args.version,
|
||||
system_mapping=args.map_system,
|
||||
arch_mapping=args.map_arch,
|
||||
)
|
||||
|
||||
if args.url_only:
|
||||
print(asset["browser_download_url"])
|
||||
return
|
||||
|
||||
files = download_asset(asset, extract_files=args.extract_files)
|
||||
|
||||
print(f"Downloaded {', '.join(str(f) for f in files)}")
|
||||
|
||||
# Optionally execute post command
|
||||
if args.exec:
|
||||
check_call(args.exec, shell=True)
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
main()
|
3
setup.py
3
setup.py
@ -25,6 +25,7 @@ setup(
|
||||
"Programming Language :: Python :: 3.5",
|
||||
"Programming Language :: Python :: 3.6",
|
||||
"Programming Language :: Python :: 3.7",
|
||||
"Programming Language :: Python :: 3.9",
|
||||
],
|
||||
keywords="",
|
||||
packages=find_packages(
|
||||
@ -36,7 +37,7 @@ setup(
|
||||
"tests",
|
||||
]
|
||||
),
|
||||
install_requires=[],
|
||||
install_requires=["requests"],
|
||||
entry_points={
|
||||
"console_scripts": [
|
||||
"release-gitter=release_gitter:main",
|
||||
|
Loading…
x
Reference in New Issue
Block a user