Merge pull request #2379 from dbnicholson/gh-actions-tests
Replace Travis CI with GitHub Actions
This commit is contained in:
commit
b6a55ab8f3
|
|
@ -0,0 +1,27 @@
|
||||||
|
---
|
||||||
|
name: Rust
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches: [main]
|
||||||
|
pull_request:
|
||||||
|
branches: [main]
|
||||||
|
|
||||||
|
env:
|
||||||
|
CARGO_TERM_COLOR: always
|
||||||
|
ACTIONS_LINTS_TOOLCHAIN: 1.50.0
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
linting:
|
||||||
|
name: "Lints, pinned toolchain"
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout repository
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
- name: Install toolchain
|
||||||
|
uses: actions-rs/toolchain@v1
|
||||||
|
with:
|
||||||
|
toolchain: ${{ env['ACTIONS_LINTS_TOOLCHAIN'] }}
|
||||||
|
default: true
|
||||||
|
components: rustfmt, clippy
|
||||||
|
- name: cargo fmt (check)
|
||||||
|
run: cargo fmt -- --check -l
|
||||||
|
|
@ -1,27 +1,106 @@
|
||||||
---
|
---
|
||||||
name: Rust
|
name: Tests
|
||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
branches: [main]
|
branches: [main]
|
||||||
pull_request:
|
pull_request:
|
||||||
branches: [main]
|
branches: [main]
|
||||||
|
|
||||||
env:
|
|
||||||
CARGO_TERM_COLOR: always
|
|
||||||
ACTIONS_LINTS_TOOLCHAIN: 1.50.0
|
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
linting:
|
tests:
|
||||||
name: "Lints, pinned toolchain"
|
# Distro configuration matrix
|
||||||
|
#
|
||||||
|
# Each build is run in a Docker container specific to the distro.
|
||||||
|
# When adding a new distro, handle the dependency installation in
|
||||||
|
# `ci/gh-install.sh`. The matrix configuration options are:
|
||||||
|
#
|
||||||
|
# name: A friendly name to use for the job.
|
||||||
|
#
|
||||||
|
# image: The Docker image to use.
|
||||||
|
#
|
||||||
|
# pre-checkout-setup: Commands to run before the git repo checkout.
|
||||||
|
# If git is not in the Docker image, it must be installed here.
|
||||||
|
# Otherwise, the checkout action uses the GitHub REST API, which
|
||||||
|
# doesn't result in an actual git repo. A real git repo is
|
||||||
|
# required to checkout the submodules.
|
||||||
|
#
|
||||||
|
# extra-packages: Packages to install in addition to those in
|
||||||
|
# `ci/gh-install.sh`. This can be used to support features from
|
||||||
|
# additional `configure` options.
|
||||||
|
#
|
||||||
|
# configure-options: Options to pass to `configure`.
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
include:
|
||||||
|
- name: Debian Buster (10) with sign-ed25519
|
||||||
|
image: debian:buster-slim
|
||||||
|
pre-checkout-setup: |
|
||||||
|
apt-get update
|
||||||
|
apt-get install -y git
|
||||||
|
extra-packages: >-
|
||||||
|
libsodium-dev
|
||||||
|
configure-options: >-
|
||||||
|
--with-ed25519-libsodium
|
||||||
|
|
||||||
|
- name: Debian Buster (10) with curl, sign-ed25519 and no gpgme
|
||||||
|
image: debian:buster-slim
|
||||||
|
pre-checkout-setup: |
|
||||||
|
apt-get update
|
||||||
|
apt-get install -y git
|
||||||
|
extra-packages: >-
|
||||||
|
libsodium-dev
|
||||||
|
configure-options: >-
|
||||||
|
--with-curl
|
||||||
|
--with-ed25519-libsodium
|
||||||
|
--without-gpgme
|
||||||
|
|
||||||
|
# A 32 bit build to act as a proxy for frequently deployed 32
|
||||||
|
# bit armv7
|
||||||
|
- name: Debian Buster (10) 32 bit
|
||||||
|
image: i386/debian:buster-slim
|
||||||
|
# This is pretty nasty. The checkout action uses an x86_64
|
||||||
|
# node binary in the container, so we need to provide an
|
||||||
|
# x86_64 ld.so and libstdc++.
|
||||||
|
pre-checkout-setup: |
|
||||||
|
dpkg --add-architecture amd64
|
||||||
|
apt-get update
|
||||||
|
apt-get install -y git libc6:amd64 libstdc++6:amd64
|
||||||
|
|
||||||
|
- name: Ubuntu Focal (20.04)
|
||||||
|
image: ubuntu:focal
|
||||||
|
pre-checkout-setup: |
|
||||||
|
apt-get update
|
||||||
|
apt-get install -y git
|
||||||
|
|
||||||
|
- name: Ubuntu Groovy (20.10)
|
||||||
|
image: ubuntu:groovy
|
||||||
|
pre-checkout-setup: |
|
||||||
|
apt-get update
|
||||||
|
apt-get install -y git
|
||||||
|
|
||||||
|
name: ${{ matrix.name }}
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
container:
|
||||||
|
image: ${{ matrix.image }}
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
|
- name: Pre-checkout setup
|
||||||
|
run: ${{ matrix.pre-checkout-setup }}
|
||||||
|
if: ${{ matrix.pre-checkout-setup }}
|
||||||
|
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@v2
|
uses: actions/checkout@v2
|
||||||
- name: Install toolchain
|
|
||||||
uses: actions-rs/toolchain@v1
|
|
||||||
with:
|
with:
|
||||||
toolchain: ${{ env['ACTIONS_LINTS_TOOLCHAIN'] }}
|
submodules: true
|
||||||
default: true
|
|
||||||
components: rustfmt, clippy
|
- name: Install dependencies
|
||||||
- name: cargo fmt (check)
|
run: ./ci/gh-install.sh ${{ matrix.extra-packages }}
|
||||||
run: cargo fmt -- --check -l
|
|
||||||
|
- name: Build and test
|
||||||
|
run: ./ci/gh-build.sh ${{ matrix.configure-options }}
|
||||||
|
env:
|
||||||
|
# GitHub hosted runners currently have 2 CPUs, so run 2
|
||||||
|
# parallel make jobs.
|
||||||
|
#
|
||||||
|
# https://docs.github.com/en/actions/using-github-hosted-runners/about-github-hosted-runners
|
||||||
|
MAKEFLAGS: -j2
|
||||||
|
|
|
||||||
18
.travis.yml
18
.travis.yml
|
|
@ -1,18 +0,0 @@
|
||||||
language: c
|
|
||||||
sudo: required
|
|
||||||
|
|
||||||
env:
|
|
||||||
# debian has libsodium-dev, ubuntu doesn't in core at least
|
|
||||||
- ci_docker=debian:buster-slim ci_distro=debian ci_suite=buster ci_configopts="--with-ed25519-libsodium" ci_pkgs="libsodium-dev"
|
|
||||||
- ci_docker=debian:buster-slim ci_distro=debian ci_suite=buster ci_configopts="--with-curl --with-ed25519-libsodium --without-gpgme" ci_pkgs="libsodium-dev"
|
|
||||||
# A 32 bit build to act as a proxy for frequently deployed 32 bit armv7
|
|
||||||
- ci_docker=i386/debian:buster-slim ci_distro=debian ci_suite=buster
|
|
||||||
- ci_docker=ubuntu:focal ci_distro=ubuntu ci_suite=focal
|
|
||||||
- ci_docker=ubuntu:groovy ci_distro=ubuntu ci_suite=groovy
|
|
||||||
|
|
||||||
script:
|
|
||||||
- ci/travis-install.sh
|
|
||||||
- ci_parallel=2 ci_sudo=yes ci/travis-build.sh
|
|
||||||
|
|
||||||
notifications:
|
|
||||||
email: false
|
|
||||||
|
|
@ -0,0 +1,62 @@
|
||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# Copyright © 2015-2016 Collabora Ltd.
|
||||||
|
# Copyright © 2021 Endless OS Foundation LLC
|
||||||
|
#
|
||||||
|
# Permission is hereby granted, free of charge, to any person
|
||||||
|
# obtaining a copy of this software and associated documentation files
|
||||||
|
# (the "Software"), to deal in the Software without restriction,
|
||||||
|
# including without limitation the rights to use, copy, modify, merge,
|
||||||
|
# publish, distribute, sublicense, and/or sell copies of the Software,
|
||||||
|
# and to permit persons to whom the Software is furnished to do so,
|
||||||
|
# subject to the following conditions:
|
||||||
|
#
|
||||||
|
# The above copyright notice and this permission notice shall be
|
||||||
|
# included in all copies or substantial portions of the Software.
|
||||||
|
#
|
||||||
|
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||||||
|
# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
|
||||||
|
# BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
|
||||||
|
# ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
||||||
|
# CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
# SOFTWARE.
|
||||||
|
|
||||||
|
set -euo pipefail
|
||||||
|
set -x
|
||||||
|
|
||||||
|
NOCONFIGURE=1 ./autogen.sh
|
||||||
|
|
||||||
|
srcdir="$(pwd)"
|
||||||
|
mkdir ci-build
|
||||||
|
cd ci-build
|
||||||
|
|
||||||
|
make="make V=1 VERBOSE=1"
|
||||||
|
|
||||||
|
../configure \
|
||||||
|
--enable-always-build-tests \
|
||||||
|
"$@"
|
||||||
|
|
||||||
|
${make}
|
||||||
|
|
||||||
|
# Run the tests both using check and distcheck and dump the logs on
|
||||||
|
# failures. For distcheck the logs will be inside the dist directory, so
|
||||||
|
# tell make to use the current directory.
|
||||||
|
if ! ${make} check; then
|
||||||
|
cat test-suite.log || :
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
if ! ${make} distcheck \
|
||||||
|
TEST_SUITE_LOG=$(pwd)/test-suite.log \
|
||||||
|
DISTCHECK_CONFIGURE_FLAGS="$*"
|
||||||
|
then
|
||||||
|
cat test-suite.log || :
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Show the installed files
|
||||||
|
${make} install DESTDIR=$(pwd)/DESTDIR
|
||||||
|
( cd DESTDIR && find . )
|
||||||
|
|
||||||
|
# vim:set sw=4 sts=4 et:
|
||||||
|
|
@ -1,6 +1,7 @@
|
||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
|
|
||||||
# Copyright © 2015-2016 Collabora Ltd.
|
# Copyright © 2015-2016 Collabora Ltd.
|
||||||
|
# Copyright © 2021 Endless OS Foundation LLC
|
||||||
#
|
#
|
||||||
# Permission is hereby granted, free of charge, to any person
|
# Permission is hereby granted, free of charge, to any person
|
||||||
# obtaining a copy of this software and associated documentation files
|
# obtaining a copy of this software and associated documentation files
|
||||||
|
|
@ -27,68 +28,18 @@ set -x
|
||||||
|
|
||||||
NULL=
|
NULL=
|
||||||
|
|
||||||
# ci_distro:
|
# Get the OS release info
|
||||||
# OS distribution in which we are testing
|
. /etc/os-release
|
||||||
# Typical values: ubuntu, debian; maybe fedora in future
|
|
||||||
: "${ci_distro:=debian}"
|
|
||||||
|
|
||||||
# ci_docker:
|
case "$ID" in
|
||||||
# If non-empty, this is the name of a Docker image. travis-install.sh will
|
|
||||||
# fetch it with "docker pull" and use it as a base for a new Docker image
|
|
||||||
# named "ci-image" in which we will do our testing.
|
|
||||||
: "${ci_docker:=}"
|
|
||||||
|
|
||||||
# ci_in_docker:
|
|
||||||
# Used internally by travis-install.sh. If yes, we are inside the Docker image
|
|
||||||
# (ci_docker is empty in this case).
|
|
||||||
: "${ci_in_docker:=no}"
|
|
||||||
|
|
||||||
# ci_suite:
|
|
||||||
# OS suite (release, branch) in which we are testing.
|
|
||||||
# Typical values for ci_distro=ubuntu: xenial, trusty
|
|
||||||
# Typical values for ci_distro=debian: sid, jessie
|
|
||||||
# Typical values for ci_distro=fedora might be 25, rawhide
|
|
||||||
: "${ci_suite:=stretch}"
|
|
||||||
|
|
||||||
# ci_configopts: Additional arguments for configure
|
|
||||||
: "${ci_configopts:=}"
|
|
||||||
|
|
||||||
# ci_pkgs: Additional packages to be installed
|
|
||||||
: "${ci_pkgs:=}"
|
|
||||||
|
|
||||||
if [ $(id -u) = 0 ]; then
|
|
||||||
sudo=
|
|
||||||
else
|
|
||||||
sudo=sudo
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ -n "$ci_docker" ]; then
|
|
||||||
sed \
|
|
||||||
-e "s,@ci_distro@,${ci_distro}," \
|
|
||||||
-e "s,@ci_docker@,${ci_docker}," \
|
|
||||||
-e "s,@ci_suite@,${ci_suite}," \
|
|
||||||
-e "s,@ci_pkgs@,${ci_pkgs}," \
|
|
||||||
< ci/travis-Dockerfile.in > Dockerfile
|
|
||||||
exec docker build -t ci-image .
|
|
||||||
fi
|
|
||||||
|
|
||||||
case "$ci_distro" in
|
|
||||||
(debian)
|
|
||||||
# Docker images use httpredir.debian.org but it seems to be
|
|
||||||
# unreliable; use a CDN instead
|
|
||||||
sed -i -e 's/httpredir\.debian\.org/deb.debian.org/g' /etc/apt/sources.list
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
|
|
||||||
case "$ci_distro" in
|
|
||||||
(debian|ubuntu)
|
(debian|ubuntu)
|
||||||
# Make debconf run non-interactively since its questions can't
|
# Make debconf run non-interactively since its questions can't
|
||||||
# be answered.
|
# be answered.
|
||||||
export DEBIAN_FRONTEND=noninteractive
|
export DEBIAN_FRONTEND=noninteractive
|
||||||
|
|
||||||
# TODO: fetch this list from the Debian packaging git repository?
|
# TODO: fetch this list from the Debian packaging git repository?
|
||||||
$sudo apt-get -y update
|
apt-get -y update
|
||||||
$sudo apt-get -y install \
|
apt-get -y install \
|
||||||
attr \
|
attr \
|
||||||
bison \
|
bison \
|
||||||
cpio \
|
cpio \
|
||||||
|
|
@ -107,6 +58,7 @@ case "$ci_distro" in
|
||||||
libarchive-dev \
|
libarchive-dev \
|
||||||
libattr1-dev \
|
libattr1-dev \
|
||||||
libcap-dev \
|
libcap-dev \
|
||||||
|
libcurl4-openssl-dev \
|
||||||
libfuse-dev \
|
libfuse-dev \
|
||||||
libgirepository1.0-dev \
|
libgirepository1.0-dev \
|
||||||
libglib2.0-dev \
|
libglib2.0-dev \
|
||||||
|
|
@ -115,25 +67,16 @@ case "$ci_distro" in
|
||||||
libmount-dev \
|
libmount-dev \
|
||||||
libselinux1-dev \
|
libselinux1-dev \
|
||||||
libsoup2.4-dev \
|
libsoup2.4-dev \
|
||||||
libcurl4-openssl-dev \
|
libsystemd-dev \
|
||||||
procps \
|
procps \
|
||||||
zlib1g-dev \
|
|
||||||
python3-yaml \
|
python3-yaml \
|
||||||
${ci_pkgs:-} \
|
systemd \
|
||||||
${NULL}
|
zlib1g-dev \
|
||||||
|
"$@"
|
||||||
if [ "$ci_in_docker" = yes ]; then
|
|
||||||
# Add the user that we will use to do the build inside the
|
|
||||||
# Docker container, and let them use sudo
|
|
||||||
adduser --disabled-password user </dev/null
|
|
||||||
apt-get -y install sudo systemd-sysv
|
|
||||||
echo "user ALL=(ALL) NOPASSWD: ALL" > /etc/sudoers.d/nopasswd
|
|
||||||
chmod 0440 /etc/sudoers.d/nopasswd
|
|
||||||
fi
|
|
||||||
;;
|
;;
|
||||||
|
|
||||||
(*)
|
(*)
|
||||||
echo "Don't know how to set up ${ci_distro}" >&2
|
echo "Don't know how to set up ${ID}" >&2
|
||||||
exit 1
|
exit 1
|
||||||
;;
|
;;
|
||||||
esac
|
esac
|
||||||
|
|
@ -1,10 +0,0 @@
|
||||||
FROM @ci_docker@
|
|
||||||
ENV container docker
|
|
||||||
|
|
||||||
ADD ci/travis-install.sh /travis-install.sh
|
|
||||||
RUN ci_suite="@ci_suite@" ci_distro="@ci_distro@" ci_pkgs="@ci_pkgs@" ci_in_docker=yes /travis-install.sh
|
|
||||||
|
|
||||||
ADD . /home/user/ostree
|
|
||||||
RUN chown -R user:user /home/user/ostree
|
|
||||||
WORKDIR /home/user/ostree
|
|
||||||
USER user
|
|
||||||
|
|
@ -1,110 +0,0 @@
|
||||||
#!/bin/bash
|
|
||||||
|
|
||||||
# Copyright © 2015-2016 Collabora Ltd.
|
|
||||||
#
|
|
||||||
# Permission is hereby granted, free of charge, to any person
|
|
||||||
# obtaining a copy of this software and associated documentation files
|
|
||||||
# (the "Software"), to deal in the Software without restriction,
|
|
||||||
# including without limitation the rights to use, copy, modify, merge,
|
|
||||||
# publish, distribute, sublicense, and/or sell copies of the Software,
|
|
||||||
# and to permit persons to whom the Software is furnished to do so,
|
|
||||||
# subject to the following conditions:
|
|
||||||
#
|
|
||||||
# The above copyright notice and this permission notice shall be
|
|
||||||
# included in all copies or substantial portions of the Software.
|
|
||||||
#
|
|
||||||
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
||||||
# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
||||||
# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
||||||
# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
|
|
||||||
# BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
|
|
||||||
# ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
||||||
# CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
# SOFTWARE.
|
|
||||||
|
|
||||||
set -euo pipefail
|
|
||||||
set -x
|
|
||||||
|
|
||||||
NULL=
|
|
||||||
|
|
||||||
# ci_docker:
|
|
||||||
# If non-empty, this is the name of a Docker image. travis-install.sh will
|
|
||||||
# fetch it with "docker pull" and use it as a base for a new Docker image
|
|
||||||
# named "ci-image" in which we will do our testing.
|
|
||||||
#
|
|
||||||
# If empty, we test on "bare metal".
|
|
||||||
# Typical values: ubuntu:xenial, debian:jessie-slim
|
|
||||||
: "${ci_docker:=}"
|
|
||||||
|
|
||||||
# ci_parallel:
|
|
||||||
# A number of parallel jobs, passed to make -j
|
|
||||||
: "${ci_parallel:=1}"
|
|
||||||
|
|
||||||
# ci_sudo:
|
|
||||||
# If yes, assume we can get root using sudo; if no, only use current user
|
|
||||||
: "${ci_sudo:=no}"
|
|
||||||
|
|
||||||
# ci_test:
|
|
||||||
# If yes, run tests; if no, just build
|
|
||||||
: "${ci_test:=yes}"
|
|
||||||
|
|
||||||
# ci_test_fatal:
|
|
||||||
# If yes, test failures break the build; if no, they are reported but ignored
|
|
||||||
: "${ci_test_fatal:=yes}"
|
|
||||||
|
|
||||||
# ci_configopts:
|
|
||||||
# Additional args for configure
|
|
||||||
: "${ci_configopts:=}"
|
|
||||||
|
|
||||||
if [ -n "$ci_docker" ]; then
|
|
||||||
exec docker run \
|
|
||||||
--env=ci_docker="" \
|
|
||||||
--env=ci_parallel="${ci_parallel}" \
|
|
||||||
--env=ci_sudo=yes \
|
|
||||||
--env=ci_test="${ci_test}" \
|
|
||||||
--env=ci_test_fatal="${ci_test_fatal}" \
|
|
||||||
--env=ci_configopts="${ci_configopts}" \
|
|
||||||
--privileged \
|
|
||||||
ci-image \
|
|
||||||
ci/travis-build.sh
|
|
||||||
fi
|
|
||||||
|
|
||||||
maybe_fail_tests () {
|
|
||||||
if [ "$ci_test_fatal" = yes ]; then
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
NOCONFIGURE=1 ./autogen.sh
|
|
||||||
|
|
||||||
srcdir="$(pwd)"
|
|
||||||
mkdir ci-build
|
|
||||||
cd ci-build
|
|
||||||
|
|
||||||
make="make -j${ci_parallel} V=1 VERBOSE=1"
|
|
||||||
|
|
||||||
../configure \
|
|
||||||
--enable-always-build-tests \
|
|
||||||
${ci_configopts} \
|
|
||||||
"$@"
|
|
||||||
|
|
||||||
${make}
|
|
||||||
[ "$ci_test" = no ] || ${make} check || maybe_fail_tests
|
|
||||||
cat test-suite.log || :
|
|
||||||
[ "$ci_test" = no ] || ${make} distcheck DISTCHECK_CONFIGURE_FLAGS="${ci_configopts}" || maybe_fail_tests
|
|
||||||
cat test-suite.log || :
|
|
||||||
|
|
||||||
${make} install DESTDIR=$(pwd)/DESTDIR
|
|
||||||
( cd DESTDIR && find . )
|
|
||||||
|
|
||||||
if [ "$ci_sudo" = yes ] && [ "$ci_test" = yes ]; then
|
|
||||||
sudo ${make} install
|
|
||||||
env \
|
|
||||||
LD_LIBRARY_PATH=/usr/local/lib \
|
|
||||||
GI_TYPELIB_PATH=/usr/local/lib/girepository-1.0 \
|
|
||||||
${make} installcheck || \
|
|
||||||
maybe_fail_tests
|
|
||||||
cat test-suite.log || :
|
|
||||||
fi
|
|
||||||
|
|
||||||
# vim:set sw=4 sts=4 et:
|
|
||||||
Loading…
Reference in New Issue