mirror of
https://github.com/mag37/dockcheck.git
synced 2026-04-17 18:07:46 +00:00
Compare commits
38 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
5ef1236e16 | ||
|
|
8b77b917da | ||
|
|
5a722e6d40 | ||
|
|
8b8ac84a44 | ||
|
|
d5c90fb98d | ||
|
|
4302d45033 | ||
|
|
a107c18d5f | ||
|
|
dd64079efd | ||
|
|
27e9663167 | ||
|
|
03a3e49650 | ||
|
|
44f9742bde | ||
|
|
07a9269e76 | ||
|
|
3f1be334d5 | ||
|
|
dd42100a3d | ||
|
|
64c1f097b1 | ||
|
|
8f37417eab | ||
|
|
fb44099b0f | ||
|
|
bde81aad22 | ||
|
|
08175c87ab | ||
|
|
a7bfb49fad | ||
|
|
8ae13837f1 | ||
|
|
72a4ab889c | ||
|
|
a4f58c317a | ||
|
|
67fe04cb64 | ||
|
|
d8b30b2363 | ||
|
|
22c8d5e423 | ||
|
|
05c7c8f0dd | ||
|
|
8ff701428b | ||
|
|
f740527595 | ||
|
|
5d5d7eaffe | ||
|
|
986de413fe | ||
|
|
c48b94cd7a | ||
|
|
270456c583 | ||
|
|
7b18a27834 | ||
|
|
c189f40842 | ||
|
|
7f182cddab | ||
|
|
9162ad2457 | ||
|
|
c635d03dbd |
43
README.md
43
README.md
@@ -1,14 +1,17 @@
|
|||||||
# dockcheck
|
# dockcheck
|
||||||
### A script checking updates for docker images **without the need of pulling** - then having the option to auto-update either all or selecting specific containers.
|
### A script checking updates for docker images **without the need of pulling** - then optionally auto-update chosen containers.
|
||||||
|
|
||||||
With the help of [`regctl`](https://github.com/regclient/regclient). This is just a concept for inspiration, use with care.
|
With the help of [`regctl`](https://github.com/regclient/regclient). This is just a concept for inspiration, use with care.
|
||||||
___
|
___
|
||||||
|
|
||||||
## Dependencies:
|
## Dependencies:
|
||||||
Running docker (duh) and compose, either standalone or plugin.
|
Running docker (duh) and compose, either standalone or plugin.
|
||||||
`regctl` by [regclient](https://github.com/regclient/regclient)
|
`regctl` by [regclient](https://github.com/regclient/regclient) (will ask to download `regctl` if not in `PATH` or `PWD`)
|
||||||
The script will ask to download `regctl` if it's not in PATH or current directory.
|
|
||||||
___
|
___
|
||||||
|
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
## `dockcheck.sh`
|
## `dockcheck.sh`
|
||||||
```bash
|
```bash
|
||||||
$ ./dockcheck.sh -h
|
$ ./dockcheck.sh -h
|
||||||
@@ -21,10 +24,6 @@ Options:
|
|||||||
-n No updates, only checking availability.
|
-n No updates, only checking availability.
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||

|
|
||||||
|
|
||||||
Basic example:
|
Basic example:
|
||||||
```bash
|
```bash
|
||||||
$ ./dockcheck.sh
|
$ ./dockcheck.sh
|
||||||
@@ -40,34 +39,24 @@ Containers with updates available:
|
|||||||
3) whoogle-search
|
3) whoogle-search
|
||||||
|
|
||||||
|
|
||||||
Do you want to update? y/[n] y
|
Choose what containers to update:
|
||||||
What containers do you like to update?
|
Enter number(s) separated by comma, [q] to quit: 1,3
|
||||||
Enter number(s) separated by comma: 1,3
|
|
||||||
|
|
||||||
```
|
```
|
||||||
Then it proceedes to run `pull` and `up -d` on every container with updates.
|
Then it proceedes to run `pull` and `up -d` on every container with updates.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
And with `-n` *No updates* and `gl` for `*gl*` filtering:
|
|
||||||
```bash
|
|
||||||
$ ./dockcheck.sh -n gl
|
|
||||||
. . .
|
|
||||||
Containers with updates available:
|
|
||||||
whoogle-search
|
|
||||||
|
|
||||||
Containers on latest version:
|
|
||||||
glances
|
|
||||||
|
|
||||||
No updates installed, exiting
|
|
||||||
```
|
|
||||||
|
|
||||||
### :beetle: Squashed Bugs:
|
### :beetle: Squashed Bugs:
|
||||||
- ~~No options for running without updates or auto update.~~
|
- ~~No options for running without updates or auto update.~~
|
||||||
- ~~No filter to check only specific containers.~~
|
- ~~No filter to check only specific containers.~~
|
||||||
- ~~Faulty registry checkups stopped the updates completely.~~
|
- ~~Faulty registry checkups stopped the updates completely.~~
|
||||||
- ~~No clear checks to skip containers producing errors.~~
|
- ~~No clear checks to skip containers producing errors.~~
|
||||||
- ~~Multi-digest images didn't correctly check with registry, giving false positives on updates.~~
|
- ~~Multi-digest images didn't correctly check with registry, giving false positives on updates.~~
|
||||||
|
- ~~Not working with filenames other than `docker-compose.yml`~~
|
||||||
|
- ~~Lists are not alphabetically sorted (due to stacks and other parameters)~~
|
||||||
|
- ~~Old `docker-compose` binary-check sometimes returned false error~~
|
||||||
|
- ~~Stacks gets updated as whole, even if only one service is chosen.~~
|
||||||
|
- ~~Path broken occationally (from inspect) - probably due to old docker-compose binary.~~
|
||||||
|
|
||||||
### :hammer: Known issues
|
### :hammer: Known issues
|
||||||
- ~~No granular choice of what to update (except initial name filter).~~
|
- ~~No granular choice of what to update (except initial name filter).~~
|
||||||
@@ -90,3 +79,9 @@ Updates available for local_nginx.
|
|||||||
nginx_reverse is already latest.
|
nginx_reverse is already latest.
|
||||||
Updates available for paperless-ng.
|
Updates available for paperless-ng.
|
||||||
```
|
```
|
||||||
|
## Also check out a spinoff brother-project [Palleri/dockcheck-web](https://github.com/Palleri/dockcheck-web) for a WebUI-front!
|
||||||
|
---
|
||||||
|
|
||||||
|
## Special Thanks:
|
||||||
|
:bison: [t0rnis](https://github.com/t0rnis)
|
||||||
|
:leopard: [Palleri](https://github.com/Palleri)
|
||||||
|
|||||||
60
dockcheck.sh
Normal file → Executable file
60
dockcheck.sh
Normal file → Executable file
@@ -1,6 +1,12 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
|
VERSION="v0.1.5"
|
||||||
|
Github="https://github.com/mag37/dockcheck"
|
||||||
|
|
||||||
# Help Function:
|
### Check if there's a new release of the script:
|
||||||
|
LatestRelease="$(curl -s -r 0-30 https://raw.githubusercontent.com/mag37/dockcheck/main/dockcheck.sh | sed -n "/VERSION/s/VERSION=//p" | tr -d '"')"
|
||||||
|
[ "$VERSION" != "$LatestRelease" ] && printf "New version available! Latest: "$LatestRelease" - Local: "$VERSION" \nGrab it here: "$Github" \n\n"
|
||||||
|
|
||||||
|
### Help Function:
|
||||||
Help() {
|
Help() {
|
||||||
echo "Syntax: dockcheck.sh [OPTION] [part of name to filter]"
|
echo "Syntax: dockcheck.sh [OPTION] [part of name to filter]"
|
||||||
echo "Example: dockcheck.sh -a ng"
|
echo "Example: dockcheck.sh -a ng"
|
||||||
@@ -20,7 +26,7 @@ while getopts "aynh" options; do
|
|||||||
done
|
done
|
||||||
shift "$((OPTIND-1))"
|
shift "$((OPTIND-1))"
|
||||||
|
|
||||||
### Set $1 to a variable for later
|
### Set $1 to a variable for name filtering later.
|
||||||
SearchName="$1"
|
SearchName="$1"
|
||||||
|
|
||||||
### Check if required binary exists in PATH or directory:
|
### Check if required binary exists in PATH or directory:
|
||||||
@@ -49,7 +55,7 @@ fi
|
|||||||
### Check docker compose binary:
|
### Check docker compose binary:
|
||||||
if docker compose &> /dev/null ; then
|
if docker compose &> /dev/null ; then
|
||||||
DockerBin="docker compose"
|
DockerBin="docker compose"
|
||||||
elif docker-compose &> /dev/null; then
|
elif docker-compose -v &> /dev/null; then
|
||||||
DockerBin="docker-compose"
|
DockerBin="docker-compose"
|
||||||
else
|
else
|
||||||
printf "%s\n" "No docker compose binary available, quitting."
|
printf "%s\n" "No docker compose binary available, quitting."
|
||||||
@@ -68,21 +74,22 @@ done
|
|||||||
### Choose from list -function:
|
### Choose from list -function:
|
||||||
choosecontainers() {
|
choosecontainers() {
|
||||||
while [[ "$ChoiceClean" =~ [A-Za-z] || -z "$ChoiceClean" ]]; do
|
while [[ "$ChoiceClean" =~ [A-Za-z] || -z "$ChoiceClean" ]]; do
|
||||||
printf "What containers do you like to update? \n"
|
read -p "Enter number(s) separated by comma, [q] to quit: " Choice
|
||||||
# options
|
if [[ "$Choice" =~ [qQnN] ]] ; then
|
||||||
read -p 'Enter number(s) separated by comma (eg. 1,3,4): ' Choice
|
exit 0
|
||||||
if [ "$Choice" == "0" ] ; then
|
elif [ "$Choice" == "0" ] ; then
|
||||||
SelectedUpdates=( ${NumberedUpdates[@]:1} )
|
SelectedUpdates=( "${NumberedUpdates[@]:1}" )
|
||||||
ChoiceClean=$(echo $Choice|sed 's/[,.:;]/ /g')
|
ChoiceClean=$(echo "$Choice" |sed 's/[,.:;]/ /g')
|
||||||
else
|
else
|
||||||
ChoiceClean=$(echo $Choice|sed 's/[,.:;]/ /g')
|
ChoiceClean=$(echo "$Choice" |sed 's/[,.:;]/ /g')
|
||||||
for s in $ChoiceClean; do
|
for s in $ChoiceClean; do
|
||||||
SelectedUpdates+=( ${NumberedUpdates[$s]} )
|
SelectedUpdates+=( "${NumberedUpdates[$s]}" )
|
||||||
done
|
done
|
||||||
fi
|
fi
|
||||||
done
|
done
|
||||||
printf "\nYou've SelectedUpdates:\n"
|
printf "\nUpdating containers:\n"
|
||||||
printf "%s\n" "${SelectedUpdates[@]}"
|
printf "%s\n" "${SelectedUpdates[@]}"
|
||||||
|
printf "\n"
|
||||||
}
|
}
|
||||||
|
|
||||||
### Check the image-hash of every running container VS the registry
|
### Check the image-hash of every running container VS the registry
|
||||||
@@ -91,7 +98,7 @@ for i in $(docker ps --filter "name=$SearchName" --format '{{.Names}}') ; do
|
|||||||
RepoUrl=$(docker inspect "$i" --format='{{.Config.Image}}')
|
RepoUrl=$(docker inspect "$i" --format='{{.Config.Image}}')
|
||||||
LocalHash=$(docker image inspect "$RepoUrl" --format '{{.RepoDigests}}')
|
LocalHash=$(docker image inspect "$RepoUrl" --format '{{.RepoDigests}}')
|
||||||
RegHash=$($regbin image digest --list "$RepoUrl" 2>/dev/null)
|
RegHash=$($regbin image digest --list "$RepoUrl" 2>/dev/null)
|
||||||
# Check if regtcl produces errors - add to GotErrors if so.
|
# Add container to GotErrors if regctl encounter problems.
|
||||||
if [ $? -eq 0 ] ; then
|
if [ $? -eq 0 ] ; then
|
||||||
if [[ "$LocalHash" = *"$RegHash"* ]] ; then NoUpdates+=("$i"); else GotUpdates+=("$i"); fi
|
if [[ "$LocalHash" = *"$RegHash"* ]] ; then NoUpdates+=("$i"); else GotUpdates+=("$i"); fi
|
||||||
else
|
else
|
||||||
@@ -99,6 +106,12 @@ for i in $(docker ps --filter "name=$SearchName" --format '{{.Names}}') ; do
|
|||||||
fi
|
fi
|
||||||
done
|
done
|
||||||
|
|
||||||
|
### Sort arrays alphabetically
|
||||||
|
IFS=$'\n'
|
||||||
|
NoUpdates=($(sort <<<"${NoUpdates[*]}"))
|
||||||
|
GotUpdates=($(sort <<<"${GotUpdates[*]}"))
|
||||||
|
GotErrors=($(sort <<<"${GotErrors[*]}"))
|
||||||
|
unset IFS
|
||||||
### Create new Array to use for the numbered list:
|
### Create new Array to use for the numbered list:
|
||||||
NumberedUpdates=(ALL "${GotUpdates[@]}")
|
NumberedUpdates=(ALL "${GotUpdates[@]}")
|
||||||
|
|
||||||
@@ -119,18 +132,25 @@ fi
|
|||||||
### Optionally get updates if there's any
|
### Optionally get updates if there's any
|
||||||
if [ -n "$GotUpdates" ] ; then
|
if [ -n "$GotUpdates" ] ; then
|
||||||
if [ -z "$UpdYes" ] ; then
|
if [ -z "$UpdYes" ] ; then
|
||||||
printf "\n\033[36;1mDo you want to update? y/[n]\033[0m "
|
printf "\n\033[36;1mChoose what containers to update.\033[0m\n"
|
||||||
read UpdYes
|
choosecontainers
|
||||||
[ "$UpdYes" != "${UpdYes#[Yy]}" ] && choosecontainers
|
|
||||||
else
|
else
|
||||||
SelectedUpdates=( "${GotUpdates[@]}" )
|
SelectedUpdates=( "${GotUpdates[@]}" )
|
||||||
fi
|
fi
|
||||||
if [ "$UpdYes" != "${UpdYes#[Yy]}" ] ; then
|
if [ "$UpdYes" == "${UpdYes#[Nn]}" ] ; then
|
||||||
for i in "${SelectedUpdates[@]}"
|
for i in "${SelectedUpdates[@]}"
|
||||||
do
|
do
|
||||||
ContPath=$(docker inspect "$i" --format '{{ index .Config.Labels "com.docker.compose.project.working_dir"}}')
|
ContPath=$(docker inspect "$i" --format '{{ index .Config.Labels "com.docker.compose.project.working_dir" }}')
|
||||||
$DockerBin -f "$ContPath/docker-compose.yml" pull
|
ContConfigFile=$(docker inspect "$i" --format '{{ index .Config.Labels "com.docker.compose.project.config_files" }}')
|
||||||
$DockerBin -f "$ContPath/docker-compose.yml" up -d
|
ContName=$(docker inspect "$i" --format '{{ index .Config.Labels "com.docker.compose.service" }}')
|
||||||
|
### Checking if "com.docker.compose.project.config_files" returns the full path to the config file or just the file name
|
||||||
|
if [[ $ContConfigFile = '/'* ]] ; then
|
||||||
|
ComposeFile="$ContConfigFile"
|
||||||
|
else
|
||||||
|
ComposeFile="$ContPath/$ContConfigFile"
|
||||||
|
fi
|
||||||
|
$DockerBin -f "$ComposeFile" pull "$ContName"
|
||||||
|
$DockerBin -f "$ComposeFile" up -d "$ContName"
|
||||||
done
|
done
|
||||||
else
|
else
|
||||||
printf "\nNo updates installed, exiting.\n"
|
printf "\nNo updates installed, exiting.\n"
|
||||||
|
|||||||
@@ -1,11 +1,12 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
|
VERSION="v0.1.5"
|
||||||
|
Github="https://github.com/mag37/dockcheck"
|
||||||
|
|
||||||
### DOCKER RUN - VERSION
|
### Check if there's a new release of the script:
|
||||||
|
LatestRelease="$(curl -s -r 0-30 https://raw.githubusercontent.com/mag37/dockcheck/main/dockcheck.sh | sed -n "/VERSION/s/VERSION=//p" | tr -d '"')"
|
||||||
|
[ "$VERSION" != "$LatestRelease" ] && printf "New version available! Latest: "$LatestRelease" - Local: "$VERSION" \nGrab it here: "$Github" \n\n"
|
||||||
|
|
||||||
### WARNING WONT REBUILD CONTAINERS - ONLY GRAB NEW IMAGES
|
### Help Function:
|
||||||
### If running docker compose, use the main version. (recommended!)
|
|
||||||
|
|
||||||
# Help Function:
|
|
||||||
Help() {
|
Help() {
|
||||||
echo "Syntax: dockcheck.sh [OPTION] [part of name to filter]"
|
echo "Syntax: dockcheck.sh [OPTION] [part of name to filter]"
|
||||||
echo "Example: dockcheck.sh -a ng"
|
echo "Example: dockcheck.sh -a ng"
|
||||||
@@ -25,7 +26,7 @@ while getopts "aynh" options; do
|
|||||||
done
|
done
|
||||||
shift "$((OPTIND-1))"
|
shift "$((OPTIND-1))"
|
||||||
|
|
||||||
### Set $1 to a variable for later
|
### Set $1 to a variable for name filtering later.
|
||||||
SearchName="$1"
|
SearchName="$1"
|
||||||
|
|
||||||
### Check if required binary exists in PATH or directory:
|
### Check if required binary exists in PATH or directory:
|
||||||
@@ -51,15 +52,6 @@ else
|
|||||||
exit
|
exit
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
### Check docker compose binary:
|
|
||||||
if docker compose &> /dev/null ; then
|
|
||||||
DockerBin="docker compose"
|
|
||||||
elif docker-compose &> /dev/null; then
|
|
||||||
DockerBin="docker-compose"
|
|
||||||
else
|
|
||||||
printf "%s\n" "No docker compose binary available, quitting."
|
|
||||||
exit
|
|
||||||
fi
|
|
||||||
|
|
||||||
### Numbered List -function:
|
### Numbered List -function:
|
||||||
options() {
|
options() {
|
||||||
@@ -73,21 +65,22 @@ done
|
|||||||
### Choose from list -function:
|
### Choose from list -function:
|
||||||
choosecontainers() {
|
choosecontainers() {
|
||||||
while [[ "$ChoiceClean" =~ [A-Za-z] || -z "$ChoiceClean" ]]; do
|
while [[ "$ChoiceClean" =~ [A-Za-z] || -z "$ChoiceClean" ]]; do
|
||||||
printf "What containers do you like to update? \n"
|
read -p "Enter number(s) separated by comma, [q] to quit: " Choice
|
||||||
# options
|
if [[ "$Choice" =~ [qQnN] ]] ; then
|
||||||
read -p 'Enter number(s) separated by comma (eg. 1,3,4): ' Choice
|
exit 0
|
||||||
if [ "$Choice" == "0" ] ; then
|
elif [ "$Choice" == "0" ] ; then
|
||||||
SelectedUpdates=( ${NumberedUpdates[@]:1} )
|
SelectedUpdates=( "${NumberedUpdates[@]:1}" )
|
||||||
ChoiceClean=$(echo $Choice|sed 's/[,.:;]/ /g')
|
ChoiceClean=$(echo "$Choice" |sed 's/[,.:;]/ /g')
|
||||||
else
|
else
|
||||||
ChoiceClean=$(echo $Choice|sed 's/[,.:;]/ /g')
|
ChoiceClean=$(echo "$Choice" |sed 's/[,.:;]/ /g')
|
||||||
for s in $ChoiceClean; do
|
for s in $ChoiceClean; do
|
||||||
SelectedUpdates+=( ${NumberedUpdates[$s]} )
|
SelectedUpdates+=( "${NumberedUpdates[$s]}" )
|
||||||
done
|
done
|
||||||
fi
|
fi
|
||||||
done
|
done
|
||||||
printf "\nYou've SelectedUpdates:\n"
|
printf "\nUpdating containers:\n"
|
||||||
printf "%s\n" "${SelectedUpdates[@]}"
|
printf "%s\n" "${SelectedUpdates[@]}"
|
||||||
|
printf "\n"
|
||||||
}
|
}
|
||||||
|
|
||||||
### Check the image-hash of every running container VS the registry
|
### Check the image-hash of every running container VS the registry
|
||||||
@@ -96,7 +89,7 @@ for i in $(docker ps --filter "name=$SearchName" --format '{{.Names}}') ; do
|
|||||||
RepoUrl=$(docker inspect "$i" --format='{{.Config.Image}}')
|
RepoUrl=$(docker inspect "$i" --format='{{.Config.Image}}')
|
||||||
LocalHash=$(docker image inspect "$RepoUrl" --format '{{.RepoDigests}}')
|
LocalHash=$(docker image inspect "$RepoUrl" --format '{{.RepoDigests}}')
|
||||||
RegHash=$($regbin image digest --list "$RepoUrl" 2>/dev/null)
|
RegHash=$($regbin image digest --list "$RepoUrl" 2>/dev/null)
|
||||||
# Check if regtcl produces errors - add to GotErrors if so.
|
# Add container to GotErrors if regctl encounter problems.
|
||||||
if [ $? -eq 0 ] ; then
|
if [ $? -eq 0 ] ; then
|
||||||
if [[ "$LocalHash" = *"$RegHash"* ]] ; then NoUpdates+=("$i"); else GotUpdates+=("$i"); fi
|
if [[ "$LocalHash" = *"$RegHash"* ]] ; then NoUpdates+=("$i"); else GotUpdates+=("$i"); fi
|
||||||
else
|
else
|
||||||
@@ -104,6 +97,12 @@ for i in $(docker ps --filter "name=$SearchName" --format '{{.Names}}') ; do
|
|||||||
fi
|
fi
|
||||||
done
|
done
|
||||||
|
|
||||||
|
### Sort arrays alphabetically
|
||||||
|
IFS=$'\n'
|
||||||
|
NoUpdates=($(sort <<<"${NoUpdates[*]}"))
|
||||||
|
GotUpdates=($(sort <<<"${GotUpdates[*]}"))
|
||||||
|
GotErrors=($(sort <<<"${GotErrors[*]}"))
|
||||||
|
unset IFS
|
||||||
### Create new Array to use for the numbered list:
|
### Create new Array to use for the numbered list:
|
||||||
NumberedUpdates=(ALL "${GotUpdates[@]}")
|
NumberedUpdates=(ALL "${GotUpdates[@]}")
|
||||||
|
|
||||||
@@ -124,13 +123,12 @@ fi
|
|||||||
### Optionally get updates if there's any
|
### Optionally get updates if there's any
|
||||||
if [ -n "$GotUpdates" ] ; then
|
if [ -n "$GotUpdates" ] ; then
|
||||||
if [ -z "$UpdYes" ] ; then
|
if [ -z "$UpdYes" ] ; then
|
||||||
printf "\n\033[36;1mDo you want to update? y/[n]\033[0m "
|
printf "\n\033[36;1mChoose what container-images to update.\033[0m\n"
|
||||||
read UpdYes
|
choosecontainers
|
||||||
[ "$UpdYes" != "${UpdYes#[Yy]}" ] && choosecontainers
|
|
||||||
else
|
else
|
||||||
SelectedUpdates=( "${GotUpdates[@]}" )
|
SelectedUpdates=( "${GotUpdates[@]}" )
|
||||||
fi
|
fi
|
||||||
if [ "$UpdYes" != "${UpdYes#[Yy]}" ] ; then
|
if [ "$UpdYes" == "${UpdYes#[Nn]}" ] ; then
|
||||||
for i in "${SelectedUpdates[@]}"; do
|
for i in "${SelectedUpdates[@]}"; do
|
||||||
ContImage=$(docker inspect "$i" --format='{{.Config.Image}}')
|
ContImage=$(docker inspect "$i" --format='{{.Config.Image}}')
|
||||||
docker pull $ContImage
|
docker pull $ContImage
|
||||||
|
|||||||
BIN
example.gif
Normal file
BIN
example.gif
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 345 KiB |
BIN
example_run.gif
BIN
example_run.gif
Binary file not shown.
|
Before Width: | Height: | Size: 72 KiB |
Reference in New Issue
Block a user