update theme for latest mirrormanager
various fixes and updates for running in ECS, too
This commit is contained in:
parent
4d7aab6541
commit
363191a9b5
9 changed files with 144 additions and 38 deletions
|
@ -1,5 +1,5 @@
|
||||||
# Stage 1: Build stage with necessary build dependencies
|
# Stage 1: Build stage with necessary build dependencies
|
||||||
FROM quay.io/fedora/python-312:latest AS build-stage
|
FROM quay.io/fedora/python-312:20240814 AS prebuild
|
||||||
LABEL \
|
LABEL \
|
||||||
name="python-312-with-rust" \
|
name="python-312-with-rust" \
|
||||||
vendor="Fedora Infrastructure" \
|
vendor="Fedora Infrastructure" \
|
||||||
|
@ -19,6 +19,10 @@ RUN dnf install -y \
|
||||||
libffi-devel \
|
libffi-devel \
|
||||||
openssl-devel
|
openssl-devel
|
||||||
|
|
||||||
|
|
||||||
|
# Stage 2: Build mirrormanager2 python code
|
||||||
|
FROM prebuild AS mirrormanager2-build
|
||||||
|
|
||||||
# Clone MirrorManager2 source code from the Git repo
|
# Clone MirrorManager2 source code from the Git repo
|
||||||
RUN mkdir -p /opt/mirrormanager2
|
RUN mkdir -p /opt/mirrormanager2
|
||||||
WORKDIR /opt/mirrormanager2
|
WORKDIR /opt/mirrormanager2
|
||||||
|
@ -27,16 +31,38 @@ RUN git clone https://github.com/fedora-infra/mirrormanager2.git .
|
||||||
RUN sed -e 's/signed_fpca/signed_rosca/' -i mirrormanager2/perms.py mirrormanager2/auth.py
|
RUN sed -e 's/signed_fpca/signed_rosca/' -i mirrormanager2/perms.py mirrormanager2/auth.py
|
||||||
RUN pip install --prefix=/install .
|
RUN pip install --prefix=/install .
|
||||||
|
|
||||||
RUN pip install --prefix=/install flask_session
|
RUN pip install --prefix=/install flask_session psycopg2
|
||||||
|
|
||||||
# Stage 2: Final stage with runtime dependencies
|
# Stage 3: Build generate-mirrorlist-cache and mirrorlist-server
|
||||||
FROM quay.io/fedora/python-312:latest AS runtime
|
FROM prebuild AS mirrorlist-build
|
||||||
|
|
||||||
|
RUN mkdir -p /opt/mirrorlist-server
|
||||||
|
WORKDIR /opt/mirrorlist-server
|
||||||
|
RUN git clone https://github.com/adrianreber/mirrorlist-server.git .
|
||||||
|
RUN echo -e '[profile.release-with-debug]\ninherits = "release"\ndebug = true\n' >> Cargo.toml
|
||||||
|
|
||||||
|
# NOTE(neil): 20241217 #![deny(warnings)] causes deprecated/removed lints to be errors...
|
||||||
|
RUN sed -i 's,#!\[deny(warnings)\],\#!\[allow(renamed_and_removed_lints)\],' src/bin/mirrorlist-server.rs src/bin/generate-mirrorlist-cache.rs
|
||||||
|
|
||||||
|
RUN cargo build --profile=release-with-debug
|
||||||
|
|
||||||
|
# Stage 4: Build scan-primary-mirror
|
||||||
|
FROM prebuild AS scan-primary-mirror-build
|
||||||
|
|
||||||
|
RUN mkdir -p /opt/scan-primary-mirror
|
||||||
|
WORKDIR /opt/scan-primary-mirror
|
||||||
|
RUN git clone https://github.com/adrianreber/scan-primary-mirror.git .
|
||||||
|
RUN RUSTFLAGS=-g cargo build --release
|
||||||
|
|
||||||
|
# Stage 5: Final stage with runtime dependencies
|
||||||
|
FROM quay.io/fedora/python-312:20240814 AS runtime
|
||||||
LABEL \
|
LABEL \
|
||||||
name="python-312-with-rust" \
|
name="python-312-with-rust" \
|
||||||
vendor="Fedora Infrastructure" \
|
vendor="Fedora Infrastructure" \
|
||||||
license="MIT"
|
license="MIT"
|
||||||
|
|
||||||
USER root
|
USER root
|
||||||
|
|
||||||
# Add only runtime dependencies
|
# Add only runtime dependencies
|
||||||
RUN dnf install -y \
|
RUN dnf install -y \
|
||||||
python3-pyrpmmd \
|
python3-pyrpmmd \
|
||||||
|
@ -44,33 +70,41 @@ RUN dnf install -y \
|
||||||
uwsgi-plugin-python3 \
|
uwsgi-plugin-python3 \
|
||||||
logrotate
|
logrotate
|
||||||
|
|
||||||
# Copy installed dependencies from the build stage
|
# Copy installation of mirrormanager from it's build stage
|
||||||
COPY --from=build-stage /install /usr/
|
COPY --from=mirrormanager2-build /install /opt/app-root/
|
||||||
|
|
||||||
# Copy in the tree
|
# Copy mirrorlist-server and generate-mirrorlist-cache from mirrorlist-server build
|
||||||
#COPY --from=build-stage /opt/mirrormanager2 /opt/mirrormanager2
|
COPY --from=mirrorlist-build /opt/mirrorlist-server/target/release-with-debug/generate-mirrorlist-cache /usr/local/bin/
|
||||||
|
COPY --from=mirrorlist-build /opt/mirrorlist-server/target/release-with-debug/mirrorlist-server /usr/local/bin/
|
||||||
|
|
||||||
|
# Copy scan-primary-mirror from its build step
|
||||||
|
COPY --from=scan-primary-mirror-build /opt/scan-primary-mirror/target/release/scan-primary-mirror /usr/local/bin/
|
||||||
|
|
||||||
|
# Copy in the wsgi entry point
|
||||||
ADD run.py /opt/mirrormanager2/
|
ADD run.py /opt/mirrormanager2/
|
||||||
|
|
||||||
FROM runtime AS database
|
# Stage 6: flatten :)
|
||||||
COPY client_secrets.json /etc/mirrormanager/
|
|
||||||
COPY mirrormanager2.cfg /etc/mirrormanager/
|
|
||||||
ENV MM2_CONFIG=/etc/mirrormanager/mirrormanager2.cfg
|
|
||||||
RUN /usr/bin/python3 -m flask -A mirrormanager2.app db sync
|
|
||||||
|
|
||||||
FROM runtime as final
|
FROM runtime as final
|
||||||
COPY --from=database /var/tmp/mirrormanager2_dev.sqlite /var/tmp/mirrormanager2_dev.sqlite
|
|
||||||
|
|
||||||
# Set working directory
|
LABEL \
|
||||||
|
name="python-312-with-rust-mirrormanager" \
|
||||||
|
vendor="Rocky Linux Infrastructure" \
|
||||||
|
license="MIT"
|
||||||
|
|
||||||
WORKDIR /opt/mirrormanager2
|
WORKDIR /opt/mirrormanager2
|
||||||
|
|
||||||
# Expose necessary ports
|
|
||||||
EXPOSE 5000
|
EXPOSE 5000
|
||||||
|
|
||||||
|
# @TODO(neil): probably this shouldn't be here? maybe SSM -> parse out in mm2.cfg?
|
||||||
COPY client_secrets.json /etc/mirrormanager/
|
COPY client_secrets.json /etc/mirrormanager/
|
||||||
COPY mirrormanager2.cfg /etc/mirrormanager/
|
COPY mirrormanager2.cfg /etc/mirrormanager/
|
||||||
|
COPY static/rocky /opt/app-root/lib/python3.12/site-packages/mirrormanager2/static/rocky
|
||||||
|
COPY templates/rocky /opt/app-root/lib/python3.12/site-packages/mirrormanager2/templates/rocky
|
||||||
|
|
||||||
ENV MM2_CONFIG=/etc/mirrormanager/mirrormanager2.cfg
|
ENV MM2_CONFIG=/etc/mirrormanager/mirrormanager2.cfg
|
||||||
|
|
||||||
COPY static/rocky /usr/lib/python3.12/site-packages/mirrormanager2/static/rocky
|
ENV SETUP_DB=false
|
||||||
COPY templates/rocky /usr/lib/python3.12/site-packages/mirrormanager2/templates/rocky
|
RUN test "$SETUP_DB" || python -m flask -A mirrormanager2.app db sync
|
||||||
|
|
||||||
# Define entrypoint script to start the application
|
# Define entrypoint script to start the application
|
||||||
CMD [ "uwsgi", "--socket", "0.0.0.0:5000", \
|
CMD [ "uwsgi", "--socket", "0.0.0.0:5000", \
|
||||||
|
@ -80,5 +114,6 @@ CMD [ "uwsgi", "--socket", "0.0.0.0:5000", \
|
||||||
"--enable-threads", \
|
"--enable-threads", \
|
||||||
"--master", \
|
"--master", \
|
||||||
"-b", "65535", \
|
"-b", "65535", \
|
||||||
|
"-H", "/opt/app-root/", \
|
||||||
"--wsgi-file", "/opt/mirrormanager2/run.py" ]
|
"--wsgi-file", "/opt/mirrormanager2/run.py" ]
|
||||||
|
|
||||||
|
|
|
@ -10,7 +10,7 @@ import os
|
||||||
|
|
||||||
|
|
||||||
# url to the database server:
|
# url to the database server:
|
||||||
SQLALCHEMY_DATABASE_URI = 'sqlite:////var/tmp/mirrormanager2_dev.sqlite'
|
SQLALCHEMY_DATABASE_URI = os.environ.get('MM2_SQLALCHEMY_DATABASE_URI', 'sqlite:////var/tmp/mirrormanager2_dev.sqlite')
|
||||||
|
|
||||||
# the number of items to display on the search pages
|
# the number of items to display on the search pages
|
||||||
# Default: ``50``.
|
# Default: ``50``.
|
||||||
|
@ -21,7 +21,7 @@ SECRET_KEY = os.environ.get('MM2_SECRET_KEY')
|
||||||
|
|
||||||
# Seed used to make the password harder to brute force in case of leaking
|
# Seed used to make the password harder to brute force in case of leaking
|
||||||
# This should be kept really secret!
|
# This should be kept really secret!
|
||||||
PASSWORD_SEED = os.environ.get('MM2_PASSWORD_SEED')
|
PASSWORD_SEED = os.environ.get('MM2_PASSWORD_SEED') # only valid for local auth
|
||||||
|
|
||||||
###
|
###
|
||||||
# Other configuration items for the web-app
|
# Other configuration items for the web-app
|
||||||
|
@ -63,7 +63,7 @@ ADMIN_EMAIL = "infrastructure@rockylinux.org"
|
||||||
|
|
||||||
# Email address used in the "From" field of the emails sent.
|
# Email address used in the "From" field of the emails sent.
|
||||||
# Default: ``nobody@fedoraproject.org``.
|
# Default: ``nobody@fedoraproject.org``.
|
||||||
#EMAIL_FROM = "nobody@fedoraproject.org"
|
EMAIL_FROM = "nobody@rockylinux.org"
|
||||||
|
|
||||||
# SMTP server to use,
|
# SMTP server to use,
|
||||||
# Default: ``localhost``.
|
# Default: ``localhost``.
|
||||||
|
@ -74,7 +74,7 @@ ADMIN_EMAIL = "infrastructure@rockylinux.org"
|
||||||
# SMTP_PASSWORD = 'password'
|
# SMTP_PASSWORD = 'password'
|
||||||
|
|
||||||
# Countries which have to be excluded.
|
# Countries which have to be excluded.
|
||||||
#EMBARGOED_COUNTRIES = ["CU", "IR", "KP", "SD", "SY"]
|
EMBARGOED_COUNTRIES = ["CU", "IR", "KP", "SD", "SY", "RU"]
|
||||||
|
|
||||||
# When this is set to True, an additional menu item is shown which
|
# When this is set to True, an additional menu item is shown which
|
||||||
# displays the maps generated with mm2_generate-worldmap.
|
# displays the maps generated with mm2_generate-worldmap.
|
||||||
|
|
8
run.py
8
run.py
|
@ -4,6 +4,14 @@ from flask_session import Session
|
||||||
from cachelib.file import FileSystemCache
|
from cachelib.file import FileSystemCache
|
||||||
import os
|
import os
|
||||||
|
|
||||||
|
password = os.environ.get('MM2_DATABASE_PASSWORD')
|
||||||
|
if password:
|
||||||
|
user = os.environ.get('DB_USER')
|
||||||
|
host = os.environ.get('DB_HOST')
|
||||||
|
port = os.environ.get('DB_PORT')
|
||||||
|
name = os.environ.get('DB_NAME')
|
||||||
|
os.environ["MM2_SQLALCHEMY_DATABASE_URI"] = f"postgresql://{user}:{password}@{host}:{port}/{name}"
|
||||||
|
|
||||||
application = create_app()
|
application = create_app()
|
||||||
application.debug = os.environ.get("MM2_DEBUG", False)
|
application.debug = os.environ.get("MM2_DEBUG", False)
|
||||||
application.config['SESSION_TYPE'] = "cachelib"
|
application.config['SESSION_TYPE'] = "cachelib"
|
||||||
|
|
15
start-dev.sh
15
start-dev.sh
|
@ -2,10 +2,18 @@ POD=mirrormanager2
|
||||||
|
|
||||||
podman pod exists $POD || podman pod create -p 5000:5000 -n $POD
|
podman pod exists $POD || podman pod create -p 5000:5000 -n $POD
|
||||||
|
|
||||||
|
podman container exists postgres || podman run --pod $POD \
|
||||||
|
--name postgres \
|
||||||
|
--replace -d \
|
||||||
|
-e POSTGRES_PASSWORD=mirrormanager \
|
||||||
|
-e POSTGRES_DB=mirrormanager \
|
||||||
|
-v pgdata:/var/lib/postgresql/data \
|
||||||
|
docker.io/library/postgres:15
|
||||||
|
|
||||||
test -d tmp || mkdir tmp
|
test -d tmp || mkdir tmp
|
||||||
test -f client_secrets.json || (echo "missing client_secrets" && exit 2)
|
test -f client_secrets.json || (echo "missing client_secrets" && exit 2)
|
||||||
|
|
||||||
podman build -t git.resf.org/infrastructure/mirrormanager2:dev -f Containerfile
|
#podman build -t git.resf.org/infrastructure/mirrormanager2:dev -f Containerfile
|
||||||
podman rm --force mm2 -t 1
|
podman rm --force mm2 -t 1
|
||||||
podman run \
|
podman run \
|
||||||
--pod $POD \
|
--pod $POD \
|
||||||
|
@ -14,14 +22,15 @@ podman run \
|
||||||
-e 'MM2_CONFIG=/etc/mirrormanager/mirrormanager2.cfg' \
|
-e 'MM2_CONFIG=/etc/mirrormanager/mirrormanager2.cfg' \
|
||||||
-e "MM2_SECRET_KEY=$(openssl rand -hex 32)" \
|
-e "MM2_SECRET_KEY=$(openssl rand -hex 32)" \
|
||||||
-e "MM2_PASSWORD_SEED=$(openssl rand -hex 32)" \
|
-e "MM2_PASSWORD_SEED=$(openssl rand -hex 32)" \
|
||||||
|
-e 'MM2_SQLALCHEMY_DATABASE_URI=postgresql://postgres:mirrormanager@postgres:5432/mirrormanager' \
|
||||||
-e 'MM2_THEME_FOLDER=rocky' \
|
-e 'MM2_THEME_FOLDER=rocky' \
|
||||||
-e 'FLASK_DEBUG=1' \
|
-e 'FLASK_DEBUG=1' \
|
||||||
-e 'MM2_DEBUG=1' \
|
-e 'MM2_DEBUG=1' \
|
||||||
-v $PWD/mirrormanager2.cfg:/etc/mirrormanager/mirrormanager2.cfg:z,ro \
|
-v $PWD/mirrormanager2.cfg:/etc/mirrormanager/mirrormanager2.cfg:z,ro \
|
||||||
-v $PWD/client_secrets.json:/etc/mirrormanager/client_secrets.json:z,ro \
|
-v $PWD/client_secrets.json:/etc/mirrormanager/client_secrets.json:z,ro \
|
||||||
-v $PWD/tmp:/var/tmp:z,rw \
|
-v $PWD/tmp:/var/tmp:z,rw \
|
||||||
-v $PWD/static:/usr/lib/python3.12/site-packages/mirrormanager2/static:z,ro \
|
-v $PWD/static:/opt/app-root/lib/python3.12/site-packages/mirrormanager2/static:z,ro \
|
||||||
-v $PWD/templates:/usr/lib/python3.12/site-packages/mirrormanager2/templates:z,ro \
|
-v $PWD/templates:/opt/app-root/lib/python3.12/site-packages/mirrormanager2/templates:z,ro \
|
||||||
-d git.resf.org/infrastructure/mirrormanager2:dev
|
-d git.resf.org/infrastructure/mirrormanager2:dev
|
||||||
|
|
||||||
# Setup sqlite database
|
# Setup sqlite database
|
||||||
|
|
|
@ -32,7 +32,8 @@
|
||||||
--bs-danger: #dc3545;
|
--bs-danger: #dc3545;
|
||||||
--bs-light: #f8f9fa;
|
--bs-light: #f8f9fa;
|
||||||
--bs-dark: #212529;
|
--bs-dark: #212529;
|
||||||
--bs-primary-rgb: 60, 151, 214;
|
/*--bs-primary-rgb: 60, 151, 214;*/
|
||||||
|
--bs-primary-rgb: 16, 185, 91;
|
||||||
--bs-secondary-rgb: 108, 117, 125;
|
--bs-secondary-rgb: 108, 117, 125;
|
||||||
--bs-success-rgb: 25, 135, 84;
|
--bs-success-rgb: 25, 135, 84;
|
||||||
--bs-info-rgb: 13, 202, 240;
|
--bs-info-rgb: 13, 202, 240;
|
||||||
|
|
|
@ -77,6 +77,7 @@
|
||||||
{% endset %}
|
{% endset %}
|
||||||
<p>
|
<p>
|
||||||
<a href="https://github.com/fedora-infra/mirrormanager2/">mirrormanager</a>
|
<a href="https://github.com/fedora-infra/mirrormanager2/">mirrormanager</a>
|
||||||
|
-- {{version}}
|
||||||
-- <a href="http://mirrormanager.rtfd.org" rel="noopener noreferrer"
|
-- <a href="http://mirrormanager.rtfd.org" rel="noopener noreferrer"
|
||||||
target="_blank">Documentation</a>
|
target="_blank">Documentation</a>
|
||||||
-- <a href="http://mirrormanager.readthedocs.org/en/latest/contributors.html">Authors</a></p>
|
-- <a href="http://mirrormanager.readthedocs.org/en/latest/contributors.html">Authors</a></p>
|
||||||
|
@ -277,3 +278,34 @@ href="{{ url_for('static', filename='mirrormanager2.css') }}"/>
|
||||||
{{ form.csrf_token }}
|
{{ form.csrf_token }}
|
||||||
</form>
|
</form>
|
||||||
{% endmacro %}
|
{% endmacro %}
|
||||||
|
|
||||||
|
|
||||||
|
{% macro pagination_bar(result) %}
|
||||||
|
{% if result.total_pages > 1 %}
|
||||||
|
<nav aria-label="Pagination">
|
||||||
|
<ul class="pagination pagination-sm justify-content-center my-4">
|
||||||
|
{# Page list #}
|
||||||
|
{% for page_number in result.truncated_pages_list(margin=6) %}
|
||||||
|
{% if page_number == result.page_number %}
|
||||||
|
<li class="page-item active" aria-current="page">
|
||||||
|
<span class="page-link">
|
||||||
|
{{ page_number }}
|
||||||
|
<span class="sr-only">(current)</span>
|
||||||
|
</span>
|
||||||
|
</li>
|
||||||
|
{% elif page_number == None %}
|
||||||
|
<li class="page-item disabled">
|
||||||
|
<span class="page-link">...</span>
|
||||||
|
</li>
|
||||||
|
{% else %}
|
||||||
|
<li class="page-item">
|
||||||
|
<a class="page-link" href="{{ result.page_url(page_number) }}">
|
||||||
|
{{ page_number }}
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
{% endif %}
|
||||||
|
{% endfor %}
|
||||||
|
</ul>
|
||||||
|
</nav>
|
||||||
|
{% endif %}
|
||||||
|
{% endmacro %}
|
||||||
|
|
|
@ -231,14 +231,26 @@
|
||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% if category.directories %}
|
{% if category.directories %}
|
||||||
<h3>Up-to-Date Directories this host carries</h3>
|
<div class="accordion accordion-flush" id="accordion-category-{{category.id}}">
|
||||||
<ul>
|
<div class="accordion-item">
|
||||||
{% for dir in category.directories %}
|
<h2 class="accordion-header fw-bold" id="heading-category-{{category.id}}">
|
||||||
{% if dir.up2date %}
|
<button class="accordion-button collapsed" type="button" data-bs-toggle="collapse" data-bs-target="#collapse-category-{{category.id}}" aria-expanded="false" aria-controls="flush-collapseOne">
|
||||||
<li>{{ dir.path }}</li>
|
<strong>Up-to-Date Directories this host carries <span class="badge bg-primary"> {{category.directories|length}}</span></strong>
|
||||||
{% endif %}
|
</button>
|
||||||
{% endfor %}
|
</h2>
|
||||||
</ul>
|
<div id="collapse-category-{{category.id}}" class="accordion-collapse collapse" aria-labelledby="#heading-category-{{category.id}}" data-bs-parent="#accordion-category-{{category.id}}">
|
||||||
|
<div class="accordion-body">
|
||||||
|
<ul>
|
||||||
|
{% for dir in category.directories %}
|
||||||
|
{% if dir.up2date %}
|
||||||
|
<li>{{ dir.path }}</li>
|
||||||
|
{% endif %}
|
||||||
|
{% endfor %}
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
</li>
|
</li>
|
||||||
|
|
|
@ -34,8 +34,14 @@ I2 means both Internet2 and its peer high speed research and development
|
||||||
networks globally.
|
networks globally.
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
|
|
||||||
<table class="table table-sm mt-5">
|
<table class="table table-sm mt-5">
|
||||||
<tr id="matrixheadings">
|
<tr id="matrixtitle">
|
||||||
|
<th colspan="{{ arches | length + 2 }}">
|
||||||
|
Mirror list filtering matrix
|
||||||
|
</th>
|
||||||
|
</tr>
|
||||||
|
<tr id="matrixheadings" >
|
||||||
<th>Projects</th>
|
<th>Projects</th>
|
||||||
<th>Versions</th>
|
<th>Versions</th>
|
||||||
<th colspan="{{ arches | length}}">
|
<th colspan="{{ arches | length}}">
|
||||||
|
|
|
@ -1,5 +1,7 @@
|
||||||
{% extends "master.html" %}
|
{% extends "master.html" %}
|
||||||
|
|
||||||
|
{% from "_macros.html" import pagination_bar %}
|
||||||
|
|
||||||
{% block title %}Mirrors{% endblock %}
|
{% block title %}Mirrors{% endblock %}
|
||||||
{%block tag %}mirrors{% endblock %}
|
{%block tag %}mirrors{% endblock %}
|
||||||
|
|
||||||
|
@ -29,8 +31,8 @@
|
||||||
<th>Internet2</th>
|
<th>Internet2</th>
|
||||||
<th>Comment</th>
|
<th>Comment</th>
|
||||||
</tr>
|
</tr>
|
||||||
{% for mirror in mirrors %}
|
{% for mirror in mirrors.items %}
|
||||||
<tr>
|
<tr class="mirror-row">
|
||||||
<td>{{ mirror.country }}</td>
|
<td>{{ mirror.country }}</td>
|
||||||
<td>
|
<td>
|
||||||
<a href="{{ mirror.site.org_url }}">
|
<a href="{{ mirror.site.org_url }}">
|
||||||
|
@ -58,6 +60,7 @@
|
||||||
</tr>
|
</tr>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</table>
|
</table>
|
||||||
|
{{pagination_bar(mirrors)}}
|
||||||
{% else %}
|
{% else %}
|
||||||
<p>
|
<p>
|
||||||
There are currently no active mirrors registered.
|
There are currently no active mirrors registered.
|
||||||
|
|
Loading…
Reference in a new issue