diff --git a/.docker/base/Dockerfile b/.docker/base/Dockerfile index e8b2c960..e6ac2aef 100644 --- a/.docker/base/Dockerfile +++ b/.docker/base/Dockerfile @@ -116,6 +116,7 @@ COPY .docker/base/dbus /usr/bin/dbus COPY .docker/base/default.pa /etc/pulse/default.pa COPY .docker/base/supervisord.conf /etc/neko/supervisord.conf COPY .docker/base/xorg.conf /etc/neko/xorg.conf +COPY .docker/base/rendergroup-init.conf /etc/neko/supervisord/rendergroup-init.conf COPY .docker/base/add-render-group.sh /usr/bin/add-render-group.sh # diff --git a/.docker/base/Dockerfile.arm64 b/.docker/base/Dockerfile.arm64 new file mode 100644 index 00000000..65cf0edd --- /dev/null +++ b/.docker/base/Dockerfile.arm64 @@ -0,0 +1,129 @@ +# +# STAGE 1: SERVER +# +FROM golang:1.18-bullseye as server +WORKDIR /src + +# +# install dependencies +RUN set -eux; apt-get update; \ + apt-get install -y --no-install-recommends git cmake make libx11-dev libxrandr-dev libxtst-dev \ + libgstreamer1.0-dev libgstreamer-plugins-base1.0-dev gstreamer1.0-plugins-base gstreamer1.0-plugins-good gstreamer1.0-plugins-bad gstreamer1.0-plugins-ugly; \ + # + # install libclipboard + set -eux; \ + cd /tmp; \ + git clone --depth=1 https://github.com/jtanx/libclipboard; \ + cd libclipboard; \ + cmake .; \ + make -j4; \ + make install; \ + rm -rf /tmp/libclipboard; \ + # + # clean up + apt-get clean -y; \ + rm -rf /var/lib/apt/lists/* /var/cache/apt/* + +# +# build server +COPY server/ . +RUN go get -v -t -d . && go build -o bin/neko cmd/neko/main.go + +# +# STAGE 2: CLIENT +# +FROM node:14-bullseye-slim as client +WORKDIR /src + +# +# install dependencies +COPY client/package*.json ./ +RUN npm install + +# +# build client +COPY client/ . +RUN npm run build + +# +# STAGE 3: RUNTIME +# +FROM debian:bullseye-slim + +# +# avoid warnings by switching to noninteractive +ENV DEBIAN_FRONTEND=noninteractive + +# +# set custom user +ARG USERNAME=neko +ARG USER_UID=1000 +ARG USER_GID=$USER_UID + +RUN set -eux; \ + apt-get update; \ + # + # install dependencies + apt-get install -y --no-install-recommends wget ca-certificates supervisor; \ + apt-get install -y --no-install-recommends pulseaudio dbus-x11 xserver-xorg-video-dummy; \ + apt-get install -y --no-install-recommends libcairo2 libxcb1 libxrandr2 libxv1 libopus0 libvpx6; \ + # + # gst + vaapi plugin + apt-get install -y --no-install-recommends libgstreamer1.0-dev libgstreamer-plugins-base1.0-dev \ + gstreamer1.0-plugins-base gstreamer1.0-plugins-good gstreamer1.0-plugins-bad gstreamer1.0-plugins-ugly gstreamer1.0-pulseaudio;\ + # + # fonts + apt-get install -y --no-install-recommends fonts-takao-mincho fonts-wqy-zenhei; \ + # + # create a non-root user + groupadd --gid $USER_GID $USERNAME; \ + useradd --uid $USER_UID --gid $USERNAME --shell /bin/bash --create-home $USERNAME; \ + adduser $USERNAME audio; \ + adduser $USERNAME video; \ + adduser $USERNAME pulse; \ + # + # setup pulseaudio + mkdir -p /home/$USERNAME/.config/pulse/; \ + echo "default-server=unix:/tmp/pulseaudio.socket" > /home/$USERNAME/.config/pulse/client.conf; \ + # + # workaround for an X11 problem: http://blog.tigerteufel.de/?p=476 + mkdir /tmp/.X11-unix; \ + chmod 1777 /tmp/.X11-unix; \ + chown $USERNAME /tmp/.X11-unix/; \ + # + # make directories for neko + mkdir -p /etc/neko /var/www /var/log/neko; \ + chmod 1777 /var/log/neko; \ + chown $USERNAME /var/log/neko/; \ + chown -R $USERNAME:$USERNAME /home/$USERNAME; \ + # + # clean up + apt-get clean -y; \ + rm -rf /var/lib/apt/lists/* /var/cache/apt/* + +# +# copy config files +COPY .docker/base/dbus /usr/bin/dbus +COPY .docker/base/default.pa /etc/pulse/default.pa +COPY .docker/base/supervisord.conf /etc/neko/supervisord.conf +COPY .docker/base/xorg.conf /etc/neko/xorg.conf + +# +# set default envs +ENV USER=$USERNAME +ENV DISPLAY=:99.0 +ENV NEKO_PASSWORD=neko +ENV NEKO_PASSWORD_ADMIN=admin +ENV NEKO_BIND=:8080 + +# +# copy static files from previous stages +COPY --from=server /src/bin/neko /usr/bin/neko +COPY --from=client /src/dist/ /var/www + +HEALTHCHECK --interval=10s --timeout=5s --retries=8 \ + CMD wget -O - http://localhost:${NEKO_BIND#*:}/health || exit 1 + +# +# run neko +CMD ["/usr/bin/supervisord", "-c", "/etc/neko/supervisord.conf"] diff --git a/.docker/base/rendergroup-init.conf b/.docker/base/rendergroup-init.conf new file mode 100644 index 00000000..8a187080 --- /dev/null +++ b/.docker/base/rendergroup-init.conf @@ -0,0 +1,12 @@ +[program:rendergroup-init] +environment=RENDER_GID="%(ENV_RENDER_GID)s",USER="%(ENV_USER)s" +command=/usr/bin/add-render-group.sh +startsecs=0 +startretries=0 +autorestart=false +priority=10 +user=root +stdout_logfile=/var/log/neko/rendergroup.log +stdout_logfile_maxbytes=1MB +stdout_logfile_backups=10 +redirect_stderr=true diff --git a/.github/workflows/tags.yml b/.github/workflows/tags.yml index 16dfd70f..214ddf6d 100644 --- a/.github/workflows/tags.yml +++ b/.github/workflows/tags.yml @@ -10,7 +10,7 @@ env: IMAGE_NAME: m1k1o/neko jobs: - build-base: + build-base-amd64: runs-on: ubuntu-latest # # do not run on forks @@ -50,7 +50,52 @@ jobs: with: context: ./ file: .docker/base/Dockerfile - platforms: linux/amd64,linux/arm64 + platforms: linux/amd64 + push: true + tags: ${{ steps.meta.outputs.tags }} + labels: ${{ steps.meta.outputs.labels }} + + build-base-arm64: + runs-on: ubuntu-latest + # + # do not run on forks + # + if: github.repository_owner == 'm1k1o' + steps: + - + name: Checkout + uses: actions/checkout@v2 + - + name: Set up QEMU + uses: docker/setup-qemu-action@v1 + - + name: Set up Docker Buildx + uses: docker/setup-buildx-action@v1 + - + name: Extract metadata (tags, labels) for Docker + uses: docker/metadata-action@v3 + id: meta + with: + images: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }}/base + tags: | + type=semver,pattern={{version}} + type=semver,pattern={{major}}.{{minor}} + type=semver,pattern={{major}} + type=sha,format=long + - + name: Log in to the Container registry + uses: docker/login-action@v1 + with: + registry: ${{ env.REGISTRY }} + username: ${{ github.actor }} + password: ${{ secrets.GHCR_ACCESS_TOKEN }} + - + name: Build and push + uses: docker/build-push-action@v2 + with: + context: ./ + file: .docker/base/Dockerfile + platforms: linux/arm64 push: true tags: ${{ steps.meta.outputs.tags }} labels: ${{ steps.meta.outputs.labels }} @@ -61,7 +106,7 @@ jobs: # do not run on forks # if: github.repository_owner == 'm1k1o' - needs: [ build-base ] + needs: [ build-base-amd64, build-base-arm64 ] strategy: # Will build all images even if some fail. matrix: