mirror of
https://github.com/vdsm/virtual-dsm.git
synced 2025-11-07 02:23:42 +08:00
Compare commits
141 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
5332d387f4 | ||
|
|
f0d08ef263 | ||
|
|
e4334f9499 | ||
|
|
627ec56262 | ||
|
|
251cf8121e | ||
|
|
87fad1b0e9 | ||
|
|
698516ac8c | ||
|
|
dae5d75674 | ||
|
|
95facffa9b | ||
|
|
682e0a9952 | ||
|
|
9a97dfdc70 | ||
|
|
2f383699f9 | ||
|
|
8137a137b3 | ||
|
|
1339d51796 | ||
|
|
dce447c974 | ||
|
|
ef5b650991 | ||
|
|
fd19c7b4f3 | ||
|
|
727297642c | ||
|
|
cd457801e7 | ||
|
|
392e7afdfe | ||
|
|
b425e34907 | ||
|
|
7c2e2fc7b1 | ||
|
|
9bac0c94a8 | ||
|
|
631568681e | ||
|
|
6599861dbb | ||
|
|
3bcd831531 | ||
|
|
fafd4a4fca | ||
|
|
5b69178f08 | ||
|
|
479a30369d | ||
|
|
ba522a4acb | ||
|
|
3ab6d2c418 | ||
|
|
b71b2245cb | ||
|
|
bbea0eb429 | ||
|
|
c28c5cfbaa | ||
|
|
be3fd30cb2 | ||
|
|
e3942da906 | ||
|
|
d66be1a228 | ||
|
|
72085d3711 | ||
|
|
e7cdbb1db5 | ||
|
|
7a592e0cea | ||
|
|
107a4b87d5 | ||
|
|
8b0ec3bef7 | ||
|
|
aaded40a4f | ||
|
|
7f77bb88ab | ||
|
|
95c3b2caad | ||
|
|
150c450bf6 | ||
|
|
932c23afba | ||
|
|
2e0107e46f | ||
|
|
d22a3a4c7d | ||
|
|
f93f870626 | ||
|
|
a2e55c5dda | ||
|
|
32748509ea | ||
|
|
970a662170 | ||
|
|
8925323a6e | ||
|
|
10915a601c | ||
|
|
2cc1af19b1 | ||
|
|
6670ca4fe1 | ||
|
|
d3f77c848c | ||
|
|
3f2ca67051 | ||
|
|
3812101366 | ||
|
|
db72acfc4f | ||
|
|
1b3d760f5f | ||
|
|
539f5de6d9 | ||
|
|
fe2d072056 | ||
|
|
469ee67942 | ||
|
|
95991d8f5d | ||
|
|
7e12585429 | ||
|
|
c335078aac | ||
|
|
f1fbbb5623 | ||
|
|
b6502e0a38 | ||
|
|
2fab3e5897 | ||
|
|
a4ea89d6e7 | ||
|
|
c451f253fa | ||
|
|
03121b6c6d | ||
|
|
007d20c315 | ||
|
|
26d6fa9fcc | ||
|
|
b9f3e52ba4 | ||
|
|
03d2665725 | ||
|
|
ba7fd2fe4a | ||
|
|
a8bcae16a4 | ||
|
|
2f19d31a81 | ||
|
|
54692e3a75 | ||
|
|
029235a34d | ||
|
|
180573d69f | ||
|
|
8fa900335a | ||
|
|
a527080ccd | ||
|
|
ce6d60c611 | ||
|
|
ff9fd9b377 | ||
|
|
9e61be15e6 | ||
|
|
b1d53b42ca | ||
|
|
143a2151fb | ||
|
|
7fd29e30b3 | ||
|
|
efe46e1fdc | ||
|
|
2cf4ca07f4 | ||
|
|
b88207f0dd | ||
|
|
70e10b1d56 | ||
|
|
ced994d94a | ||
|
|
354bd2429b | ||
|
|
c1d3d15d4e | ||
|
|
95b2b83ac6 | ||
|
|
c3c4d966b4 | ||
|
|
a768fecfde | ||
|
|
01e41a4014 | ||
|
|
eb4852683b | ||
|
|
6218333fec | ||
|
|
f32d8cbefc | ||
|
|
3e985502c2 | ||
|
|
ad3132e8c2 | ||
|
|
830ace0e47 | ||
|
|
1c36893729 | ||
|
|
a87aaab6f7 | ||
|
|
21699b8960 | ||
|
|
87ee25d404 | ||
|
|
c1714f9e6b | ||
|
|
754765b766 | ||
|
|
419f0cf571 | ||
|
|
55d9ac521f | ||
|
|
3406b3b471 | ||
|
|
f067ad2458 | ||
|
|
7eafd0a969 | ||
|
|
116f30bc0a | ||
|
|
04aa20e836 | ||
|
|
3bf4cc861b | ||
|
|
3c6620a3f9 | ||
|
|
ab0ea5a1d8 | ||
|
|
f894ad2686 | ||
|
|
570340d4e5 | ||
|
|
7dbe706282 | ||
|
|
0c9559f695 | ||
|
|
3113e2b64e | ||
|
|
f0ce992a27 | ||
|
|
6334cfc8bc | ||
|
|
451a569617 | ||
|
|
44d82d6544 | ||
|
|
618ec66401 | ||
|
|
d24ae86c12 | ||
|
|
32db74e50d | ||
|
|
503c89f08c | ||
|
|
c9e6e65991 | ||
|
|
04bd8a1639 | ||
|
|
a024294e19 |
6
.github/renovate.json
vendored
6
.github/renovate.json
vendored
@@ -1,6 +1,4 @@
|
|||||||
{
|
{
|
||||||
"extends": [
|
"$schema": "https://docs.renovatebot.com/renovate-schema.json",
|
||||||
"config:base",
|
"extends": ["config:recommended", ":disableDependencyDashboard"]
|
||||||
":disableDependencyDashboard"
|
|
||||||
]
|
|
||||||
}
|
}
|
||||||
|
|||||||
87
.github/workflows/build.yml
vendored
87
.github/workflows/build.yml
vendored
@@ -7,12 +7,16 @@ on:
|
|||||||
- master
|
- master
|
||||||
paths-ignore:
|
paths-ignore:
|
||||||
- '**/*.md'
|
- '**/*.md'
|
||||||
- '**/*.yml'
|
- '**/*.yml'
|
||||||
- '.gitignore'
|
- '.gitignore'
|
||||||
- '.dockerignore'
|
- '.dockerignore'
|
||||||
- '.github/**'
|
- '.github/**'
|
||||||
- '.github/workflows/**'
|
- '.github/workflows/**'
|
||||||
|
|
||||||
|
concurrency:
|
||||||
|
group: build
|
||||||
|
cancel-in-progress: false
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
shellcheck:
|
shellcheck:
|
||||||
name: Check
|
name: Check
|
||||||
@@ -22,37 +26,34 @@ jobs:
|
|||||||
needs: shellcheck
|
needs: shellcheck
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
permissions:
|
permissions:
|
||||||
|
actions: write
|
||||||
packages: write
|
packages: write
|
||||||
contents: read
|
contents: read
|
||||||
steps:
|
steps:
|
||||||
-
|
|
||||||
name: Set up Docker Buildx
|
|
||||||
uses: docker/setup-buildx-action@v3
|
|
||||||
-
|
-
|
||||||
name: Checkout
|
name: Checkout
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
with:
|
with:
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
-
|
-
|
||||||
name: Prepare Docker build
|
name: Docker metadata
|
||||||
id: prepare
|
id: meta
|
||||||
run: |
|
uses: docker/metadata-action@v5
|
||||||
|
with:
|
||||||
PLATFORMS="linux/amd64,linux/arm64"
|
context: git
|
||||||
VERSION="${{ vars.MAJOR }}.${{ vars.MINOR }}"
|
images: |
|
||||||
|
${{ secrets.DOCKERHUB_REPO }}
|
||||||
TAGS=()
|
ghcr.io/${{ github.repository }}
|
||||||
TAGS=("${{ github.repository }}:latest")
|
tags: |
|
||||||
TAGS+=("${{ github.repository }}:${VERSION}")
|
type=raw,value=latest,priority=100
|
||||||
#TAGS+=("${{ secrets.DOCKERHUB_MIRROR }}:latest")
|
type=raw,value=${{ vars.MAJOR }}.${{ vars.MINOR }}
|
||||||
#TAGS+=("${{ secrets.DOCKERHUB_MIRROR }}:${VERSION}")
|
labels: |
|
||||||
TAGS+=("ghcr.io/${{ github.repository }}:latest")
|
org.opencontainers.image.title=${{ vars.NAME }}
|
||||||
TAGS+=("ghcr.io/${{ github.repository }}:${VERSION}")
|
env:
|
||||||
|
DOCKER_METADATA_ANNOTATIONS_LEVELS: manifest,index
|
||||||
echo "tags=${TAGS[@]}" >> $GITHUB_OUTPUT
|
-
|
||||||
echo "version=${VERSION}" >> $GITHUB_OUTPUT
|
name: Set up Docker Buildx
|
||||||
echo "docker_platforms=${PLATFORMS}" >> $GITHUB_OUTPUT
|
uses: docker/setup-buildx-action@v3
|
||||||
echo "build_date=$(date -u +'%Y-%m-%dT%H:%M:%SZ')" >> $GITHUB_OUTPUT
|
|
||||||
-
|
-
|
||||||
name: Login into Docker Hub
|
name: Login into Docker Hub
|
||||||
uses: docker/login-action@v3
|
uses: docker/login-action@v3
|
||||||
@@ -66,36 +67,26 @@ jobs:
|
|||||||
registry: ghcr.io
|
registry: ghcr.io
|
||||||
username: ${{ github.actor }}
|
username: ${{ github.actor }}
|
||||||
password: ${{ secrets.GITHUB_TOKEN }}
|
password: ${{ secrets.GITHUB_TOKEN }}
|
||||||
-
|
|
||||||
name: Build Docker image
|
|
||||||
run: |
|
|
||||||
|
|
||||||
TAGS=(${{ steps.prepare.outputs.tags }})
|
|
||||||
|
|
||||||
echo "Build date: ${{ steps.prepare.outputs.build_date }}"
|
|
||||||
echo "Docker platform: ${{ steps.prepare.outputs.docker_platforms }}"
|
|
||||||
echo "Tags: ${{ steps.prepare.outputs.tags }}"
|
|
||||||
|
|
||||||
docker buildx build --platform ${{ steps.prepare.outputs.docker_platforms }} \
|
|
||||||
--output "type=image,push=true" \
|
|
||||||
--progress=plain \
|
|
||||||
--build-arg "BUILD_ARG=${GITHUB_RUN_ID}" \
|
|
||||||
--build-arg "VERSION_ARG=${{ steps.prepare.outputs.version }}" \
|
|
||||||
--build-arg "DATE_ARG=${{ steps.prepare.outputs.build_date }}" \
|
|
||||||
--build-arg "VCS_REF=${GITHUB_SHA::8}" \
|
|
||||||
$(printf "%s" "${TAGS[@]/#/ --tag }" ) .
|
|
||||||
-
|
-
|
||||||
name: Clear Docker credentials
|
name: Build Docker image
|
||||||
run: |
|
uses: docker/build-push-action@v5
|
||||||
rm -f ${HOME}/.docker/config.json
|
with:
|
||||||
|
context: .
|
||||||
|
push: true
|
||||||
|
provenance: false
|
||||||
|
platforms: linux/amd64,linux/arm64,linux/arm
|
||||||
|
tags: ${{ steps.meta.outputs.tags }}
|
||||||
|
labels: ${{ steps.meta.outputs.labels }}
|
||||||
|
annotations: ${{ steps.meta.outputs.annotations }}
|
||||||
|
build-args: |
|
||||||
|
VERSION_ARG=${{ steps.meta.outputs.version }}
|
||||||
-
|
-
|
||||||
name: Create a release
|
name: Create a release
|
||||||
uses: action-pack/github-release@v2
|
uses: action-pack/github-release@v2
|
||||||
env:
|
|
||||||
GITHUB_TOKEN: ${{ secrets.REPO_ACCESS_TOKEN }}
|
|
||||||
with:
|
with:
|
||||||
tag: "v${{ steps.prepare.outputs.version }}"
|
tag: "v${{ steps.meta.outputs.version }}"
|
||||||
title: "v${{ steps.prepare.outputs.version }}"
|
title: "v${{ steps.meta.outputs.version }}"
|
||||||
|
token: ${{ secrets.REPO_ACCESS_TOKEN }}
|
||||||
-
|
-
|
||||||
name: Increment version variable
|
name: Increment version variable
|
||||||
uses: action-pack/bump@v2
|
uses: action-pack/bump@v2
|
||||||
|
|||||||
2
.github/workflows/check.yml
vendored
2
.github/workflows/check.yml
vendored
@@ -11,4 +11,4 @@ jobs:
|
|||||||
- name: Run ShellCheck
|
- name: Run ShellCheck
|
||||||
uses: ludeeus/action-shellcheck@master
|
uses: ludeeus/action-shellcheck@master
|
||||||
env:
|
env:
|
||||||
SHELLCHECK_OPTS: -x -e SC2001 -e SC2002 -e SC2223 -e SC2034 -e SC2064 -e SC2317 -e SC2028 -e SC2153 -e SC2004
|
SHELLCHECK_OPTS: -x --source-path=src -e SC2001 -e SC2002 -e SC2223 -e SC2034 -e SC2064 -e SC2317 -e SC2028 -e SC2153 -e SC2004
|
||||||
|
|||||||
36
Dockerfile
36
Dockerfile
@@ -9,12 +9,15 @@ FROM qemux/qemu-host as builder
|
|||||||
|
|
||||||
FROM debian:trixie-slim
|
FROM debian:trixie-slim
|
||||||
|
|
||||||
|
ARG TARGETPLATFORM
|
||||||
ARG DEBCONF_NOWARNINGS="yes"
|
ARG DEBCONF_NOWARNINGS="yes"
|
||||||
ARG DEBIAN_FRONTEND noninteractive
|
ARG DEBIAN_FRONTEND noninteractive
|
||||||
|
|
||||||
RUN apt-get update && apt-get -y upgrade && \
|
RUN apt-get update && apt-get -y upgrade \
|
||||||
apt-get --no-install-recommends -y install \
|
&& if [ "$TARGETPLATFORM" != "linux/amd64" ]; then extra="qemu-user"; fi \
|
||||||
tini \
|
&& apt-get --no-install-recommends -y install \
|
||||||
|
jq \
|
||||||
|
tini \
|
||||||
curl \
|
curl \
|
||||||
cpio \
|
cpio \
|
||||||
wget \
|
wget \
|
||||||
@@ -26,16 +29,18 @@ RUN apt-get update && apt-get -y upgrade && \
|
|||||||
iptables \
|
iptables \
|
||||||
iproute2 \
|
iproute2 \
|
||||||
dnsmasq \
|
dnsmasq \
|
||||||
|
fakeroot \
|
||||||
net-tools \
|
net-tools \
|
||||||
|
qemu-utils \
|
||||||
ca-certificates \
|
ca-certificates \
|
||||||
netcat-openbsd \
|
netcat-openbsd \
|
||||||
qemu-system-x86 \
|
qemu-system-x86 \
|
||||||
|
"$extra" \
|
||||||
&& apt-get clean \
|
&& apt-get clean \
|
||||||
&& rm -rf /var/lib/apt/lists/* /tmp/* /var/tmp/*
|
&& rm -rf /var/lib/apt/lists/* /tmp/* /var/tmp/*
|
||||||
|
|
||||||
COPY run/*.sh /run/
|
COPY ./src /run/
|
||||||
COPY --from=builder /qemu-host.bin /run/host.bin
|
COPY --from=builder /qemu-host.bin /run/host.bin
|
||||||
|
|
||||||
RUN chmod +x /run/*.sh && chmod +x /run/*.bin
|
RUN chmod +x /run/*.sh && chmod +x /run/*.bin
|
||||||
|
|
||||||
VOLUME /storage
|
VOLUME /storage
|
||||||
@@ -46,24 +51,13 @@ EXPOSE 139
|
|||||||
EXPOSE 445
|
EXPOSE 445
|
||||||
EXPOSE 5000
|
EXPOSE 5000
|
||||||
|
|
||||||
ENV CPU_CORES "1"
|
ENV RAM_SIZE "1G"
|
||||||
ENV DISK_SIZE "16G"
|
ENV DISK_SIZE "16G"
|
||||||
ENV RAM_SIZE "512M"
|
ENV CPU_CORES "1"
|
||||||
|
|
||||||
ARG DATE_ARG=""
|
|
||||||
ARG BUILD_ARG=0
|
|
||||||
ARG VERSION_ARG="0.0"
|
ARG VERSION_ARG="0.0"
|
||||||
ENV VERSION=$VERSION_ARG
|
RUN echo "$VERSION_ARG" > /run/version
|
||||||
|
|
||||||
LABEL org.opencontainers.image.licenses="MIT"
|
HEALTHCHECK --interval=60s --start-period=45s --retries=2 CMD /run/check.sh
|
||||||
LABEL org.opencontainers.image.title="Virtual DSM"
|
|
||||||
LABEL org.opencontainers.image.created=${DATE_ARG}
|
|
||||||
LABEL org.opencontainers.image.revision=${BUILD_ARG}
|
|
||||||
LABEL org.opencontainers.image.version=${VERSION_ARG}
|
|
||||||
LABEL org.opencontainers.image.source="https://github.com/vdsm/virtual-dsm/"
|
|
||||||
LABEL org.opencontainers.image.url="https://hub.docker.com/r/vdsm/virtual-dsm/"
|
|
||||||
LABEL org.opencontainers.image.description="Virtual DSM in a docker container"
|
|
||||||
|
|
||||||
HEALTHCHECK --interval=60s --retries=2 CMD /run/check.sh
|
ENTRYPOINT ["/usr/bin/tini", "-s", "/run/entry.sh"]
|
||||||
|
|
||||||
ENTRYPOINT ["/usr/bin/tini", "-s", "/run/run.sh"]
|
|
||||||
|
|||||||
@@ -4,19 +4,20 @@ services:
|
|||||||
container_name: dsm
|
container_name: dsm
|
||||||
image: vdsm/virtual-dsm:latest
|
image: vdsm/virtual-dsm:latest
|
||||||
environment:
|
environment:
|
||||||
CPU_CORES: "1"
|
|
||||||
DISK_SIZE: "16G"
|
DISK_SIZE: "16G"
|
||||||
RAM_SIZE: "512M"
|
RAM_SIZE: "1G"
|
||||||
|
CPU_CORES: "1"
|
||||||
devices:
|
devices:
|
||||||
- /dev/kvm
|
- /dev/kvm
|
||||||
|
- /dev/net/tun
|
||||||
- /dev/vhost-net
|
- /dev/vhost-net
|
||||||
device_cgroup_rules:
|
device_cgroup_rules:
|
||||||
- 'c *:* rwm'
|
- 'c *:* rwm'
|
||||||
cap_add:
|
cap_add:
|
||||||
- NET_ADMIN
|
- NET_ADMIN
|
||||||
ports:
|
ports:
|
||||||
- 5000:5000
|
- 5000:5000
|
||||||
volumes:
|
volumes:
|
||||||
- /opt/dsm:/storage
|
- /var/dsm:/storage
|
||||||
restart: on-failure
|
restart: on-failure
|
||||||
stop_grace_period: 1m
|
stop_grace_period: 2m
|
||||||
|
|||||||
84
readme.md
84
readme.md
@@ -10,11 +10,12 @@
|
|||||||
[![Pulls]][hub_url]
|
[![Pulls]][hub_url]
|
||||||
|
|
||||||
</div></h1>
|
</div></h1>
|
||||||
|
|
||||||
Virtual DSM in a docker container.
|
Virtual DSM in a docker container.
|
||||||
|
|
||||||
## Features
|
## Features
|
||||||
|
|
||||||
- Multi-platform
|
- Multiple disks
|
||||||
- KVM acceleration
|
- KVM acceleration
|
||||||
- GPU passthrough
|
- GPU passthrough
|
||||||
- Upgrades supported
|
- Upgrades supported
|
||||||
@@ -33,15 +34,14 @@ services:
|
|||||||
DISK_SIZE: "16G"
|
DISK_SIZE: "16G"
|
||||||
devices:
|
devices:
|
||||||
- /dev/kvm
|
- /dev/kvm
|
||||||
- /dev/vhost-net
|
|
||||||
cap_add:
|
cap_add:
|
||||||
- NET_ADMIN
|
- NET_ADMIN
|
||||||
ports:
|
ports:
|
||||||
- 5000:5000
|
- 5000:5000
|
||||||
volumes:
|
volumes:
|
||||||
- /opt/dsm:/storage
|
- /var/dsm:/storage
|
||||||
restart: on-failure
|
restart: on-failure
|
||||||
stop_grace_period: 1m
|
stop_grace_period: 2m
|
||||||
```
|
```
|
||||||
|
|
||||||
Via `docker run`
|
Via `docker run`
|
||||||
@@ -58,67 +58,59 @@ docker run -it --rm -p 5000:5000 --device=/dev/kvm --cap-add NET_ADMIN --stop-ti
|
|||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
environment:
|
environment:
|
||||||
DISK_SIZE: "256G"
|
DISK_SIZE: "128G"
|
||||||
```
|
```
|
||||||
|
|
||||||
This can also be used to resize the existing disk to a larger capacity without any data loss.
|
This can also be used to resize the existing disk to a larger capacity without any data loss.
|
||||||
|
|
||||||
* ### How do I change the location of the virtual disk?
|
* ### How do I change the location of the virtual disk?
|
||||||
|
|
||||||
To change the virtual disk's location from the default Docker volume, include the following bind mount in your compose file:
|
To change the location of the virtual disk, include the following bind mount in your compose file:
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
volumes:
|
volumes:
|
||||||
- /home/user/data:/storage
|
- /var/dsm:/storage
|
||||||
```
|
```
|
||||||
|
|
||||||
Replace the example path `/home/user/data` with the desired storage folder.
|
Replace the example path `/var/dsm` with the desired storage folder.
|
||||||
|
|
||||||
* ### How do I change the space reserved by the virtual disk?
|
|
||||||
|
|
||||||
By default, the entire disk space is reserved in advance. To create a growable disk that only reserves the space that is actually used, add the following environment variable:
|
|
||||||
|
|
||||||
```yaml
|
|
||||||
environment:
|
|
||||||
ALLOCATE: "N"
|
|
||||||
```
|
|
||||||
|
|
||||||
Keep in mind that this will not affect any of your existing disks, it only applies to newly created disks.
|
|
||||||
|
|
||||||
* ### How do I add multiple disks?
|
* ### How do I add multiple disks?
|
||||||
|
|
||||||
To mount extra volumes, modify your compose file like this:
|
To create additional disks, modify your compose file like this:
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
environment:
|
environment:
|
||||||
DISK2_SIZE: "32G"
|
DISK2_SIZE: "32G"
|
||||||
DISK3_SIZE: "64G"
|
DISK3_SIZE: "64G"
|
||||||
volumes:
|
volumes:
|
||||||
- /home/example2:/storage2
|
- /home/example:/storage2
|
||||||
- /home/example3:/storage3
|
- /mnt/data/example:/storage3
|
||||||
```
|
```
|
||||||
|
|
||||||
Additionally, it's also possible to passthrough raw disk devices like this:
|
* ### How do I create a growable disk?
|
||||||
|
|
||||||
|
By default, the entire capacity of the disk is reserved in advance.
|
||||||
|
|
||||||
|
To create a growable disk that only allocates space that is actually used, add the following environment variables:
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
environment:
|
environment:
|
||||||
DEVICE2: "/dev/vdc1"
|
ALLOCATE: "N"
|
||||||
DEVICE3: "/dev/vdc2"
|
DISK_FMT: "qcow2"
|
||||||
devices:
|
```
|
||||||
- /dev/vdc1
|
|
||||||
- /dev/vdc2
|
|
||||||
```
|
|
||||||
|
|
||||||
Please beware that any pre-existing partitions and data on those devices will be wiped.
|
Please note that this may reduce the write performance of the disk.
|
||||||
|
|
||||||
* ### How do I increase the amount of CPU or RAM?
|
* ### How do I increase the amount of CPU or RAM?
|
||||||
|
|
||||||
By default, a single core and 512 MB of RAM are allocated to the container. To increase this, add the following environment variables:
|
By default, a single core and 1 GB of RAM are allocated to the container.
|
||||||
|
|
||||||
|
To increase this, add the following environment variables:
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
environment:
|
environment:
|
||||||
|
RAM_SIZE: "4G"
|
||||||
CPU_CORES: "4"
|
CPU_CORES: "4"
|
||||||
RAM_SIZE: "2048M"
|
|
||||||
```
|
```
|
||||||
|
|
||||||
* ### How do I verify if my system supports KVM?
|
* ### How do I verify if my system supports KVM?
|
||||||
@@ -183,7 +175,20 @@ docker run -it --rm -p 5000:5000 --device=/dev/kvm --cap-add NET_ADMIN --stop-ti
|
|||||||
- 'c *:* rwm'
|
- 'c *:* rwm'
|
||||||
```
|
```
|
||||||
|
|
||||||
Please note that even if you don't need DHCP, it's still recommended to enable this feature as it prevents NAT issues and increases performance by using a `macvtap` interface.
|
Please note that even if you don't want DHCP, it's still recommended to enable this feature as it prevents NAT issues and increases performance by using a `macvtap` interface. In that case just set a static IP from the DSM control panel after you enabled this mode.
|
||||||
|
|
||||||
|
* ### How do I passthrough the GPU?
|
||||||
|
|
||||||
|
To passthrough your Intel GPU, add the following lines to your compose file:
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
environment:
|
||||||
|
GPU: "Y"
|
||||||
|
devices:
|
||||||
|
- /dev/dri
|
||||||
|
```
|
||||||
|
|
||||||
|
This can be used to enable the facial recognition function in Synology Photos for example.
|
||||||
|
|
||||||
* ### How do I install a specific version of vDSM?
|
* ### How do I install a specific version of vDSM?
|
||||||
|
|
||||||
@@ -196,19 +201,6 @@ docker run -it --rm -p 5000:5000 --device=/dev/kvm --cap-add NET_ADMIN --stop-ti
|
|||||||
|
|
||||||
With this method, it is even possible to switch between different versions while keeping all your file data intact.
|
With this method, it is even possible to switch between different versions while keeping all your file data intact.
|
||||||
|
|
||||||
* ### How do I passthrough my GPU?
|
|
||||||
|
|
||||||
To passthrough your Intel GPU, add the following lines to your compose file:
|
|
||||||
|
|
||||||
```yaml
|
|
||||||
environment:
|
|
||||||
GPU: "Y"
|
|
||||||
devices:
|
|
||||||
- /dev/dri
|
|
||||||
```
|
|
||||||
|
|
||||||
This can be used to enable the facial recognition function in Synology Photos for example.
|
|
||||||
|
|
||||||
* ### What are the differences compared to the standard DSM?
|
* ### What are the differences compared to the standard DSM?
|
||||||
|
|
||||||
There are only two minor differences: the Virtual Machine Manager package is not available, and Surveillance Station will not include any free licenses.
|
There are only two minor differences: the Virtual Machine Manager package is not available, and Surveillance Station will not include any free licenses.
|
||||||
|
|||||||
61
run/check.sh
61
run/check.sh
@@ -1,61 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -u
|
|
||||||
|
|
||||||
[ ! -f "/run/qemu.pid" ] && echo "QEMU not running yet.." && exit 0
|
|
||||||
[ -f "/run/qemu.count" ] && echo "QEMU is shutting down.." && exit 1
|
|
||||||
|
|
||||||
file="/run/dsm.url"
|
|
||||||
|
|
||||||
if [ ! -f "$file" ]; then
|
|
||||||
|
|
||||||
# Retrieve IP from guest VM for Docker healthcheck
|
|
||||||
RESPONSE=$(curl -s -m 16 -S http://127.0.0.1:2210/read?command=10 2>&1)
|
|
||||||
|
|
||||||
if [[ ! "${RESPONSE}" =~ "\"success\"" ]] ; then
|
|
||||||
echo "Failed to connect to guest: $RESPONSE" && exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Retrieve the HTTP port number
|
|
||||||
if [[ ! "${RESPONSE}" =~ "\"http_port\"" ]] ; then
|
|
||||||
echo "Failed to parse response from guest: $RESPONSE" && exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
rest=${RESPONSE#*http_port}
|
|
||||||
rest=${rest#*:}
|
|
||||||
rest=${rest%%,*}
|
|
||||||
PORT=${rest%%\"*}
|
|
||||||
|
|
||||||
[ -z "${PORT}" ] && echo "Guest has not set a portnumber yet.." && exit 1
|
|
||||||
|
|
||||||
# Retrieve the IP address
|
|
||||||
if [[ ! "${RESPONSE}" =~ "eth0" ]] ; then
|
|
||||||
echo "Failed to parse response from guest: $RESPONSE" && exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
rest=${RESPONSE#*eth0}
|
|
||||||
rest=${rest#*ip}
|
|
||||||
rest=${rest#*:}
|
|
||||||
rest=${rest#*\"}
|
|
||||||
IP=${rest%%\"*}
|
|
||||||
|
|
||||||
[ -z "${IP}" ] && echo "Guest has not received an IP yet.." && exit 1
|
|
||||||
|
|
||||||
echo "${IP}:${PORT}" > $file
|
|
||||||
|
|
||||||
fi
|
|
||||||
|
|
||||||
LOCATION=$(cat "$file")
|
|
||||||
|
|
||||||
if ! curl -m 20 -ILfSs "http://${LOCATION}/" > /dev/null; then
|
|
||||||
rm -f $file
|
|
||||||
echo "Failed to reach http://${LOCATION}"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ "$LOCATION" == "20.20"* ]]; then
|
|
||||||
echo "Healthcheck OK"
|
|
||||||
else
|
|
||||||
echo "Healthcheck OK ( ${LOCATION%:*} )"
|
|
||||||
fi
|
|
||||||
|
|
||||||
exit 0
|
|
||||||
@@ -1,40 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
set -Eeuo pipefail
|
|
||||||
|
|
||||||
KVM_ERR=""
|
|
||||||
KVM_OPTS=""
|
|
||||||
|
|
||||||
if [ -e /dev/kvm ] && sh -c 'echo -n > /dev/kvm' &> /dev/null; then
|
|
||||||
if ! grep -q -e vmx -e svm /proc/cpuinfo; then
|
|
||||||
KVM_ERR="(vmx/svm disabled)"
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
[ -e /dev/kvm ] && KVM_ERR="(no write access)" || KVM_ERR="(device file missing)"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ -n "${KVM_ERR}" ]; then
|
|
||||||
if [ "$ARCH" == "amd64" ]; then
|
|
||||||
error "KVM acceleration not detected ${KVM_ERR}, see the FAQ about this."
|
|
||||||
[[ "${DEBUG}" != [Yy1]* ]] && exit 88
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
KVM_OPTS=",accel=kvm -enable-kvm -cpu host"
|
|
||||||
fi
|
|
||||||
|
|
||||||
DEF_OPTS="-nographic -nodefaults -boot strict=on -display none"
|
|
||||||
RAM_OPTS=$(echo "-m ${RAM_SIZE}" | sed 's/MB/M/g;s/GB/G/g;s/TB/T/g')
|
|
||||||
CPU_OPTS="-smp ${CPU_CORES},sockets=1,dies=1,cores=${CPU_CORES},threads=1"
|
|
||||||
MAC_OPTS="-machine type=q35,usb=off,dump-guest-core=off,hpet=off${KVM_OPTS}"
|
|
||||||
EXTRA_OPTS="-device virtio-balloon-pci,id=balloon0,bus=pcie.0,addr=0x4"
|
|
||||||
EXTRA_OPTS="$EXTRA_OPTS -object rng-random,id=objrng0,filename=/dev/urandom"
|
|
||||||
EXTRA_OPTS="$EXTRA_OPTS -device virtio-rng-pci,rng=objrng0,id=rng0,bus=pcie.0,addr=0x1c"
|
|
||||||
|
|
||||||
if [[ "${GPU}" == [Yy1]* ]] && [[ "$ARCH" == "amd64" ]]; then
|
|
||||||
DEF_OPTS="-nodefaults -boot strict=on -display egl-headless,rendernode=/dev/dri/renderD128"
|
|
||||||
DEF_OPTS="${DEF_OPTS} -device virtio-vga,id=video0,max_outputs=1,bus=pcie.0,addr=0x1"
|
|
||||||
fi
|
|
||||||
|
|
||||||
ARGS="${DEF_OPTS} ${CPU_OPTS} ${RAM_OPTS} ${MAC_OPTS} ${MON_OPTS} ${SERIAL_OPTS} ${NET_OPTS} ${DISK_OPTS} ${EXTRA_OPTS} ${ARGUMENTS}"
|
|
||||||
ARGS=$(echo "$ARGS" | sed 's/\t/ /g' | tr -s ' ')
|
|
||||||
|
|
||||||
return 0
|
|
||||||
241
run/disk.sh
241
run/disk.sh
@@ -1,241 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -Eeuo pipefail
|
|
||||||
|
|
||||||
# Docker environment variables
|
|
||||||
|
|
||||||
: ${DISK_IO:='native'} # I/O Mode, can be set to 'native', 'threads' or 'io_turing'
|
|
||||||
: ${DISK_CACHE:='none'} # Caching mode, can be set to 'writeback' for better performance
|
|
||||||
: ${DISK_DISCARD:='on'} # Controls whether unmap (TRIM) commands are passed to the host.
|
|
||||||
: ${DISK_ROTATION:='1'} # Rotation rate, set to 1 for SSD storage and increase for HDD
|
|
||||||
|
|
||||||
BOOT="$STORAGE/$BASE.boot.img"
|
|
||||||
SYSTEM="$STORAGE/$BASE.system.img"
|
|
||||||
|
|
||||||
[ ! -f "$BOOT" ] && error "Virtual DSM boot-image does not exist ($BOOT)" && exit 81
|
|
||||||
[ ! -f "$SYSTEM" ] && error "Virtual DSM system-image does not exist ($SYSTEM)" && exit 82
|
|
||||||
|
|
||||||
DATA="${STORAGE}/data.img"
|
|
||||||
|
|
||||||
if [[ ! -f "${DATA}" ]] && [[ -f "$STORAGE/data$DISK_SIZE.img" ]]; then
|
|
||||||
# Fallback for legacy installs
|
|
||||||
DATA="$STORAGE/data$DISK_SIZE.img"
|
|
||||||
fi
|
|
||||||
|
|
||||||
MIN_SIZE=6442450944
|
|
||||||
DISK_SIZE=$(echo "${DISK_SIZE}" | sed 's/MB/M/g;s/GB/G/g;s/TB/T/g')
|
|
||||||
DATA_SIZE=$(numfmt --from=iec "${DISK_SIZE}")
|
|
||||||
|
|
||||||
if (( DATA_SIZE < MIN_SIZE )); then
|
|
||||||
error "Please increase DISK_SIZE to at least 6 GB." && exit 83
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ -f "${DATA}" ]; then
|
|
||||||
|
|
||||||
OLD_SIZE=$(stat -c%s "${DATA}")
|
|
||||||
|
|
||||||
if [ "$DATA_SIZE" -gt "$OLD_SIZE" ]; then
|
|
||||||
|
|
||||||
info "Resizing data disk from $OLD_SIZE to $DATA_SIZE bytes.."
|
|
||||||
|
|
||||||
if [[ "${ALLOCATE}" == [Nn]* ]]; then
|
|
||||||
|
|
||||||
# Resize file by changing its length
|
|
||||||
if ! truncate -s "${DATA_SIZE}" "${DATA}"; then
|
|
||||||
error "Could not resize the file for the virtual disk." && exit 85
|
|
||||||
fi
|
|
||||||
|
|
||||||
else
|
|
||||||
|
|
||||||
REQ=$((DATA_SIZE-OLD_SIZE))
|
|
||||||
|
|
||||||
# Check free diskspace
|
|
||||||
SPACE=$(df --output=avail -B 1 "${STORAGE}" | tail -n 1)
|
|
||||||
|
|
||||||
if (( REQ > SPACE )); then
|
|
||||||
error "Not enough free space to resize virtual disk to ${DISK_SIZE}."
|
|
||||||
error "Specify a smaller size or disable preallocation with ALLOCATE=N." && exit 84
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Resize file by allocating more space
|
|
||||||
if ! fallocate -l "${DATA_SIZE}" "${DATA}"; then
|
|
||||||
if ! truncate -s "${DATA_SIZE}" "${DATA}"; then
|
|
||||||
error "Could not resize the file for the virtual disk." && exit 85
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ "${ALLOCATE}" == [Zz]* ]]; then
|
|
||||||
|
|
||||||
GB=$(( (REQ + 1073741823)/1073741824 ))
|
|
||||||
|
|
||||||
info "Preallocating ${GB} GB of diskspace, please wait..."
|
|
||||||
dd if=/dev/urandom of="${DATA}" seek="${OLD_SIZE}" count="${REQ}" bs=1M iflag=count_bytes oflag=seek_bytes status=none
|
|
||||||
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ "$DATA_SIZE" -lt "$OLD_SIZE" ]; then
|
|
||||||
|
|
||||||
info "Shrinking existing disks is not supported yet!"
|
|
||||||
info "Creating backup of old drive in storage folder..."
|
|
||||||
|
|
||||||
mv -f "${DATA}" "${DATA}.bak"
|
|
||||||
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ ! -f "${DATA}" ]; then
|
|
||||||
|
|
||||||
if [[ "${ALLOCATE}" == [Nn]* ]]; then
|
|
||||||
|
|
||||||
# Create an empty file
|
|
||||||
if ! truncate -s "${DATA_SIZE}" "${DATA}"; then
|
|
||||||
rm -f "${DATA}"
|
|
||||||
error "Could not create a file for the virtual disk." && exit 87
|
|
||||||
fi
|
|
||||||
|
|
||||||
else
|
|
||||||
|
|
||||||
# Check free diskspace
|
|
||||||
SPACE=$(df --output=avail -B 1 "${STORAGE}" | tail -n 1)
|
|
||||||
|
|
||||||
if (( DATA_SIZE > SPACE )); then
|
|
||||||
error "Not enough free space to create a virtual disk of ${DISK_SIZE}."
|
|
||||||
error "Specify a smaller size or disable preallocation with ALLOCATE=N." && exit 86
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Create an empty file
|
|
||||||
if ! fallocate -l "${DATA_SIZE}" "${DATA}"; then
|
|
||||||
if ! truncate -s "${DATA_SIZE}" "${DATA}"; then
|
|
||||||
rm -f "${DATA}"
|
|
||||||
error "Could not create a file for the virtual disk." && exit 87
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ "${ALLOCATE}" == [Zz]* ]]; then
|
|
||||||
|
|
||||||
info "Preallocating ${DISK_SIZE} of diskspace, please wait..."
|
|
||||||
dd if=/dev/urandom of="${DATA}" count="${DATA_SIZE}" bs=1M iflag=count_bytes status=none
|
|
||||||
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Check if file exists
|
|
||||||
if [ ! -f "${DATA}" ]; then
|
|
||||||
error "Virtual disk does not exist ($DATA)" && exit 88
|
|
||||||
fi
|
|
||||||
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Check the filesize
|
|
||||||
SIZE=$(stat -c%s "${DATA}")
|
|
||||||
|
|
||||||
if [[ SIZE -ne DATA_SIZE ]]; then
|
|
||||||
error "Virtual disk has the wrong size: ${SIZE}" && exit 89
|
|
||||||
fi
|
|
||||||
|
|
||||||
DISK_OPTS="\
|
|
||||||
-device virtio-scsi-pci,id=hw-synoboot,bus=pcie.0,addr=0xa \
|
|
||||||
-drive file=${BOOT},if=none,id=drive-synoboot,format=raw,cache=${DISK_CACHE},aio=${DISK_IO},discard=${DISK_DISCARD},detect-zeroes=on \
|
|
||||||
-device scsi-hd,bus=hw-synoboot.0,channel=0,scsi-id=0,lun=0,drive=drive-synoboot,id=synoboot0,rotation_rate=${DISK_ROTATION},bootindex=1 \
|
|
||||||
-device virtio-scsi-pci,id=hw-synosys,bus=pcie.0,addr=0xb \
|
|
||||||
-drive file=${SYSTEM},if=none,id=drive-synosys,format=raw,cache=${DISK_CACHE},aio=${DISK_IO},discard=${DISK_DISCARD},detect-zeroes=on \
|
|
||||||
-device scsi-hd,bus=hw-synosys.0,channel=0,scsi-id=0,lun=0,drive=drive-synosys,id=synosys0,rotation_rate=${DISK_ROTATION},bootindex=2 \
|
|
||||||
-device virtio-scsi-pci,id=hw-userdata,bus=pcie.0,addr=0xc \
|
|
||||||
-drive file=${DATA},if=none,id=drive-userdata,format=raw,cache=${DISK_CACHE},aio=${DISK_IO},discard=${DISK_DISCARD},detect-zeroes=on \
|
|
||||||
-device scsi-hd,bus=hw-userdata.0,channel=0,scsi-id=0,lun=0,drive=drive-userdata,id=userdata0,rotation_rate=${DISK_ROTATION},bootindex=3"
|
|
||||||
|
|
||||||
: ${DISK2_SIZE:=''}
|
|
||||||
EXTRA_SIZE=DISK2_SIZE
|
|
||||||
EXTRA_DISK="/storage2/data.img"
|
|
||||||
|
|
||||||
if [ -d "$(dirname "${EXTRA_DISK}")" ]; then
|
|
||||||
|
|
||||||
if [ ! -f "${EXTRA_DISK}" ]; then
|
|
||||||
[ -z "$EXTRA_SIZE" ] && EXTRA_SIZE="16G"
|
|
||||||
if ! truncate -s "${EXTRA_SIZE}" "${EXTRA_DISK}"; then
|
|
||||||
error "Could not create the file for the second disk." && exit 53
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ -n "$EXTRA_SIZE" ]; then
|
|
||||||
CUR_SIZE=$(stat -c%s "${EXTRA_DISK}")
|
|
||||||
DATA_SIZE=$(numfmt --from=iec "${EXTRA_SIZE}")
|
|
||||||
if [ "$DATA_SIZE" -gt "$CUR_SIZE" ]; then
|
|
||||||
truncate -s "${EXTRA_SIZE}" "${EXTRA_DISK}"
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
DISK_OPTS="${DISK_OPTS} \
|
|
||||||
-device virtio-scsi-pci,id=hw-userdata2,bus=pcie.0,addr=0xd \
|
|
||||||
-drive file=${EXTRA_DISK},if=none,id=drive-userdata2,format=raw,cache=${DISK_CACHE},aio=${DISK_IO},discard=${DISK_DISCARD},detect-zeroes=on \
|
|
||||||
-device scsi-hd,bus=hw-userdata2.0,channel=0,scsi-id=0,lun=0,drive=drive-userdata2,id=userdata2,rotation_rate=${DISK_ROTATION},bootindex=4"
|
|
||||||
|
|
||||||
fi
|
|
||||||
|
|
||||||
: ${DISK3_SIZE:=''}
|
|
||||||
EXTRA_SIZE=DISK3_SIZE
|
|
||||||
EXTRA_DISK="/storage3/data.img"
|
|
||||||
|
|
||||||
if [ -d "$(dirname "${EXTRA_DISK}")" ]; then
|
|
||||||
|
|
||||||
if [ ! -f "${EXTRA_DISK}" ]; then
|
|
||||||
[ -z "$EXTRA_SIZE" ] && EXTRA_SIZE="16G"
|
|
||||||
if ! truncate -s "${EXTRA_SIZE}" "${EXTRA_DISK}"; then
|
|
||||||
error "Could not create the file for the third disk." && exit 54
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ -n "$EXTRA_SIZE" ]; then
|
|
||||||
CUR_SIZE=$(stat -c%s "${EXTRA_DISK}")
|
|
||||||
DATA_SIZE=$(numfmt --from=iec "${EXTRA_SIZE}")
|
|
||||||
if [ "$DATA_SIZE" -gt "$CUR_SIZE" ]; then
|
|
||||||
truncate -s "${EXTRA_SIZE}" "${EXTRA_DISK}"
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
DISK_OPTS="${DISK_OPTS} \
|
|
||||||
-device virtio-scsi-pci,id=hw-userdata3,bus=pcie.0,addr=0xe \
|
|
||||||
-drive file=${EXTRA_DISK},if=none,id=drive-userdata3,format=raw,cache=${DISK_CACHE},aio=${DISK_IO},discard=${DISK_DISCARD},detect-zeroes=on \
|
|
||||||
-device scsi-hd,bus=hw-userdata3.0,channel=0,scsi-id=0,lun=0,drive=drive-userdata3,id=userdata3,rotation_rate=${DISK_ROTATION},bootindex=5"
|
|
||||||
|
|
||||||
fi
|
|
||||||
|
|
||||||
: ${DEVICE:=''} # Docker variable to passthrough a block device, like /dev/vdc1.
|
|
||||||
: ${DEVICE2:=''}
|
|
||||||
: ${DEVICE3:=''}
|
|
||||||
|
|
||||||
if [ -n "${DEVICE}" ]; then
|
|
||||||
|
|
||||||
[ ! -b "${DEVICE}" ] && error "Device ${DEVICE} cannot be found! Please add it to the 'devices' section of your compose file." && exit 55
|
|
||||||
|
|
||||||
DISK_OPTS="${DISK_OPTS} \
|
|
||||||
-device virtio-scsi-pci,id=hw-userdata4,bus=pcie.0,addr=0xf \
|
|
||||||
-drive file=${DEVICE},if=none,id=drive-userdata4,format=raw,cache=${DISK_CACHE},aio=${DISK_IO},discard=${DISK_DISCARD},detect-zeroes=on \
|
|
||||||
-device scsi-hd,bus=hw-userdata4.0,channel=0,scsi-id=0,lun=0,drive=drive-userdata4,id=userdata4,rotation_rate=${DISK_ROTATION},bootindex=6"
|
|
||||||
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ -n "${DEVICE2}" ]; then
|
|
||||||
|
|
||||||
[ ! -b "${DEVICE2}" ] && error "Device ${DEVICE2} cannot be found! Please add it to the 'devices' section of your compose file." && exit 56
|
|
||||||
|
|
||||||
DISK_OPTS="${DISK_OPTS} \
|
|
||||||
-device virtio-scsi-pci,id=hw-userdata5,bus=pcie.0,addr=0x5 \
|
|
||||||
-drive file=${DEVICE2},if=none,id=drive-userdata5,format=raw,cache=${DISK_CACHE},aio=${DISK_IO},discard=${DISK_DISCARD},detect-zeroes=on \
|
|
||||||
-device scsi-hd,bus=hw-userdata5.0,channel=0,scsi-id=0,lun=0,drive=drive-userdata5,id=userdata5,rotation_rate=${DISK_ROTATION},bootindex=7"
|
|
||||||
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ -n "${DEVICE3}" ]; then
|
|
||||||
|
|
||||||
[ ! -b "${DEVICE3}" ] && error "Device ${DEVICE3} cannot be found! Please add it to the 'devices' section of your compose file." && exit 57
|
|
||||||
|
|
||||||
DISK_OPTS="${DISK_OPTS} \
|
|
||||||
-device virtio-scsi-pci,id=hw-userdata6,bus=pcie.0,addr=0x6 \
|
|
||||||
-drive file=${DEVICE3},if=none,id=drive-userdata6,format=raw,cache=${DISK_CACHE},aio=${DISK_IO},discard=${DISK_DISCARD},detect-zeroes=on \
|
|
||||||
-device scsi-hd,bus=hw-userdata6.0,channel=0,scsi-id=0,lun=0,drive=drive-userdata6,id=userdata6,rotation_rate=${DISK_ROTATION},bootindex=8"
|
|
||||||
|
|
||||||
fi
|
|
||||||
|
|
||||||
return 0
|
|
||||||
45
run/gpu.sh
45
run/gpu.sh
@@ -1,45 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
set -Eeuo pipefail
|
|
||||||
|
|
||||||
if [[ "${GPU}" != [Yy1]* ]] || [[ "$ARCH" != "amd64" ]]; then
|
|
||||||
return 0
|
|
||||||
fi
|
|
||||||
|
|
||||||
[ ! -d /dev/dri ] && mkdir -m 755 /dev/dri
|
|
||||||
|
|
||||||
if [ ! -c /dev/dri/card0 ]; then
|
|
||||||
mknod /dev/dri/card0 c 226 0
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ ! -c /dev/dri/renderD128 ]; then
|
|
||||||
mknod /dev/dri/renderD128 c 226 128
|
|
||||||
fi
|
|
||||||
|
|
||||||
chmod 666 /dev/dri/card0
|
|
||||||
chmod 666 /dev/dri/renderD128
|
|
||||||
|
|
||||||
if ! apt-mark showinstall | grep -q "xserver-xorg-video-intel"; then
|
|
||||||
|
|
||||||
info "Installing Intel GPU drivers..."
|
|
||||||
|
|
||||||
export DEBCONF_NOWARNINGS="yes"
|
|
||||||
export DEBIAN_FRONTEND="noninteractive"
|
|
||||||
|
|
||||||
apt-get -qq update
|
|
||||||
apt-get -qq --no-install-recommends -y install xserver-xorg-video-intel > /dev/null
|
|
||||||
|
|
||||||
fi
|
|
||||||
|
|
||||||
if ! apt-mark showinstall | grep -q "qemu-system-modules-opengl"; then
|
|
||||||
|
|
||||||
info "Installing OpenGL module..."
|
|
||||||
|
|
||||||
export DEBCONF_NOWARNINGS="yes"
|
|
||||||
export DEBIAN_FRONTEND="noninteractive"
|
|
||||||
|
|
||||||
apt-get -qq update
|
|
||||||
apt-get -qq --no-install-recommends -y install qemu-system-modules-opengl > /dev/null
|
|
||||||
|
|
||||||
fi
|
|
||||||
|
|
||||||
return 0
|
|
||||||
74
run/power.sh
74
run/power.sh
@@ -1,74 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -Eeuo pipefail
|
|
||||||
|
|
||||||
# Configure QEMU for graceful shutdown
|
|
||||||
|
|
||||||
QEMU_PORT=7100
|
|
||||||
QEMU_TIMEOUT=50
|
|
||||||
|
|
||||||
QEMU_PID=/run/qemu.pid
|
|
||||||
QEMU_COUNT=/run/qemu.count
|
|
||||||
|
|
||||||
rm -f "${QEMU_PID}"
|
|
||||||
rm -f "${QEMU_COUNT}"
|
|
||||||
|
|
||||||
_trap(){
|
|
||||||
func="$1" ; shift
|
|
||||||
for sig ; do
|
|
||||||
trap "$func $sig" "$sig"
|
|
||||||
done
|
|
||||||
}
|
|
||||||
|
|
||||||
_graceful_shutdown() {
|
|
||||||
|
|
||||||
set +e
|
|
||||||
|
|
||||||
[ ! -f "${QEMU_PID}" ] && exit 130
|
|
||||||
[ -f "${QEMU_COUNT}" ] && return
|
|
||||||
|
|
||||||
echo && info "Received $1 signal, shutting down..."
|
|
||||||
echo 0 > "${QEMU_COUNT}"
|
|
||||||
|
|
||||||
# Don't send the powerdown signal because vDSM ignores ACPI signals
|
|
||||||
# echo 'system_powerdown' | nc -q 1 -w 1 localhost "${QEMU_PORT}" > /dev/null
|
|
||||||
|
|
||||||
# Send shutdown command to guest agent via serial port
|
|
||||||
RESPONSE=$(curl -s -m 5 -S http://127.0.0.1:2210/read?command=6 2>&1)
|
|
||||||
|
|
||||||
if [[ ! "${RESPONSE}" =~ "\"success\"" ]] ; then
|
|
||||||
|
|
||||||
echo && error "Could not send shutdown command to the guest ($RESPONSE)"
|
|
||||||
|
|
||||||
kill -15 "$(cat "${QEMU_PID}")"
|
|
||||||
pkill -f qemu-system-x86_64 || true
|
|
||||||
|
|
||||||
fi
|
|
||||||
|
|
||||||
while [ "$(cat ${QEMU_COUNT})" -lt "${QEMU_TIMEOUT}" ]; do
|
|
||||||
|
|
||||||
# Increase the counter
|
|
||||||
echo $(($(cat ${QEMU_COUNT})+1)) > ${QEMU_COUNT}
|
|
||||||
|
|
||||||
# Try to connect to qemu
|
|
||||||
if echo 'info version'| nc -q 1 -w 1 localhost "${QEMU_PORT}" >/dev/null 2>&1 ; then
|
|
||||||
|
|
||||||
sleep 1
|
|
||||||
|
|
||||||
CNT="$(cat ${QEMU_COUNT})/${QEMU_TIMEOUT}"
|
|
||||||
[[ "${DEBUG}" == [Yy1]* ]] && info "Shutting down, waiting... (${CNT})"
|
|
||||||
|
|
||||||
fi
|
|
||||||
|
|
||||||
done
|
|
||||||
|
|
||||||
echo && echo "❯ Quitting..."
|
|
||||||
echo 'quit' | nc -q 1 -w 1 localhost "${QEMU_PORT}" >/dev/null 2>&1 || true
|
|
||||||
|
|
||||||
closeNetwork
|
|
||||||
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
_trap _graceful_shutdown SIGTERM SIGHUP SIGINT SIGABRT SIGQUIT
|
|
||||||
|
|
||||||
MON_OPTS="-monitor telnet:localhost:${QEMU_PORT},server,nowait,nodelay"
|
|
||||||
66
run/print.sh
66
run/print.sh
@@ -1,66 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -Eeuo pipefail
|
|
||||||
|
|
||||||
info () { echo -e >&2 "\E[1;34m❯\E[1;36m $1\E[0m" ; }
|
|
||||||
error () { echo -e >&2 "\E[1;31m❯ ERROR: $1\E[0m" ; }
|
|
||||||
|
|
||||||
file="/run/dsm.url"
|
|
||||||
|
|
||||||
while [ ! -f "$file" ]
|
|
||||||
do
|
|
||||||
|
|
||||||
sleep 3
|
|
||||||
[ -f "$file" ] && continue
|
|
||||||
|
|
||||||
# Retrieve IP from guest VM
|
|
||||||
|
|
||||||
set +e
|
|
||||||
RESPONSE=$(curl -s -m 16 -S http://127.0.0.1:2210/read?command=10 2>&1)
|
|
||||||
set -e
|
|
||||||
|
|
||||||
if [[ ! "${RESPONSE}" =~ "\"success\"" ]] ; then
|
|
||||||
error "Failed to connect to guest: $RESPONSE" && continue
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Retrieve the HTTP port number
|
|
||||||
if [[ ! "${RESPONSE}" =~ "\"http_port\"" ]] ; then
|
|
||||||
error "Failed to parse response from guest: $RESPONSE" && continue
|
|
||||||
fi
|
|
||||||
|
|
||||||
rest=${RESPONSE#*http_port}
|
|
||||||
rest=${rest#*:}
|
|
||||||
rest=${rest%%,*}
|
|
||||||
PORT=${rest%%\"*}
|
|
||||||
|
|
||||||
[ -z "${PORT}" ] && continue
|
|
||||||
|
|
||||||
# Retrieve the IP address
|
|
||||||
if [[ ! "${RESPONSE}" =~ "eth0" ]] ; then
|
|
||||||
error "Failed to parse response from guest: $RESPONSE" && continue
|
|
||||||
fi
|
|
||||||
|
|
||||||
rest=${RESPONSE#*eth0}
|
|
||||||
rest=${rest#*ip}
|
|
||||||
rest=${rest#*:}
|
|
||||||
rest=${rest#*\"}
|
|
||||||
IP=${rest%%\"*}
|
|
||||||
|
|
||||||
[ -z "${IP}" ] && continue
|
|
||||||
|
|
||||||
echo "${IP}:${PORT}" > $file
|
|
||||||
|
|
||||||
done
|
|
||||||
|
|
||||||
LOCATION=$(cat "$file")
|
|
||||||
|
|
||||||
if [[ "$LOCATION" == "20.20"* ]]; then
|
|
||||||
MSG="port ${LOCATION##*:}"
|
|
||||||
else
|
|
||||||
MSG="http://${LOCATION}"
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo "" >&2
|
|
||||||
info "--------------------------------------------------------"
|
|
||||||
info " You can now login to DSM at ${MSG}"
|
|
||||||
info "--------------------------------------------------------"
|
|
||||||
echo "" >&2
|
|
||||||
35
run/reset.sh
35
run/reset.sh
@@ -1,35 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -Eeuo pipefail
|
|
||||||
|
|
||||||
info () { echo -e "\E[1;34m❯ \E[1;36m$1\E[0m" ; }
|
|
||||||
error () { echo -e >&2 "\E[1;31m❯ ERROR: $1\E[0m" ; }
|
|
||||||
trap 'error "Status $? while: ${BASH_COMMAND} (line $LINENO/$BASH_LINENO)"' ERR
|
|
||||||
|
|
||||||
[ ! -f "/run/run.sh" ] && error "Script must run inside Docker container!" && exit 11
|
|
||||||
[ "$(id -u)" -ne "0" ] && error "Script must be executed with root privileges." && exit 12
|
|
||||||
|
|
||||||
# Docker environment variables
|
|
||||||
|
|
||||||
: ${URL:=''} # URL of the PAT file
|
|
||||||
: ${GPU:='N'} # Enable GPU passthrough
|
|
||||||
: ${DEBUG:='N'} # Enable debugging mode
|
|
||||||
: ${ALLOCATE:='Y'} # Preallocate diskspace
|
|
||||||
: ${ARGUMENTS:=''} # Extra QEMU parameters
|
|
||||||
: ${CPU_CORES:='1'} # Amount of CPU cores
|
|
||||||
: ${DISK_SIZE:='16G'} # Initial data disk size
|
|
||||||
: ${RAM_SIZE:='512M'} # Maximum RAM amount
|
|
||||||
|
|
||||||
# Helper variables
|
|
||||||
|
|
||||||
KERNEL=$(uname -r | cut -b 1)
|
|
||||||
MINOR=$(uname -r | cut -d '.' -f2)
|
|
||||||
ARCH=$(dpkg --print-architecture)
|
|
||||||
VERS=$(qemu-system-x86_64 --version | head -n 1 | cut -d '(' -f 1)
|
|
||||||
|
|
||||||
# Cleanup files
|
|
||||||
|
|
||||||
rm -f /run/dsm.url
|
|
||||||
rm -f /run/qemu.pid
|
|
||||||
rm -f /run/qemu.count
|
|
||||||
|
|
||||||
return 0
|
|
||||||
26
run/run.sh
26
run/run.sh
@@ -1,26 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -Eeuo pipefail
|
|
||||||
|
|
||||||
echo "❯ Starting Virtual DSM for Docker v${VERSION}..."
|
|
||||||
echo "❯ For support visit https://github.com/vdsm/virtual-dsm/"
|
|
||||||
|
|
||||||
. /run/reset.sh # Initialize system
|
|
||||||
. /run/install.sh # Run installation
|
|
||||||
. /run/disk.sh # Initialize disks
|
|
||||||
. /run/network.sh # Initialize network
|
|
||||||
. /run/gpu.sh # Initialize graphics
|
|
||||||
. /run/serial.sh # Initialize serialport
|
|
||||||
. /run/power.sh # Configure shutdown
|
|
||||||
. /run/config.sh # Configure arguments
|
|
||||||
|
|
||||||
trap - ERR
|
|
||||||
|
|
||||||
set -m
|
|
||||||
(
|
|
||||||
[[ "${DEBUG}" == [Yy1]* ]] && info "$VERS" && set -x
|
|
||||||
qemu-system-x86_64 ${ARGS:+ $ARGS} & echo $! > "${QEMU_PID}"
|
|
||||||
{ set +x; } 2>/dev/null
|
|
||||||
)
|
|
||||||
set +m
|
|
||||||
|
|
||||||
tail --pid "$(cat "${QEMU_PID}")" --follow /dev/null & wait $!
|
|
||||||
@@ -1,55 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
set -Eeuo pipefail
|
|
||||||
|
|
||||||
# Docker environment variables
|
|
||||||
|
|
||||||
: ${HOST_CPU:=''}
|
|
||||||
: ${HOST_MAC:=''}
|
|
||||||
: ${HOST_DEBUG:=''}
|
|
||||||
: ${HOST_SERIAL:=''}
|
|
||||||
: ${HOST_MODEL:=''}
|
|
||||||
: ${GUEST_SERIAL:=''}
|
|
||||||
|
|
||||||
if [ -z "$HOST_CPU" ]; then
|
|
||||||
HOST_CPU=$(lscpu | grep 'Model name' | cut -f 2 -d ":" | awk '{$1=$1}1' | sed 's# @.*##g' | sed s/"(R)"//g | sed 's/[^[:alnum:] ]\+/ /g' | sed 's/ */ /g')
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ -n "$HOST_CPU" ]; then
|
|
||||||
HOST_CPU="$HOST_CPU,,"
|
|
||||||
else
|
|
||||||
if [ "$ARCH" == "amd64" ]; then
|
|
||||||
HOST_CPU="QEMU, Virtual CPU, X86_64"
|
|
||||||
else
|
|
||||||
HOST_CPU="QEMU, Virtual CPU, $ARCH"
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
HOST_ARGS=()
|
|
||||||
HOST_ARGS+=("-cpu=${CPU_CORES}")
|
|
||||||
HOST_ARGS+=("-cpu_arch=${HOST_CPU}")
|
|
||||||
|
|
||||||
[ -n "$HOST_MAC" ] && HOST_ARGS+=("-mac=${HOST_MAC}")
|
|
||||||
[ -n "$HOST_MODEL" ] && HOST_ARGS+=("-model=${HOST_MODEL}")
|
|
||||||
[ -n "$HOST_SERIAL" ] && HOST_ARGS+=("-hostsn=${HOST_SERIAL}")
|
|
||||||
[ -n "$GUEST_SERIAL" ] && HOST_ARGS+=("-guestsn=${GUEST_SERIAL}")
|
|
||||||
|
|
||||||
if [[ "${HOST_DEBUG}" == [Yy1]* ]]; then
|
|
||||||
set -x
|
|
||||||
./run/host.bin "${HOST_ARGS[@]}" &
|
|
||||||
{ set +x; } 2>/dev/null
|
|
||||||
echo
|
|
||||||
else
|
|
||||||
./run/host.bin "${HOST_ARGS[@]}" >/dev/null &
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Configure serial ports
|
|
||||||
|
|
||||||
SERIAL_OPTS="\
|
|
||||||
-serial mon:stdio \
|
|
||||||
-device virtio-serial-pci,id=virtio-serial0,bus=pcie.0,addr=0x3 \
|
|
||||||
-chardev pty,id=charserial0 \
|
|
||||||
-device isa-serial,chardev=charserial0,id=serial0 \
|
|
||||||
-chardev socket,id=charchannel0,host=127.0.0.1,port=12345,reconnect=10 \
|
|
||||||
-device virtserialport,bus=virtio-serial0.0,nr=1,chardev=charchannel0,id=channel0,name=vchannel"
|
|
||||||
|
|
||||||
return 0
|
|
||||||
30
src/check.sh
Normal file
30
src/check.sh
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -Eeuo pipefail
|
||||||
|
|
||||||
|
: ${VM_NET_DEV:='eth0'}
|
||||||
|
|
||||||
|
[ ! -f "/run/qemu.pid" ] && echo "QEMU not running yet.." && exit 0
|
||||||
|
[ -f "/run/qemu.count" ] && echo "QEMU is shutting down.." && exit 1
|
||||||
|
|
||||||
|
file="/run/dsm.url"
|
||||||
|
[ ! -f "$file" ] && echo "DSM has not enabled networking yet.." && exit 1
|
||||||
|
|
||||||
|
location=$(cat "$file")
|
||||||
|
|
||||||
|
if ! curl -m 20 -ILfSs "http://$location/" > /dev/null; then
|
||||||
|
|
||||||
|
if [[ "$location" == "20.20"* ]]; then
|
||||||
|
ip="20.20.20.1"
|
||||||
|
port="${location##*:}"
|
||||||
|
echo "Failed to reach DSM at port $port"
|
||||||
|
else
|
||||||
|
echo "Failed to reach DSM at http://$location"
|
||||||
|
ip=$(ip address show dev "$VM_NET_DEV" | grep inet | awk '/inet / { print $2 }' | cut -f1 -d/)
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "You might need to whitelist IP $ip in the DSM firewall." && exit 1
|
||||||
|
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "Healthcheck OK"
|
||||||
|
exit 0
|
||||||
20
src/config.sh
Normal file
20
src/config.sh
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -Eeuo pipefail
|
||||||
|
|
||||||
|
DEF_OPTS="-nographic -nodefaults -boot strict=on -display none"
|
||||||
|
RAM_OPTS=$(echo "-m $RAM_SIZE" | sed 's/MB/M/g;s/GB/G/g;s/TB/T/g')
|
||||||
|
CPU_OPTS="-cpu $CPU_MODEL -smp $CPU_CORES,sockets=1,dies=1,cores=$CPU_CORES,threads=1"
|
||||||
|
MAC_OPTS="-machine type=q35,usb=off,dump-guest-core=off,hpet=off${KVM_OPTS}"
|
||||||
|
EXTRA_OPTS="-device virtio-balloon-pci,id=balloon0,bus=pcie.0,addr=0x4"
|
||||||
|
EXTRA_OPTS="$EXTRA_OPTS -object rng-random,id=objrng0,filename=/dev/urandom"
|
||||||
|
EXTRA_OPTS="$EXTRA_OPTS -device virtio-rng-pci,rng=objrng0,id=rng0,bus=pcie.0,addr=0x1c"
|
||||||
|
|
||||||
|
if [[ "$GPU" == [Yy1]* ]] && [[ "$ARCH" == "amd64" ]]; then
|
||||||
|
DEF_OPTS="-nodefaults -boot strict=on -display egl-headless,rendernode=/dev/dri/renderD128"
|
||||||
|
DEF_OPTS="$DEF_OPTS -device virtio-vga,id=video0,max_outputs=1,bus=pcie.0,addr=0x1"
|
||||||
|
fi
|
||||||
|
|
||||||
|
ARGS="$DEF_OPTS $CPU_OPTS $RAM_OPTS $MAC_OPTS $MON_OPTS $SERIAL_OPTS $NET_OPTS $DISK_OPTS $EXTRA_OPTS $ARGUMENTS"
|
||||||
|
ARGS=$(echo "$ARGS" | sed 's/\t/ /g' | tr -s ' ')
|
||||||
|
|
||||||
|
return 0
|
||||||
61
src/cpu.sh
Normal file
61
src/cpu.sh
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -Eeuo pipefail
|
||||||
|
|
||||||
|
# Docker environment variables
|
||||||
|
|
||||||
|
: ${HOST_CPU:=''}
|
||||||
|
: ${CPU_MODEL:='host'}
|
||||||
|
: ${CPU_FEATURES:='+ssse3,+sse4.1,+sse4.2'}
|
||||||
|
|
||||||
|
KVM_ERR=""
|
||||||
|
KVM_OPTS=""
|
||||||
|
|
||||||
|
if [[ "$ARCH" == "amd64" && "$KVM" != [Nn]* ]]; then
|
||||||
|
|
||||||
|
if [ -e /dev/kvm ] && sh -c 'echo -n > /dev/kvm' &> /dev/null; then
|
||||||
|
if ! grep -q -e vmx -e svm /proc/cpuinfo; then
|
||||||
|
KVM_ERR="(vmx/svm disabled)"
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
[ -e /dev/kvm ] && KVM_ERR="(no write access)" || KVM_ERR="(device file missing)"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -n "$KVM_ERR" ]; then
|
||||||
|
error "KVM acceleration not detected $KVM_ERR, this will cause a major loss of performance."
|
||||||
|
error "See the FAQ on how to enable it, or skip this error by setting KVM=N (not recommended)."
|
||||||
|
[[ "$DEBUG" != [Yy1]* ]] && exit 88
|
||||||
|
[[ "$CPU_MODEL" == "host"* ]] && CPU_MODEL="max,$CPU_FEATURES"
|
||||||
|
else
|
||||||
|
KVM_OPTS=",accel=kvm -enable-kvm"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -n "$KVM_OPTS" ]; then
|
||||||
|
if ! grep -qE '^flags.* (sse4_2)' /proc/cpuinfo; then
|
||||||
|
error "Your host CPU does not have the SSE4.2 instruction set that Virtual DSM requires to boot."
|
||||||
|
error "Disable KVM by setting KVM=N to emulate a compatible CPU, at the cost of performance."
|
||||||
|
[[ "$DEBUG" != [Yy1]* ]] && exit 89
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
else
|
||||||
|
|
||||||
|
[[ "$CPU_MODEL" == "host"* ]] && CPU_MODEL="max,$CPU_FEATURES"
|
||||||
|
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -z "$HOST_CPU" ]; then
|
||||||
|
HOST_CPU=$(lscpu | grep 'Model name' | cut -f 2 -d ":" | awk '{$1=$1}1' | sed 's# @.*##g' | sed s/"(R)"//g | sed 's/[^[:alnum:] ]\+/ /g' | sed 's/ */ /g')
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -n "$HOST_CPU" ]; then
|
||||||
|
HOST_CPU="${HOST_CPU%%,*},,"
|
||||||
|
else
|
||||||
|
HOST_CPU="QEMU, Virtual CPU,"
|
||||||
|
if [ "$ARCH" == "amd64" ]; then
|
||||||
|
HOST_CPU="$HOST_CPU X86_64"
|
||||||
|
else
|
||||||
|
HOST_CPU="$HOST_CPU $ARCH"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
return 0
|
||||||
496
src/disk.sh
Normal file
496
src/disk.sh
Normal file
@@ -0,0 +1,496 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -Eeuo pipefail
|
||||||
|
|
||||||
|
# Docker environment variables
|
||||||
|
|
||||||
|
: ${DISK_IO:='native'} # I/O Mode, can be set to 'native', 'threads' or 'io_turing'
|
||||||
|
: ${DISK_FMT:='raw'} # Disk file format, 'raw' by default for best performance
|
||||||
|
: ${DISK_CACHE:='none'} # Caching mode, can be set to 'writeback' for better performance
|
||||||
|
: ${DISK_DISCARD:='on'} # Controls whether unmap (TRIM) commands are passed to the host.
|
||||||
|
: ${DISK_ROTATION:='1'} # Rotation rate, set to 1 for SSD storage and increase for HDD
|
||||||
|
: ${DISK_FLAGS:=''} # Specifies the options for use with the qcow2 disk format
|
||||||
|
|
||||||
|
BOOT="$STORAGE/$BASE.boot.img"
|
||||||
|
SYSTEM="$STORAGE/$BASE.system.img"
|
||||||
|
|
||||||
|
[ ! -f "$BOOT" ] && error "Virtual DSM boot-image does not exist ($BOOT)" && exit 81
|
||||||
|
[ ! -f "$SYSTEM" ] && error "Virtual DSM system-image does not exist ($SYSTEM)" && exit 82
|
||||||
|
|
||||||
|
DISK_OPTS="\
|
||||||
|
-object iothread,id=io1 -object iothread,id=io2 \
|
||||||
|
-device virtio-scsi-pci,id=hw-synoboot,iothread=io1,bus=pcie.0,addr=0xa \
|
||||||
|
-drive file=$BOOT,if=none,id=drive-synoboot,format=raw,cache=$DISK_CACHE,aio=$DISK_IO,discard=$DISK_DISCARD,detect-zeroes=on \
|
||||||
|
-device scsi-hd,bus=hw-synoboot.0,channel=0,scsi-id=0,lun=0,drive=drive-synoboot,id=synoboot0,rotation_rate=$DISK_ROTATION,bootindex=1 \
|
||||||
|
-device virtio-scsi-pci,id=hw-synosys,iothread=io1,bus=pcie.0,addr=0xb \
|
||||||
|
-drive file=$SYSTEM,if=none,id=drive-synosys,format=raw,cache=$DISK_CACHE,aio=$DISK_IO,discard=$DISK_DISCARD,detect-zeroes=on \
|
||||||
|
-device scsi-hd,bus=hw-synosys.0,channel=0,scsi-id=0,lun=0,drive=drive-synosys,id=synosys0,rotation_rate=$DISK_ROTATION,bootindex=2"
|
||||||
|
|
||||||
|
fmt2ext() {
|
||||||
|
local DISK_FMT=$1
|
||||||
|
|
||||||
|
case "${DISK_FMT,,}" in
|
||||||
|
qcow2)
|
||||||
|
echo "qcow2"
|
||||||
|
;;
|
||||||
|
raw)
|
||||||
|
echo "img"
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
error "Unrecognized disk format: $DISK_FMT" && exit 78
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
}
|
||||||
|
|
||||||
|
ext2fmt() {
|
||||||
|
local DISK_EXT=$1
|
||||||
|
|
||||||
|
case "${DISK_EXT,,}" in
|
||||||
|
qcow2)
|
||||||
|
echo "qcow2"
|
||||||
|
;;
|
||||||
|
img)
|
||||||
|
echo "raw"
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
error "Unrecognized file extension: .$DISK_EXT" && exit 78
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
}
|
||||||
|
|
||||||
|
getSize() {
|
||||||
|
local DISK_FILE=$1
|
||||||
|
local DISK_EXT DISK_FMT
|
||||||
|
|
||||||
|
DISK_EXT="$(echo "${DISK_FILE//*./}" | sed 's/^.*\.//')"
|
||||||
|
DISK_FMT="$(ext2fmt "$DISK_EXT")"
|
||||||
|
|
||||||
|
case "${DISK_FMT,,}" in
|
||||||
|
raw)
|
||||||
|
stat -c%s "$DISK_FILE"
|
||||||
|
;;
|
||||||
|
qcow2)
|
||||||
|
qemu-img info "$DISK_FILE" -f "$DISK_FMT" | grep '^virtual size: ' | sed 's/.*(\(.*\) bytes)/\1/'
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
error "Unrecognized disk format: $DISK_FMT" && exit 78
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
}
|
||||||
|
|
||||||
|
isCow() {
|
||||||
|
local FS=$1
|
||||||
|
|
||||||
|
if [[ "${FS,,}" == "xfs" || "${FS,,}" == "zfs" || "${FS,,}" == "btrfs" || "${FS,,}" == "bcachefs" ]]; then
|
||||||
|
return 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
return 1
|
||||||
|
}
|
||||||
|
|
||||||
|
createDisk() {
|
||||||
|
local DISK_FILE=$1
|
||||||
|
local DISK_SPACE=$2
|
||||||
|
local DISK_DESC=$3
|
||||||
|
local DISK_FMT=$4
|
||||||
|
local FS=$5
|
||||||
|
local DATA_SIZE DIR SPACE FA
|
||||||
|
|
||||||
|
DATA_SIZE=$(numfmt --from=iec "$DISK_SPACE")
|
||||||
|
|
||||||
|
rm -f "$DISK_FILE"
|
||||||
|
|
||||||
|
if [[ "$ALLOCATE" != [Nn]* ]]; then
|
||||||
|
|
||||||
|
# Check free diskspace
|
||||||
|
DIR=$(dirname "$DISK_FILE")
|
||||||
|
SPACE=$(df --output=avail -B 1 "$DIR" | tail -n 1)
|
||||||
|
|
||||||
|
if (( DATA_SIZE > SPACE )); then
|
||||||
|
local SPACE_GB=$(( (SPACE + 1073741823)/1073741824 ))
|
||||||
|
error "Not enough free space to create a $DISK_DESC of $DISK_SPACE in $DIR, it has only $SPACE_GB GB available..."
|
||||||
|
error "Please specify a smaller ${DISK_DESC^^}_SIZE or disable preallocation by setting ALLOCATE=N." && exit 76
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
info "Creating a $DISK_TYPE $DISK_DESC image in $DISK_FMT format with a size of $DISK_SPACE..."
|
||||||
|
local FAIL="Could not create a $DISK_TYPE $DISK_FMT $DISK_DESC image of $DISK_SPACE ($DISK_FILE)"
|
||||||
|
|
||||||
|
case "${DISK_FMT,,}" in
|
||||||
|
raw)
|
||||||
|
|
||||||
|
if isCow "$FS"; then
|
||||||
|
if ! touch "$DISK_FILE"; then
|
||||||
|
error "$FAIL" && exit 77
|
||||||
|
fi
|
||||||
|
{ chattr +C "$DISK_FILE"; } || :
|
||||||
|
FA=$(lsattr "$DISK_FILE")
|
||||||
|
if [[ "$FA" != *"C"* ]]; then
|
||||||
|
error "Failed to disable COW for $DISK_DESC image $DISK_FILE on ${FS^^} filesystem (returned $FA)"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ "$ALLOCATE" == [Nn]* ]]; then
|
||||||
|
|
||||||
|
# Create an empty file
|
||||||
|
if ! truncate -s "$DATA_SIZE" "$DISK_FILE"; then
|
||||||
|
rm -f "$DISK_FILE"
|
||||||
|
error "$FAIL" && exit 77
|
||||||
|
fi
|
||||||
|
|
||||||
|
else
|
||||||
|
|
||||||
|
# Create an empty file
|
||||||
|
if ! fallocate -l "$DATA_SIZE" "$DISK_FILE"; then
|
||||||
|
if ! truncate -s "$DATA_SIZE" "$DISK_FILE"; then
|
||||||
|
rm -f "$DISK_FILE"
|
||||||
|
error "$FAIL" && exit 77
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
qcow2)
|
||||||
|
|
||||||
|
local DISK_PARAM="$DISK_ALLOC"
|
||||||
|
isCow "$FS" && DISK_PARAM="$DISK_PARAM,nocow=on"
|
||||||
|
[ -n "$DISK_FLAGS" ] && DISK_PARAM="$DISK_PARAM,$DISK_FLAGS"
|
||||||
|
|
||||||
|
if ! qemu-img create -f "$DISK_FMT" -o "$DISK_PARAM" -- "$DISK_FILE" "$DATA_SIZE" ; then
|
||||||
|
rm -f "$DISK_FILE"
|
||||||
|
error "$FAIL" && exit 70
|
||||||
|
fi
|
||||||
|
|
||||||
|
if isCow "$FS"; then
|
||||||
|
FA=$(lsattr "$DISK_FILE")
|
||||||
|
if [[ "$FA" != *"C"* ]]; then
|
||||||
|
error "Failed to disable COW for $DISK_DESC image $DISK_FILE on ${FS^^} filesystem (returned $FA)"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
resizeDisk() {
|
||||||
|
local DISK_FILE=$1
|
||||||
|
local DISK_SPACE=$2
|
||||||
|
local DISK_DESC=$3
|
||||||
|
local DISK_FMT=$4
|
||||||
|
local FS=$5
|
||||||
|
local CUR_SIZE DATA_SIZE DIR SPACE
|
||||||
|
|
||||||
|
CUR_SIZE=$(getSize "$DISK_FILE")
|
||||||
|
DATA_SIZE=$(numfmt --from=iec "$DISK_SPACE")
|
||||||
|
local REQ=$((DATA_SIZE-CUR_SIZE))
|
||||||
|
(( REQ < 1 )) && error "Shrinking disks is not supported yet, please increase ${DISK_DESC^^}_SIZE." && exit 71
|
||||||
|
|
||||||
|
if [[ "$ALLOCATE" != [Nn]* ]]; then
|
||||||
|
|
||||||
|
# Check free diskspace
|
||||||
|
DIR=$(dirname "$DISK_FILE")
|
||||||
|
SPACE=$(df --output=avail -B 1 "$DIR" | tail -n 1)
|
||||||
|
|
||||||
|
if (( REQ > SPACE )); then
|
||||||
|
local SPACE_GB=$(( (SPACE + 1073741823)/1073741824 ))
|
||||||
|
error "Not enough free space to resize $DISK_DESC to $DISK_SPACE in $DIR, it has only $SPACE_GB GB available.."
|
||||||
|
error "Please specify a smaller ${DISK_DESC^^}_SIZE or disable preallocation by setting ALLOCATE=N." && exit 74
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
local GB=$(( (CUR_SIZE + 1073741823)/1073741824 ))
|
||||||
|
info "Resizing $DISK_DESC from ${GB}G to $DISK_SPACE..."
|
||||||
|
local FAIL="Could not resize the $DISK_TYPE $DISK_FMT $DISK_DESC image from ${GB}G to $DISK_SPACE ($DISK_FILE)"
|
||||||
|
|
||||||
|
case "${DISK_FMT,,}" in
|
||||||
|
raw)
|
||||||
|
|
||||||
|
if [[ "$ALLOCATE" == [Nn]* ]]; then
|
||||||
|
|
||||||
|
# Resize file by changing its length
|
||||||
|
if ! truncate -s "$DATA_SIZE" "$DISK_FILE"; then
|
||||||
|
error "$FAIL" && exit 75
|
||||||
|
fi
|
||||||
|
|
||||||
|
else
|
||||||
|
|
||||||
|
# Resize file by allocating more space
|
||||||
|
if ! fallocate -l "$DATA_SIZE" "$DISK_FILE"; then
|
||||||
|
if ! truncate -s "$DATA_SIZE" "$DISK_FILE"; then
|
||||||
|
error "$FAIL" && exit 75
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
qcow2)
|
||||||
|
|
||||||
|
if ! qemu-img resize -f "$DISK_FMT" "--$DISK_ALLOC" "$DISK_FILE" "$DATA_SIZE" ; then
|
||||||
|
error "$FAIL" && exit 72
|
||||||
|
fi
|
||||||
|
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
convertDisk() {
|
||||||
|
local SOURCE_FILE=$1
|
||||||
|
local SOURCE_FMT=$2
|
||||||
|
local DST_FILE=$3
|
||||||
|
local DST_FMT=$4
|
||||||
|
local DISK_BASE=$5
|
||||||
|
local DISK_DESC=$6
|
||||||
|
local FS=$7
|
||||||
|
|
||||||
|
[ -f "$DST_FILE" ] && error "Conversion failed, destination file $DST_FILE already exists?" && exit 79
|
||||||
|
[ ! -f "$SOURCE_FILE" ] && error "Conversion failed, source file $SOURCE_FILE does not exists?" && exit 79
|
||||||
|
|
||||||
|
local TMP_FILE="$DISK_BASE.tmp"
|
||||||
|
rm -f "$TMP_FILE"
|
||||||
|
|
||||||
|
if [[ "$ALLOCATE" != [Nn]* ]]; then
|
||||||
|
|
||||||
|
local DIR CUR_SIZE SPACE
|
||||||
|
|
||||||
|
# Check free diskspace
|
||||||
|
DIR=$(dirname "$TMP_FILE")
|
||||||
|
CUR_SIZE=$(getSize "$SOURCE_FILE")
|
||||||
|
SPACE=$(df --output=avail -B 1 "$DIR" | tail -n 1)
|
||||||
|
|
||||||
|
if (( CUR_SIZE > SPACE )); then
|
||||||
|
local SPACE_GB=$(( (SPACE + 1073741823)/1073741824 ))
|
||||||
|
error "Not enough free space to convert $DISK_DESC to $DST_FMT in $DIR, it has only $SPACE_GB GB available..."
|
||||||
|
error "Please free up some disk space or disable preallocation by setting ALLOCATE=N." && exit 76
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
info "Converting $DISK_DESC to $DST_FMT, please wait until completed..."
|
||||||
|
|
||||||
|
local CONV_FLAGS="-p"
|
||||||
|
local DISK_PARAM="$DISK_ALLOC"
|
||||||
|
isCow "$FS" && DISK_PARAM="$DISK_PARAM,nocow=on"
|
||||||
|
|
||||||
|
if [[ "$DST_FMT" != "raw" ]]; then
|
||||||
|
if [[ "$ALLOCATE" == [Nn]* ]]; then
|
||||||
|
CONV_FLAGS="$CONV_FLAGS -c"
|
||||||
|
fi
|
||||||
|
[ -n "$DISK_FLAGS" ] && DISK_PARAM="$DISK_PARAM,$DISK_FLAGS"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# shellcheck disable=SC2086
|
||||||
|
if ! qemu-img convert -f "$SOURCE_FMT" $CONV_FLAGS -o "$DISK_PARAM" -O "$DST_FMT" -- "$SOURCE_FILE" "$TMP_FILE"; then
|
||||||
|
rm -f "$TMP_FILE"
|
||||||
|
error "Failed to convert $DISK_TYPE $DISK_DESC image to $DST_FMT format in $DIR, is there enough space available?" && exit 79
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ "$DST_FMT" == "raw" ]]; then
|
||||||
|
if [[ "$ALLOCATE" != [Nn]* ]]; then
|
||||||
|
# Work around qemu-img bug
|
||||||
|
CUR_SIZE=$(stat -c%s "$TMP_FILE")
|
||||||
|
if ! fallocate -l "$CUR_SIZE" "$TMP_FILE"; then
|
||||||
|
error "Failed to allocate $CUR_SIZE bytes for $DISK_DESC image $TMP_FILE"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
rm -f "$SOURCE_FILE"
|
||||||
|
mv "$TMP_FILE" "$DST_FILE"
|
||||||
|
|
||||||
|
if isCow "$FS"; then
|
||||||
|
FA=$(lsattr "$DST_FILE")
|
||||||
|
if [[ "$FA" != *"C"* ]]; then
|
||||||
|
error "Failed to disable COW for $DISK_DESC image $DST_FILE on ${FS^^} filesystem (returned $FA)"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
info "Conversion of $DISK_DESC to $DST_FMT completed succesfully!"
|
||||||
|
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
checkFS () {
|
||||||
|
local FS=$1
|
||||||
|
local DISK_FILE=$2
|
||||||
|
local DISK_DESC=$3
|
||||||
|
local DIR FA
|
||||||
|
|
||||||
|
DIR=$(dirname "$DISK_FILE")
|
||||||
|
[ ! -d "$DIR" ] && return 0
|
||||||
|
|
||||||
|
if [[ "${FS,,}" == "overlay"* ]]; then
|
||||||
|
info "Warning: the filesystem of $DIR is OverlayFS, this usually means it was binded to an invalid path!"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if isCow "$FS"; then
|
||||||
|
if [ -f "$DISK_FILE" ]; then
|
||||||
|
FA=$(lsattr "$DISK_FILE")
|
||||||
|
if [[ "$FA" != *"C"* ]]; then
|
||||||
|
info "Warning: COW (copy on write) is not disabled for the $DISK_DESC image file $DISK_FILE, this is recommended on ${FS^^} filesystems!"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
addDisk () {
|
||||||
|
local DISK_ID=$1
|
||||||
|
local DISK_BASE=$2
|
||||||
|
local DISK_EXT=$3
|
||||||
|
local DISK_DESC=$4
|
||||||
|
local DISK_SPACE=$5
|
||||||
|
local DISK_INDEX=$6
|
||||||
|
local DISK_ADDRESS=$7
|
||||||
|
local DISK_FMT=$8
|
||||||
|
local DISK_FILE="$DISK_BASE.$DISK_EXT"
|
||||||
|
local DIR DATA_SIZE FS PREV_FMT PREV_EXT CUR_SIZE
|
||||||
|
|
||||||
|
DIR=$(dirname "$DISK_FILE")
|
||||||
|
[ ! -d "$DIR" ] && return 0
|
||||||
|
|
||||||
|
[ -z "$DISK_SPACE" ] && DISK_SPACE="16G"
|
||||||
|
DISK_SPACE=$(echo "${DISK_SPACE^^}" | sed 's/MB/M/g;s/GB/G/g;s/TB/T/g')
|
||||||
|
DATA_SIZE=$(numfmt --from=iec "$DISK_SPACE")
|
||||||
|
|
||||||
|
if (( DATA_SIZE < 6442450944 )); then
|
||||||
|
if (( DATA_SIZE < 1 )); then
|
||||||
|
error "Invalid value for ${DISK_DESC^^}_SIZE: $DISK_SPACE" && exit 73
|
||||||
|
else
|
||||||
|
error "Please increase ${DISK_DESC^^}_SIZE to at least 6 GB." && exit 73
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
FS=$(stat -f -c %T "$DIR")
|
||||||
|
checkFS "$FS" "$DISK_FILE" "$DISK_DESC" || exit $?
|
||||||
|
|
||||||
|
if ! [ -f "$DISK_FILE" ] ; then
|
||||||
|
|
||||||
|
if [[ "${DISK_FMT,,}" != "raw" ]]; then
|
||||||
|
PREV_FMT="raw"
|
||||||
|
else
|
||||||
|
PREV_FMT="qcow2"
|
||||||
|
fi
|
||||||
|
PREV_EXT="$(fmt2ext "$PREV_FMT")"
|
||||||
|
|
||||||
|
if [ -f "$DISK_BASE.$PREV_EXT" ] ; then
|
||||||
|
convertDisk "$DISK_BASE.$PREV_EXT" "$PREV_FMT" "$DISK_FILE" "$DISK_FMT" "$DISK_BASE" "$DISK_DESC" "$FS" || exit $?
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -f "$DISK_FILE" ]; then
|
||||||
|
|
||||||
|
CUR_SIZE=$(getSize "$DISK_FILE")
|
||||||
|
|
||||||
|
if (( DATA_SIZE > CUR_SIZE )); then
|
||||||
|
resizeDisk "$DISK_FILE" "$DISK_SPACE" "$DISK_DESC" "$DISK_FMT" "$FS" || exit $?
|
||||||
|
fi
|
||||||
|
|
||||||
|
else
|
||||||
|
|
||||||
|
createDisk "$DISK_FILE" "$DISK_SPACE" "$DISK_DESC" "$DISK_FMT" "$FS" || exit $?
|
||||||
|
|
||||||
|
fi
|
||||||
|
|
||||||
|
DISK_OPTS="$DISK_OPTS \
|
||||||
|
-device virtio-scsi-pci,id=hw-$DISK_ID,iothread=io2,bus=pcie.0,addr=$DISK_ADDRESS \
|
||||||
|
-drive file=$DISK_FILE,if=none,id=drive-$DISK_ID,format=$DISK_FMT,cache=$DISK_CACHE,aio=$DISK_IO,discard=$DISK_DISCARD,detect-zeroes=on \
|
||||||
|
-device scsi-hd,bus=hw-$DISK_ID.0,channel=0,scsi-id=0,lun=0,drive=drive-$DISK_ID,id=$DISK_ID,rotation_rate=$DISK_ROTATION,bootindex=$DISK_INDEX"
|
||||||
|
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
DISK_EXT="$(fmt2ext "$DISK_FMT")" || exit $?
|
||||||
|
|
||||||
|
if [[ "$ALLOCATE" == [Nn]* ]]; then
|
||||||
|
DISK_TYPE="growable"
|
||||||
|
DISK_ALLOC="preallocation=off"
|
||||||
|
else
|
||||||
|
DISK_TYPE="preallocated"
|
||||||
|
DISK_ALLOC="preallocation=falloc"
|
||||||
|
fi
|
||||||
|
|
||||||
|
DISK1_FILE="$STORAGE/data"
|
||||||
|
if [[ ! -f "$DISK1_FILE.img" ]] && [[ -f "$STORAGE/data${DISK_SIZE}.img" ]]; then
|
||||||
|
# Fallback for legacy installs
|
||||||
|
mv "$STORAGE/data${DISK_SIZE}.img" "$DISK1_FILE.img"
|
||||||
|
fi
|
||||||
|
|
||||||
|
DISK2_FILE="/storage2/data2"
|
||||||
|
if [ ! -f "$DISK2_FILE.img" ]; then
|
||||||
|
# Fallback for legacy installs
|
||||||
|
FALLBACK="/storage2/data.img"
|
||||||
|
if [[ -f "$DISK1_FILE.img" ]] && [[ -f "$FALLBACK" ]]; then
|
||||||
|
SIZE1=$(stat -c%s "$FALLBACK")
|
||||||
|
SIZE2=$(stat -c%s "$DISK1_FILE.img")
|
||||||
|
if [[ SIZE1 -ne SIZE2 ]]; then
|
||||||
|
mv "$FALLBACK" "$DISK2_FILE.img"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
DISK3_FILE="/storage3/data3"
|
||||||
|
if [ ! -f "$DISK3_FILE.img" ]; then
|
||||||
|
# Fallback for legacy installs
|
||||||
|
FALLBACK="/storage3/data.img"
|
||||||
|
if [[ -f "$DISK1_FILE.img" ]] && [[ -f "$FALLBACK" ]]; then
|
||||||
|
SIZE1=$(stat -c%s "$FALLBACK")
|
||||||
|
SIZE2=$(stat -c%s "$DISK1_FILE.img")
|
||||||
|
if [[ SIZE1 -ne SIZE2 ]]; then
|
||||||
|
mv "$FALLBACK" "$DISK3_FILE.img"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
DISK4_FILE="/storage4/data4"
|
||||||
|
DISK5_FILE="/storage5/data5"
|
||||||
|
DISK6_FILE="/storage6/data6"
|
||||||
|
|
||||||
|
: ${DISK2_SIZE:=''}
|
||||||
|
: ${DISK3_SIZE:=''}
|
||||||
|
: ${DISK4_SIZE:=''}
|
||||||
|
: ${DISK5_SIZE:=''}
|
||||||
|
: ${DISK6_SIZE:=''}
|
||||||
|
|
||||||
|
addDisk "userdata" "$DISK1_FILE" "$DISK_EXT" "disk" "$DISK_SIZE" "3" "0xc" "$DISK_FMT" || exit $?
|
||||||
|
addDisk "userdata2" "$DISK2_FILE" "$DISK_EXT" "disk2" "$DISK2_SIZE" "4" "0xd" "$DISK_FMT" || exit $?
|
||||||
|
addDisk "userdata3" "$DISK3_FILE" "$DISK_EXT" "disk3" "$DISK3_SIZE" "5" "0xe" "$DISK_FMT" || exit $?
|
||||||
|
addDisk "userdata4" "$DISK4_FILE" "$DISK_EXT" "disk4" "$DISK4_SIZE" "9" "0x7" "$DISK_FMT" || exit $?
|
||||||
|
addDisk "userdata5" "$DISK5_FILE" "$DISK_EXT" "disk5" "$DISK5_SIZE" "10" "0x8" "$DISK_FMT" || exit $?
|
||||||
|
addDisk "userdata6" "$DISK6_FILE" "$DISK_EXT" "disk6" "$DISK6_SIZE" "11" "0x9" "$DISK_FMT" || exit $?
|
||||||
|
|
||||||
|
addDevice () {
|
||||||
|
|
||||||
|
local DISK_ID=$1
|
||||||
|
local DISK_DEV=$2
|
||||||
|
local DISK_INDEX=$3
|
||||||
|
local DISK_ADDRESS=$4
|
||||||
|
|
||||||
|
[ -z "$DISK_DEV" ] && return 0
|
||||||
|
[ ! -b "$DISK_DEV" ] && error "Device $DISK_DEV cannot be found! Please add it to the 'devices' section of your compose file." && exit 55
|
||||||
|
|
||||||
|
DISK_OPTS="$DISK_OPTS \
|
||||||
|
-device virtio-scsi-pci,id=hw-$DISK_ID,iothread=io2,bus=pcie.0,addr=$DISK_ADDRESS \
|
||||||
|
-drive file=$DISK_DEV,if=none,id=drive-$DISK_ID,format=raw,cache=$DISK_CACHE,aio=$DISK_IO,discard=$DISK_DISCARD,detect-zeroes=on \
|
||||||
|
-device scsi-hd,bus=hw-$DISK_ID.0,channel=0,scsi-id=0,lun=0,drive=drive-$DISK_ID,id=$DISK_ID,rotation_rate=$DISK_ROTATION,bootindex=$DISK_INDEX"
|
||||||
|
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
: ${DEVICE:=''} # Docker variable to passthrough a block device, like /dev/vdc1.
|
||||||
|
: ${DEVICE2:=''}
|
||||||
|
: ${DEVICE3:=''}
|
||||||
|
: ${DEVICE4:=''}
|
||||||
|
: ${DEVICE5:=''}
|
||||||
|
: ${DEVICE6:=''}
|
||||||
|
|
||||||
|
addDevice "userdata7" "$DEVICE" "6" "0xf" || exit $?
|
||||||
|
addDevice "userdata8" "$DEVICE2" "7" "0x5" || exit $?
|
||||||
|
addDevice "userdata9" "$DEVICE3" "8" "0x6" || exit $?
|
||||||
|
addDevice "userdata4" "$DEVICE4" "9" "0x7" || exit $?
|
||||||
|
addDevice "userdata5" "$DEVICE5" "10" "0x8" || exit $?
|
||||||
|
addDevice "userdata6" "$DEVICE6" "11" "0x9" || exit $?
|
||||||
|
|
||||||
|
return 0
|
||||||
34
src/entry.sh
Executable file
34
src/entry.sh
Executable file
@@ -0,0 +1,34 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -Eeuo pipefail
|
||||||
|
|
||||||
|
echo "❯ Starting Virtual DSM for Docker v$(</run/version)..."
|
||||||
|
echo "❯ For support visit https://github.com/vdsm/virtual-dsm/"
|
||||||
|
|
||||||
|
cd /run
|
||||||
|
|
||||||
|
. reset.sh # Initialize system
|
||||||
|
. install.sh # Run installation
|
||||||
|
. disk.sh # Initialize disks
|
||||||
|
. network.sh # Initialize network
|
||||||
|
. gpu.sh # Initialize graphics
|
||||||
|
. cpu.sh # Initialize processor
|
||||||
|
. serial.sh # Initialize serialport
|
||||||
|
. power.sh # Configure shutdown
|
||||||
|
. config.sh # Configure arguments
|
||||||
|
|
||||||
|
trap - ERR
|
||||||
|
|
||||||
|
if [[ "$CONSOLE" == [Yy]* ]]; then
|
||||||
|
exec qemu-system-x86_64 -pidfile "$QEMU_PID" ${ARGS:+ $ARGS}
|
||||||
|
exit $?
|
||||||
|
fi
|
||||||
|
|
||||||
|
set -m
|
||||||
|
(
|
||||||
|
[[ "$DEBUG" == [Yy1]* ]] && info "$VERS" && set -x
|
||||||
|
qemu-system-x86_64 ${ARGS:+ $ARGS} & echo $! > "$QEMU_PID"
|
||||||
|
{ set +x; } 2>/dev/null
|
||||||
|
)
|
||||||
|
set +m
|
||||||
|
|
||||||
|
tail --pid "$(cat "$QEMU_PID")" --follow /dev/null & wait $!
|
||||||
24
src/gpu.sh
Normal file
24
src/gpu.sh
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -Eeuo pipefail
|
||||||
|
|
||||||
|
if [[ "$GPU" != [Yy1]* ]] || [[ "$ARCH" != "amd64" ]]; then
|
||||||
|
return 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
[ ! -d /dev/dri ] && mkdir -m 755 /dev/dri
|
||||||
|
|
||||||
|
if [ ! -c /dev/dri/card0 ]; then
|
||||||
|
mknod /dev/dri/card0 c 226 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ ! -c /dev/dri/renderD128 ]; then
|
||||||
|
mknod /dev/dri/renderD128 c 226 128
|
||||||
|
fi
|
||||||
|
|
||||||
|
chmod 666 /dev/dri/card0
|
||||||
|
chmod 666 /dev/dri/renderD128
|
||||||
|
|
||||||
|
addPackage "xserver-xorg-video-intel" "Intel GPU drivers"
|
||||||
|
addPackage "qemu-system-modules-opengl" "OpenGL module"
|
||||||
|
|
||||||
|
return 0
|
||||||
@@ -1,11 +1,10 @@
|
|||||||
#!/usr/bin/env bash
|
#!/usr/bin/env bash
|
||||||
set -Eeuo pipefail
|
set -Eeuo pipefail
|
||||||
|
|
||||||
STORAGE="/storage"
|
: ${URL:=''} # URL of the PAT file to be downloaded.
|
||||||
[ ! -d "$STORAGE" ] && error "Storage folder (${STORAGE}) not found!" && exit 13
|
|
||||||
|
|
||||||
if [ -f "$STORAGE"/dsm.ver ]; then
|
if [ -f "$STORAGE"/dsm.ver ]; then
|
||||||
BASE=$(cat "${STORAGE}/dsm.ver")
|
BASE=$(cat "$STORAGE/dsm.ver")
|
||||||
else
|
else
|
||||||
# Fallback for old installs
|
# Fallback for old installs
|
||||||
BASE="DSM_VirtualDSM_42962"
|
BASE="DSM_VirtualDSM_42962"
|
||||||
@@ -14,26 +13,69 @@ fi
|
|||||||
[ -n "$URL" ] && BASE=$(basename "$URL" .pat)
|
[ -n "$URL" ] && BASE=$(basename "$URL" .pat)
|
||||||
|
|
||||||
if [[ -f "$STORAGE/$BASE.boot.img" ]] && [[ -f "$STORAGE/$BASE.system.img" ]]; then
|
if [[ -f "$STORAGE/$BASE.boot.img" ]] && [[ -f "$STORAGE/$BASE.system.img" ]]; then
|
||||||
# Previous installation found
|
return 0 # Previous installation found
|
||||||
return 0
|
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Display wait message
|
# Display wait message
|
||||||
/run/server.sh 5000 install &
|
/run/server.sh 5000 install &
|
||||||
|
|
||||||
# Download the required files from the Synology website
|
DL=""
|
||||||
DL="https://global.synologydownload.com/download/DSM"
|
DL_CHINA="https://cndl.synology.cn/download/DSM"
|
||||||
|
DL_GLOBAL="https://global.synologydownload.com/download/DSM"
|
||||||
|
|
||||||
if [ -z "$URL" ]; then
|
[[ "${URL,,}" == *"cndl.synology"* ]] && DL="$DL_CHINA"
|
||||||
|
[[ "${URL,,}" == *"global.synology"* ]] && DL="$DL_GLOBAL"
|
||||||
if [ "$ARCH" == "amd64" ]; then
|
|
||||||
URL="$DL/release/7.2.1/69057-1/DSM_VirtualDSM_69057.pat"
|
|
||||||
else
|
|
||||||
URL="$DL/release/7.0.1/42218/DSM_VirtualDSM_42218.pat"
|
|
||||||
fi
|
|
||||||
|
|
||||||
|
if [ -z "$DL" ]; then
|
||||||
|
[ -z "$COUNTRY" ] && setCountry
|
||||||
|
[ -z "$COUNTRY" ] && info "Warning: could not detect country to select mirror!"
|
||||||
|
[[ "${COUNTRY^^}" == "CN" ]] && DL="$DL_CHINA" || DL="$DL_GLOBAL"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
[ -z "$URL" ] && URL="$DL/release/7.2.1/69057-1/DSM_VirtualDSM_69057.pat"
|
||||||
|
|
||||||
|
BASE=$(basename "$URL" .pat)
|
||||||
|
|
||||||
|
if [[ "$URL" != "file://$STORAGE/$BASE.pat" ]]; then
|
||||||
|
rm -f "$STORAGE"/"$BASE".pat
|
||||||
|
fi
|
||||||
|
|
||||||
|
rm -f "$STORAGE"/"$BASE".agent
|
||||||
|
rm -f "$STORAGE"/"$BASE".boot.img
|
||||||
|
rm -f "$STORAGE"/"$BASE".system.img
|
||||||
|
|
||||||
|
[[ "$DEBUG" == [Yy1]* ]] && set -x
|
||||||
|
|
||||||
|
# Check filesystem
|
||||||
|
MIN_ROOT=471859200
|
||||||
|
MIN_SPACE=6442450944
|
||||||
|
FS=$(stat -f -c %T "$STORAGE")
|
||||||
|
|
||||||
|
if [[ "${FS,,}" == "overlay"* ]]; then
|
||||||
|
info "Warning: the filesystem of $STORAGE is OverlayFS, this usually means it was binded to an invalid path!"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ "${FS,,}" != "fat"* && "${FS,,}" != "vfat"* && "${FS,,}" != "exfat"* && \
|
||||||
|
"${FS,,}" != "ntfs"* && "${FS,,}" != "fuse"* && "${FS,,}" != "msdos"* ]]; then
|
||||||
|
TMP="$STORAGE/tmp"
|
||||||
|
else
|
||||||
|
TMP="/tmp/dsm"
|
||||||
|
SPACE=$(df --output=avail -B 1 /tmp | tail -n 1)
|
||||||
|
if (( MIN_SPACE > SPACE )); then
|
||||||
|
error "The ${FS^^} filesystem of $STORAGE does not support UNIX permissions, and no space left in container!" && exit 93
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
rm -rf "$TMP" && mkdir -p "$TMP"
|
||||||
|
|
||||||
|
# Check free diskspace
|
||||||
|
SPACE=$(df --output=avail -B 1 / | tail -n 1)
|
||||||
|
(( MIN_ROOT > SPACE )) && error "Not enough free space in container root, need at least 450 MB available." && exit 96
|
||||||
|
|
||||||
|
SPACE=$(df --output=avail -B 1 "$STORAGE" | tail -n 1)
|
||||||
|
SPACE_GB=$(( (SPACE + 1073741823)/1073741824 ))
|
||||||
|
(( MIN_SPACE > SPACE )) && error "Not enough free space for installation in $STORAGE, have $SPACE_GB GB available but need at least 6 GB." && exit 94
|
||||||
|
|
||||||
# Check if output is to interactive TTY
|
# Check if output is to interactive TTY
|
||||||
if [ -t 1 ]; then
|
if [ -t 1 ]; then
|
||||||
PROGRESS="--progress=bar:noscroll"
|
PROGRESS="--progress=bar:noscroll"
|
||||||
@@ -41,38 +83,12 @@ else
|
|||||||
PROGRESS="--progress=dot:giga"
|
PROGRESS="--progress=dot:giga"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
BASE=$(basename "$URL" .pat)
|
# Download the required files from the Synology website
|
||||||
|
|
||||||
rm -f "$STORAGE"/"$BASE".pat
|
|
||||||
rm -f "$STORAGE"/"$BASE".agent
|
|
||||||
rm -f "$STORAGE"/"$BASE".boot.img
|
|
||||||
rm -f "$STORAGE"/"$BASE".system.img
|
|
||||||
|
|
||||||
MIN_SPACE=6442450944
|
|
||||||
FS=$(stat -f -c %T "$STORAGE")
|
|
||||||
|
|
||||||
if [[ "$FS" != "fat"* && "$FS" != "vfat"* && "$FS" != "exfat"* && \
|
|
||||||
"$FS" != "ntfs"* && "$FS" != "fuse"* && "$FS" != "msdos"* ]]; then
|
|
||||||
TMP="$STORAGE/tmp"
|
|
||||||
else
|
|
||||||
TMP="/tmp/dsm"
|
|
||||||
SPACE=$(df --output=avail -B 1 /tmp | tail -n 1)
|
|
||||||
(( MIN_SPACE > SPACE )) && TMP="$STORAGE/tmp"
|
|
||||||
fi
|
|
||||||
|
|
||||||
rm -rf /tmp/dsm
|
|
||||||
rm -rf "$STORAGE/tmp"
|
|
||||||
rm -rf "$TMP" && mkdir -p "$TMP"
|
|
||||||
|
|
||||||
# Check free diskspace
|
|
||||||
SPACE=$(df --output=avail -B 1 "$TMP" | tail -n 1)
|
|
||||||
(( MIN_SPACE > SPACE )) && error "Not enough free space for installation, need at least 6 GB." && exit 95
|
|
||||||
|
|
||||||
[[ "${DEBUG}" == [Yy1]* ]] && set -x
|
|
||||||
|
|
||||||
|
ROOT="Y"
|
||||||
RDC="$STORAGE/dsm.rd"
|
RDC="$STORAGE/dsm.rd"
|
||||||
|
|
||||||
if [ ! -f "${RDC}" ]; then
|
if [ ! -f "$RDC" ]; then
|
||||||
|
|
||||||
info "Install: Downloading installer..."
|
info "Install: Downloading installer..."
|
||||||
|
|
||||||
@@ -81,7 +97,7 @@ if [ ! -f "${RDC}" ]; then
|
|||||||
VERIFY="b4215a4b213ff5154db0488f92c87864"
|
VERIFY="b4215a4b213ff5154db0488f92c87864"
|
||||||
LOC="$DL/release/7.0.1/42218/DSM_VirtualDSM_42218.pat"
|
LOC="$DL/release/7.0.1/42218/DSM_VirtualDSM_42218.pat"
|
||||||
|
|
||||||
{ curl -r "$POS" -sfk -o "$RD" "$LOC"; rc=$?; } || :
|
{ curl -r "$POS" -sfk -S -o "$RD" "$LOC"; rc=$?; } || :
|
||||||
(( rc != 0 )) && error "Failed to download $LOC, reason: $rc" && exit 60
|
(( rc != 0 )) && error "Failed to download $LOC, reason: $rc" && exit 60
|
||||||
|
|
||||||
SUM=$(md5sum "$RD" | cut -f 1 -d " ")
|
SUM=$(md5sum "$RD" | cut -f 1 -d " ")
|
||||||
@@ -95,7 +111,7 @@ if [ ! -f "${RDC}" ]; then
|
|||||||
{ wget "$LOC" -O "$PAT" -q --no-check-certificate --show-progress "$PROGRESS"; rc=$?; } || :
|
{ wget "$LOC" -O "$PAT" -q --no-check-certificate --show-progress "$PROGRESS"; rc=$?; } || :
|
||||||
(( rc != 0 )) && error "Failed to download $LOC, reason: $rc" && exit 60
|
(( rc != 0 )) && error "Failed to download $LOC, reason: $rc" && exit 60
|
||||||
|
|
||||||
tar --extract --file="$PAT" --directory="$(dirname "${RD}")"/. "$(basename "${RD}")"
|
tar --extract --file="$PAT" --directory="$(dirname "$RD")"/. "$(basename "$RD")"
|
||||||
rm "$PAT"
|
rm "$PAT"
|
||||||
|
|
||||||
fi
|
fi
|
||||||
@@ -104,13 +120,18 @@ if [ ! -f "${RDC}" ]; then
|
|||||||
|
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if [ -f "${RDC}" ]; then
|
if [ -f "$RDC" ]; then
|
||||||
|
|
||||||
{ xz -dc <"$RDC" >"$TMP/rd" 2>/dev/null; rc=$?; } || :
|
{ xz -dc <"$RDC" >"$TMP/rd" 2>/dev/null; rc=$?; } || :
|
||||||
(( rc != 1 )) && error "Failed to unxz $RDC, reason $rc" && exit 91
|
(( rc != 1 )) && error "Failed to unxz $RDC, reason $rc" && exit 91
|
||||||
|
|
||||||
{ (cd "$TMP" && cpio -idm <"$TMP/rd" 2>/dev/null); rc=$?; } || :
|
{ (cd "$TMP" && cpio -idm <"$TMP/rd" 2>/dev/null); rc=$?; } || :
|
||||||
(( rc != 0 )) && error "Failed to cpio $RDC, reason $rc" && exit 92
|
|
||||||
|
if (( rc != 0 )); then
|
||||||
|
ROOT="N"
|
||||||
|
{ (cd "$TMP" && fakeroot cpio -idmu <"$TMP/rd" 2>/dev/null); rc=$?; } || :
|
||||||
|
(( rc != 0 )) && error "Failed to extract $RDC, reason $rc" && exit 92
|
||||||
|
fi
|
||||||
|
|
||||||
mkdir -p /run/extract
|
mkdir -p /run/extract
|
||||||
for file in $TMP/usr/lib/libcurl.so.4 \
|
for file in $TMP/usr/lib/libcurl.so.4 \
|
||||||
@@ -143,9 +164,17 @@ info "Install: Downloading $(basename "$URL")..."
|
|||||||
PAT="/$BASE.pat"
|
PAT="/$BASE.pat"
|
||||||
rm -f "$PAT"
|
rm -f "$PAT"
|
||||||
|
|
||||||
{ wget "$URL" -O "$PAT" -q --no-check-certificate --show-progress "$PROGRESS"; rc=$?; } || :
|
if [[ "$URL" == "file://"* ]]; then
|
||||||
|
|
||||||
|
cp "${URL:7}" "$PAT"
|
||||||
|
|
||||||
|
else
|
||||||
|
|
||||||
|
{ wget "$URL" -O "$PAT" -q --no-check-certificate --show-progress "$PROGRESS"; rc=$?; } || :
|
||||||
|
(( rc != 0 )) && error "Failed to download $URL, reason: $rc" && exit 69
|
||||||
|
|
||||||
|
fi
|
||||||
|
|
||||||
(( rc != 0 )) && error "Failed to download $URL, reason: $rc" && exit 69
|
|
||||||
[ ! -f "$PAT" ] && error "Failed to download $URL" && exit 69
|
[ ! -f "$PAT" ] && error "Failed to download $URL" && exit 69
|
||||||
|
|
||||||
SIZE=$(stat -c%s "$PAT")
|
SIZE=$(stat -c%s "$PAT")
|
||||||
@@ -154,27 +183,14 @@ if ((SIZE<250000000)); then
|
|||||||
error "The specified PAT file is probably an update pack as it's too small." && exit 62
|
error "The specified PAT file is probably an update pack as it's too small." && exit 62
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
info "Install: Extracting downloaded image..."
|
||||||
|
|
||||||
if { tar tf "$PAT"; } >/dev/null 2>&1; then
|
if { tar tf "$PAT"; } >/dev/null 2>&1; then
|
||||||
|
|
||||||
info "Install: Extracting downloaded image..."
|
|
||||||
tar xpf "$PAT" -C "$TMP/."
|
tar xpf "$PAT" -C "$TMP/."
|
||||||
|
|
||||||
else
|
else
|
||||||
|
|
||||||
if [ "$ARCH" != "amd64" ]; then
|
|
||||||
|
|
||||||
info "Install: Installing QEMU..."
|
|
||||||
|
|
||||||
export DEBCONF_NOWARNINGS="yes"
|
|
||||||
export DEBIAN_FRONTEND="noninteractive"
|
|
||||||
|
|
||||||
apt-get -qq update
|
|
||||||
apt-get -qq --no-install-recommends -y install qemu-user > /dev/null
|
|
||||||
|
|
||||||
fi
|
|
||||||
|
|
||||||
info "Install: Extracting downloaded image..."
|
|
||||||
|
|
||||||
export LD_LIBRARY_PATH="/run/extract"
|
export LD_LIBRARY_PATH="/run/extract"
|
||||||
|
|
||||||
if [ "$ARCH" == "amd64" ]; then
|
if [ "$ARCH" == "amd64" ]; then
|
||||||
@@ -202,38 +218,50 @@ BOOT=$(find "$TMP" -name "*.bin.zip")
|
|||||||
BOOT=$(echo "$BOOT" | head -c -5)
|
BOOT=$(echo "$BOOT" | head -c -5)
|
||||||
unzip -q -o "$BOOT".zip -d "$TMP"
|
unzip -q -o "$BOOT".zip -d "$TMP"
|
||||||
|
|
||||||
[[ "${ALLOCATE}" == [Zz]* ]] && info "Install: Allocating diskspace..."
|
|
||||||
|
|
||||||
SYSTEM="$TMP/sys.img"
|
SYSTEM="$TMP/sys.img"
|
||||||
SYSTEM_SIZE=4954537983
|
SYSTEM_SIZE=4954537983
|
||||||
|
rm -f "$SYSTEM"
|
||||||
|
|
||||||
# Check free diskspace
|
# Check free diskspace
|
||||||
SPACE=$(df --output=avail -B 1 "$TMP" | tail -n 1)
|
SPACE=$(df --output=avail -B 1 "$TMP" | tail -n 1)
|
||||||
(( SYSTEM_SIZE > SPACE )) && error "Not enough free space to create a 4 GB system disk." && exit 87
|
SPACE_GB=$(( (SPACE + 1073741823)/1073741824 ))
|
||||||
|
(( SYSTEM_SIZE > SPACE )) && error "Not enough free space to create a 4 GB system disk, have only $SPACE_GB GB available." && exit 97
|
||||||
|
|
||||||
if ! fallocate -l "${SYSTEM_SIZE}" "${SYSTEM}"; then
|
if ! touch "$SYSTEM"; then
|
||||||
if ! truncate -s "${SYSTEM_SIZE}" "${SYSTEM}"; then
|
error "Could not create file $SYSTEM for the system disk." && exit 98
|
||||||
rm -f "${SYSTEM}" && error "Could not allocate a file for the system disk." && exit 88
|
fi
|
||||||
|
|
||||||
|
if [[ "${FS,,}" == "xfs" || "${FS,,}" == "zfs" || "${FS,,}" == "btrfs" || "${FS,,}" == "bcachefs" ]]; then
|
||||||
|
{ chattr +C "$SYSTEM"; } || :
|
||||||
|
FA=$(lsattr "$SYSTEM")
|
||||||
|
if [[ "$FA" != *"C"* ]]; then
|
||||||
|
error "Failed to disable COW for system image $SYSTEM on ${FS^^} filesystem (returned $FA)"
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if [[ "${ALLOCATE}" == [Zz]* ]]; then
|
if ! fallocate -l "$SYSTEM_SIZE" "$SYSTEM"; then
|
||||||
info "Install: Preallocating 4 GB of diskspace..."
|
if ! truncate -s "$SYSTEM_SIZE" "$SYSTEM"; then
|
||||||
dd if=/dev/urandom of="${SYSTEM}" count="${SYSTEM_SIZE}" bs=1M iflag=count_bytes status=none
|
rm -f "$SYSTEM"
|
||||||
|
error "Could not allocate file $SYSTEM for the system disk." && exit 98
|
||||||
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Check if file exists
|
# Check if file exists
|
||||||
[ ! -f "${SYSTEM}" ] && error "System disk does not exist ($SYSTEM)" && exit 89
|
[ ! -f "$SYSTEM" ] && error "System disk does not exist ($SYSTEM)" && exit 99
|
||||||
|
|
||||||
# Check the filesize
|
# Check the filesize
|
||||||
SIZE=$(stat -c%s "${SYSTEM}")
|
SIZE=$(stat -c%s "$SYSTEM")
|
||||||
[[ SIZE -ne SYSTEM_SIZE ]] && rm -f "${SYSTEM}" && error "System disk has the wrong size: ${SIZE}" && exit 90
|
|
||||||
|
if [[ SIZE -ne SYSTEM_SIZE ]]; then
|
||||||
|
rm -f "$SYSTEM"
|
||||||
|
error "System disk has the wrong size: $SIZE vs $SYSTEM_SIZE" && exit 90
|
||||||
|
fi
|
||||||
|
|
||||||
PART="$TMP/partition.fdisk"
|
PART="$TMP/partition.fdisk"
|
||||||
|
|
||||||
{ echo "label: dos"
|
{ echo "label: dos"
|
||||||
echo "label-id: 0x6f9ee2e9"
|
echo "label-id: 0x6f9ee2e9"
|
||||||
echo "device: ${SYSTEM}"
|
echo "device: $SYSTEM"
|
||||||
echo "unit: sectors"
|
echo "unit: sectors"
|
||||||
echo "sector-size: 512"
|
echo "sector-size: 512"
|
||||||
echo ""
|
echo ""
|
||||||
@@ -245,43 +273,63 @@ sfdisk -q "$SYSTEM" < "$PART"
|
|||||||
|
|
||||||
info "Install: Extracting system partition..."
|
info "Install: Extracting system partition..."
|
||||||
|
|
||||||
|
LABEL="1.44.1-42218"
|
||||||
|
OFFSET="1048576" # 2048 * 512
|
||||||
|
NUMBLOCKS="622560" # (4980480 * 512) / 4096
|
||||||
|
|
||||||
MOUNT="$TMP/system"
|
MOUNT="$TMP/system"
|
||||||
rm -rf "$MOUNT" && mkdir -p "$MOUNT"
|
rm -rf "$MOUNT" && mkdir -p "$MOUNT"
|
||||||
|
|
||||||
mv "$HDA.tgz" "$HDA.txz"
|
mv "$HDA.tgz" "$HDA.txz"
|
||||||
tar xpfJ "$HDA.txz" --absolute-names -C "$MOUNT/"
|
|
||||||
|
if [[ "$ROOT" != [Nn]* ]]; then
|
||||||
|
|
||||||
|
tar xpfJ "$HDA.txz" --absolute-names -C "$MOUNT/"
|
||||||
|
|
||||||
|
fi
|
||||||
|
|
||||||
[ -d "$PKG" ] && mv "$PKG/" "$MOUNT/.SynoUpgradePackages/"
|
[ -d "$PKG" ] && mv "$PKG/" "$MOUNT/.SynoUpgradePackages/"
|
||||||
rm -f "$MOUNT/.SynoUpgradePackages/ActiveInsight-"*
|
rm -f "$MOUNT/.SynoUpgradePackages/ActiveInsight-"*
|
||||||
|
|
||||||
[ -f "$HDP.txz" ] && tar xpfJ "$HDP.txz" --absolute-names -C "$MOUNT/"
|
[ -f "$HDP.txz" ] && tar xpfJ "$HDP.txz" --absolute-names -C "$MOUNT/"
|
||||||
[ -f "$IDB.txz" ] && tar xpfJ "$IDB.txz" --absolute-names -C "$MOUNT/usr/syno/synoman/indexdb/"
|
|
||||||
|
|
||||||
info "Install: Installing system partition..."
|
if [ -f "$IDB.txz" ]; then
|
||||||
|
INDEX_DB="$MOUNT/usr/syno/synoman/indexdb/"
|
||||||
|
mkdir -p "$INDEX_DB"
|
||||||
|
tar xpfJ "$IDB.txz" --absolute-names -C "$INDEX_DB"
|
||||||
|
fi
|
||||||
|
|
||||||
LABEL="1.44.1-42218"
|
if [[ "$ROOT" != [Nn]* ]]; then
|
||||||
OFFSET="1048576" # 2048 * 512
|
|
||||||
NUMBLOCKS="622560" # (4980480 * 512) / 4096
|
|
||||||
|
|
||||||
mke2fs -q -t ext4 -b 4096 -d "$MOUNT/" -L "$LABEL" -F -E "offset=$OFFSET" "$SYSTEM" "$NUMBLOCKS"
|
info "Install: Installing system partition..."
|
||||||
|
|
||||||
|
mke2fs -q -t ext4 -b 4096 -d "$MOUNT/" -L "$LABEL" -F -E "offset=$OFFSET" "$SYSTEM" "$NUMBLOCKS"
|
||||||
|
|
||||||
|
else
|
||||||
|
|
||||||
|
fakeroot -- bash -c "set -Eeu;\
|
||||||
|
tar xpfJ $HDA.txz --absolute-names --skip-old-files -C $MOUNT/;\
|
||||||
|
printf '%b%s%b' '\E[1;34m❯ \E[1;36m' 'Install: Installing system partition...' '\E[0m\n';\
|
||||||
|
mke2fs -q -t ext4 -b 4096 -d $MOUNT/ -L $LABEL -F -E offset=$OFFSET $SYSTEM $NUMBLOCKS"
|
||||||
|
|
||||||
|
fi
|
||||||
|
|
||||||
rm -rf "$MOUNT"
|
rm -rf "$MOUNT"
|
||||||
|
|
||||||
echo "$BASE" > "$STORAGE"/dsm.ver
|
echo "$BASE" > "$STORAGE"/dsm.ver
|
||||||
|
|
||||||
if [[ "$TMP" != "$STORAGE/tmp" ]]; then
|
if [[ "$URL" == "file://$STORAGE/$BASE.pat" ]]; then
|
||||||
# Check free diskspace
|
rm -f "$PAT"
|
||||||
SPACE=$(df --output=avail -B 1 "$STORAGE" | tail -n 1)
|
else
|
||||||
(( MIN_SPACE > SPACE )) && error "Not enough free space in storage folder, need at least 6 GB." && exit 94
|
mv -f "$PAT" "$STORAGE"/"$BASE".pat
|
||||||
fi
|
fi
|
||||||
|
|
||||||
mv -f "$PAT" "$STORAGE"/"$BASE".pat
|
|
||||||
mv -f "$BOOT" "$STORAGE"/"$BASE".boot.img
|
mv -f "$BOOT" "$STORAGE"/"$BASE".boot.img
|
||||||
mv -f "$SYSTEM" "$STORAGE"/"$BASE".system.img
|
mv -f "$SYSTEM" "$STORAGE"/"$BASE".system.img
|
||||||
|
|
||||||
rm -rf "$TMP"
|
rm -rf "$TMP"
|
||||||
|
|
||||||
{ set +x; } 2>/dev/null
|
{ set +x; } 2>/dev/null
|
||||||
[[ "${DEBUG}" == [Yy1]* ]] && echo
|
[[ "$DEBUG" == [Yy1]* ]] && echo
|
||||||
|
|
||||||
return 0
|
return 0
|
||||||
@@ -11,7 +11,6 @@ set -Eeuo pipefail
|
|||||||
: ${VM_NET_MAC:="$MAC"}
|
: ${VM_NET_MAC:="$MAC"}
|
||||||
: ${VM_NET_HOST:='VirtualDSM'}
|
: ${VM_NET_HOST:='VirtualDSM'}
|
||||||
|
|
||||||
: ${DNS_SERVERS:=''}
|
|
||||||
: ${DNSMASQ_OPTS:=''}
|
: ${DNSMASQ_OPTS:=''}
|
||||||
: ${DNSMASQ:='/usr/sbin/dnsmasq'}
|
: ${DNSMASQ:='/usr/sbin/dnsmasq'}
|
||||||
: ${DNSMASQ_CONF_DIR:='/etc/dnsmasq.d'}
|
: ${DNSMASQ_CONF_DIR:='/etc/dnsmasq.d'}
|
||||||
@@ -24,37 +23,38 @@ configureDHCP() {
|
|||||||
|
|
||||||
# Create a macvtap network for the VM guest
|
# Create a macvtap network for the VM guest
|
||||||
|
|
||||||
{ ip link add link "${VM_NET_DEV}" name "${VM_NET_TAP}" address "${VM_NET_MAC}" type macvtap mode bridge ; rc=$?; } || :
|
{ ip link add link "$VM_NET_DEV" name "$VM_NET_TAP" address "$VM_NET_MAC" type macvtap mode bridge ; rc=$?; } || :
|
||||||
|
|
||||||
if (( rc != 0 )); then
|
if (( rc != 0 )); then
|
||||||
error "Cannot create macvtap interface. Please make sure the network type is 'macvlan' and not 'ipvlan',"
|
error "Cannot create macvtap interface. Please make sure the network type is 'macvlan' and not 'ipvlan',"
|
||||||
error "and that the NET_ADMIN capability has been added to the container config: --cap-add NET_ADMIN" && exit 16
|
error "and that the NET_ADMIN capability has been added to the container config: --cap-add NET_ADMIN" && exit 16
|
||||||
fi
|
fi
|
||||||
|
|
||||||
while ! ip link set "${VM_NET_TAP}" up; do
|
while ! ip link set "$VM_NET_TAP" up; do
|
||||||
info "Waiting for address to become available..."
|
info "Waiting for address to become available..."
|
||||||
sleep 2
|
sleep 2
|
||||||
done
|
done
|
||||||
|
|
||||||
TAP_NR=$(</sys/class/net/"${VM_NET_TAP}"/ifindex)
|
local TAP_NR TAP_PATH MAJOR MINOR
|
||||||
|
TAP_NR=$(</sys/class/net/"$VM_NET_TAP"/ifindex)
|
||||||
TAP_PATH="/dev/tap${TAP_NR}"
|
TAP_PATH="/dev/tap${TAP_NR}"
|
||||||
|
|
||||||
# Create dev file (there is no udev in container: need to be done manually)
|
# Create dev file (there is no udev in container: need to be done manually)
|
||||||
IFS=: read -r MAJOR MINOR < <(cat /sys/devices/virtual/net/"${VM_NET_TAP}"/tap*/dev)
|
IFS=: read -r MAJOR MINOR < <(cat /sys/devices/virtual/net/"$VM_NET_TAP"/tap*/dev)
|
||||||
(( MAJOR < 1)) && error "Cannot find: sys/devices/virtual/net/${VM_NET_TAP}" && exit 18
|
(( MAJOR < 1)) && error "Cannot find: sys/devices/virtual/net/$VM_NET_TAP" && exit 18
|
||||||
|
|
||||||
[[ ! -e "${TAP_PATH}" ]] && [[ -e "/dev0/${TAP_PATH##*/}" ]] && ln -s "/dev0/${TAP_PATH##*/}" "${TAP_PATH}"
|
[[ ! -e "$TAP_PATH" ]] && [[ -e "/dev0/${TAP_PATH##*/}" ]] && ln -s "/dev0/${TAP_PATH##*/}" "$TAP_PATH"
|
||||||
|
|
||||||
if [[ ! -e "${TAP_PATH}" ]]; then
|
if [[ ! -e "$TAP_PATH" ]]; then
|
||||||
{ mknod "${TAP_PATH}" c "$MAJOR" "$MINOR" ; rc=$?; } || :
|
{ mknod "$TAP_PATH" c "$MAJOR" "$MINOR" ; rc=$?; } || :
|
||||||
(( rc != 0 )) && error "Cannot mknod: ${TAP_PATH} ($rc)" && exit 20
|
(( rc != 0 )) && error "Cannot mknod: $TAP_PATH ($rc)" && exit 20
|
||||||
fi
|
fi
|
||||||
|
|
||||||
{ exec 30>>"$TAP_PATH"; rc=$?; } 2>/dev/null || :
|
{ exec 30>>"$TAP_PATH"; rc=$?; } 2>/dev/null || :
|
||||||
|
|
||||||
if (( rc != 0 )); then
|
if (( rc != 0 )); then
|
||||||
error "Cannot create TAP interface ($rc). Please add the following docker settings to your "
|
error "Cannot create TAP interface ($rc). Please add the following docker settings to your "
|
||||||
error "container: --device-cgroup-rule='c ${MAJOR}:* rwm' --device=/dev/vhost-net" && exit 21
|
error "container: --device-cgroup-rule='c $MAJOR:* rwm' --device=/dev/vhost-net" && exit 21
|
||||||
fi
|
fi
|
||||||
|
|
||||||
{ exec 40>>/dev/vhost-net; rc=$?; } 2>/dev/null || :
|
{ exec 40>>/dev/vhost-net; rc=$?; } 2>/dev/null || :
|
||||||
@@ -75,42 +75,17 @@ configureDNS () {
|
|||||||
DNSMASQ_OPTS="$DNSMASQ_OPTS --dhcp-range=$VM_NET_IP,$VM_NET_IP --dhcp-host=$VM_NET_MAC,,$VM_NET_IP,$VM_NET_HOST,infinite --dhcp-option=option:netmask,255.255.255.0"
|
DNSMASQ_OPTS="$DNSMASQ_OPTS --dhcp-range=$VM_NET_IP,$VM_NET_IP --dhcp-host=$VM_NET_MAC,,$VM_NET_IP,$VM_NET_HOST,infinite --dhcp-option=option:netmask,255.255.255.0"
|
||||||
|
|
||||||
# Create lease file for faster resolve
|
# Create lease file for faster resolve
|
||||||
echo "0 $VM_NET_MAC $VM_NET_IP $VM_NET_HOST 01:${VM_NET_MAC}" > /var/lib/misc/dnsmasq.leases
|
echo "0 $VM_NET_MAC $VM_NET_IP $VM_NET_HOST 01:$VM_NET_MAC" > /var/lib/misc/dnsmasq.leases
|
||||||
chmod 644 /var/lib/misc/dnsmasq.leases
|
chmod 644 /var/lib/misc/dnsmasq.leases
|
||||||
|
|
||||||
# Build DNS options from container /etc/resolv.conf
|
# Set DNS server and gateway
|
||||||
|
DNSMASQ_OPTS="$DNSMASQ_OPTS --dhcp-option=option:dns-server,${VM_NET_IP%.*}.1 --dhcp-option=option:router,${VM_NET_IP%.*}.1"
|
||||||
if [[ "${DEBUG}" == [Yy1]* ]]; then
|
|
||||||
echo "/etc/resolv.conf:" && echo && cat /etc/resolv.conf && echo
|
|
||||||
fi
|
|
||||||
|
|
||||||
mapfile -t nameservers < <( { grep '^nameserver' /etc/resolv.conf || true; } | sed 's/\t/ /g' | sed 's/nameserver //' | sed 's/ //g')
|
|
||||||
searchdomains=$( { grep '^search' /etc/resolv.conf || true; } | sed 's/\t/ /g' | sed 's/search //' | sed 's/#.*//' | sed 's/\s*$//g' | sed 's/ /,/g')
|
|
||||||
domainname=$(echo "$searchdomains" | awk -F"," '{print $1}')
|
|
||||||
|
|
||||||
for nameserver in "${nameservers[@]}"; do
|
|
||||||
nameserver=$(echo "$nameserver" | sed 's/#.*//' )
|
|
||||||
if ! [[ "$nameserver" =~ .*:.* ]]; then
|
|
||||||
[[ -z "$DNS_SERVERS" ]] && DNS_SERVERS="$nameserver" || DNS_SERVERS="$DNS_SERVERS,$nameserver"
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
|
|
||||||
[[ -z "$DNS_SERVERS" ]] && DNS_SERVERS="1.1.1.1"
|
|
||||||
|
|
||||||
DNSMASQ_OPTS="$DNSMASQ_OPTS --dhcp-option=option:dns-server,$DNS_SERVERS --dhcp-option=option:router,${VM_NET_IP%.*}.1"
|
|
||||||
|
|
||||||
if [ -n "$searchdomains" ] && [ "$searchdomains" != "." ]; then
|
|
||||||
DNSMASQ_OPTS="$DNSMASQ_OPTS --dhcp-option=option:domain-search,$searchdomains --dhcp-option=option:domain-name,$domainname"
|
|
||||||
else
|
|
||||||
[[ -z $(hostname -d) ]] || DNSMASQ_OPTS="$DNSMASQ_OPTS --dhcp-option=option:domain-name,$(hostname -d)"
|
|
||||||
fi
|
|
||||||
|
|
||||||
DNSMASQ_OPTS=$(echo "$DNSMASQ_OPTS" | sed 's/\t/ /g' | tr -s ' ' | sed 's/^ *//')
|
DNSMASQ_OPTS=$(echo "$DNSMASQ_OPTS" | sed 's/\t/ /g' | tr -s ' ' | sed 's/^ *//')
|
||||||
|
|
||||||
[[ "${DEBUG}" == [Yy1]* ]] && set -x
|
[[ "$DEBUG" == [Yy1]* ]] && set -x
|
||||||
$DNSMASQ ${DNSMASQ_OPTS:+ $DNSMASQ_OPTS}
|
$DNSMASQ ${DNSMASQ_OPTS:+ $DNSMASQ_OPTS}
|
||||||
{ set +x; } 2>/dev/null
|
{ set +x; } 2>/dev/null
|
||||||
[[ "${DEBUG}" == [Yy1]* ]] && echo
|
[[ "$DEBUG" == [Yy1]* ]] && echo
|
||||||
|
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
@@ -120,7 +95,7 @@ configureNAT () {
|
|||||||
# Create a bridge with a static IP for the VM guest
|
# Create a bridge with a static IP for the VM guest
|
||||||
|
|
||||||
VM_NET_IP='20.20.20.21'
|
VM_NET_IP='20.20.20.21'
|
||||||
[[ "${DEBUG}" == [Yy1]* ]] && set -x
|
[[ "$DEBUG" == [Yy1]* ]] && set -x
|
||||||
|
|
||||||
{ ip link add dev dockerbridge type bridge ; rc=$?; } || :
|
{ ip link add dev dockerbridge type bridge ; rc=$?; } || :
|
||||||
|
|
||||||
@@ -137,21 +112,19 @@ configureNAT () {
|
|||||||
done
|
done
|
||||||
|
|
||||||
# QEMU Works with taps, set tap to the bridge created
|
# QEMU Works with taps, set tap to the bridge created
|
||||||
ip tuntap add dev "${VM_NET_TAP}" mode tap
|
ip tuntap add dev "$VM_NET_TAP" mode tap
|
||||||
|
|
||||||
while ! ip link set "${VM_NET_TAP}" up promisc on; do
|
while ! ip link set "$VM_NET_TAP" up promisc on; do
|
||||||
info "Waiting for tap to become available..."
|
info "Waiting for tap to become available..."
|
||||||
sleep 2
|
sleep 2
|
||||||
done
|
done
|
||||||
|
|
||||||
ip link set dev "${VM_NET_TAP}" master dockerbridge
|
ip link set dev "$VM_NET_TAP" master dockerbridge
|
||||||
|
|
||||||
# Add internet connection to the VM
|
# Add internet connection to the VM
|
||||||
IP=$(ip address show dev "${VM_NET_DEV}" | grep inet | awk '/inet / { print $2 }' | cut -f1 -d/)
|
iptables -t nat -A POSTROUTING -o "$VM_NET_DEV" -j MASQUERADE
|
||||||
|
iptables -t nat -A PREROUTING -i "$VM_NET_DEV" -d "$IP" -p tcp -j DNAT --to "$VM_NET_IP"
|
||||||
iptables -t nat -A POSTROUTING -o "${VM_NET_DEV}" -j MASQUERADE
|
iptables -t nat -A PREROUTING -i "$VM_NET_DEV" -d "$IP" -p udp -j DNAT --to "$VM_NET_IP"
|
||||||
iptables -t nat -A PREROUTING -i "${VM_NET_DEV}" -d "${IP}" -p tcp -j DNAT --to $VM_NET_IP
|
|
||||||
iptables -t nat -A PREROUTING -i "${VM_NET_DEV}" -d "${IP}" -p udp -j DNAT --to $VM_NET_IP
|
|
||||||
|
|
||||||
if (( KERNEL > 4 )); then
|
if (( KERNEL > 4 )); then
|
||||||
# Hack for guest VMs complaining about "bad udp checksums in 5 packets"
|
# Hack for guest VMs complaining about "bad udp checksums in 5 packets"
|
||||||
@@ -159,7 +132,7 @@ configureNAT () {
|
|||||||
fi
|
fi
|
||||||
|
|
||||||
{ set +x; } 2>/dev/null
|
{ set +x; } 2>/dev/null
|
||||||
[[ "${DEBUG}" == [Yy1]* ]] && echo
|
[[ "$DEBUG" == [Yy1]* ]] && echo
|
||||||
|
|
||||||
# Check port forwarding flag
|
# Check port forwarding flag
|
||||||
if [[ $(< /proc/sys/net/ipv4/ip_forward) -eq 0 ]]; then
|
if [[ $(< /proc/sys/net/ipv4/ip_forward) -eq 0 ]]; then
|
||||||
@@ -169,7 +142,7 @@ configureNAT () {
|
|||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
NET_OPTS="-netdev tap,ifname=${VM_NET_TAP},script=no,downscript=no,id=hostnet0"
|
NET_OPTS="-netdev tap,ifname=$VM_NET_TAP,script=no,downscript=no,id=hostnet0"
|
||||||
|
|
||||||
{ exec 40>>/dev/vhost-net; rc=$?; } 2>/dev/null || :
|
{ exec 40>>/dev/vhost-net; rc=$?; } 2>/dev/null || :
|
||||||
(( rc == 0 )) && NET_OPTS="$NET_OPTS,vhost=on,vhostfd=40"
|
(( rc == 0 )) && NET_OPTS="$NET_OPTS,vhost=on,vhostfd=40"
|
||||||
@@ -181,15 +154,19 @@ configureNAT () {
|
|||||||
|
|
||||||
closeNetwork () {
|
closeNetwork () {
|
||||||
|
|
||||||
if [[ "${DHCP}" == [Yy1]* ]]; then
|
if [[ "$DHCP" == [Yy1]* ]]; then
|
||||||
|
|
||||||
ip link set "${VM_NET_TAP}" down || true
|
{ pkill -f server.sh || true; } 2>/dev/null
|
||||||
ip link delete "${VM_NET_TAP}" || true
|
|
||||||
|
ip link set "$VM_NET_TAP" down || true
|
||||||
|
ip link delete "$VM_NET_TAP" || true
|
||||||
|
|
||||||
else
|
else
|
||||||
|
|
||||||
ip link set "${VM_NET_TAP}" down promisc off || true
|
{ pkill -f dnsmasq || true; } 2>/dev/null
|
||||||
ip link delete "${VM_NET_TAP}" || true
|
|
||||||
|
ip link set "$VM_NET_TAP" down promisc off || true
|
||||||
|
ip link delete "$VM_NET_TAP" || true
|
||||||
|
|
||||||
ip link set dockerbridge down || true
|
ip link set dockerbridge down || true
|
||||||
ip link delete dockerbridge || true
|
ip link delete dockerbridge || true
|
||||||
@@ -206,16 +183,20 @@ closeNetwork () {
|
|||||||
# Create the necessary file structure for /dev/net/tun
|
# Create the necessary file structure for /dev/net/tun
|
||||||
if [ ! -c /dev/net/tun ]; then
|
if [ ! -c /dev/net/tun ]; then
|
||||||
[ ! -d /dev/net ] && mkdir -m 755 /dev/net
|
[ ! -d /dev/net ] && mkdir -m 755 /dev/net
|
||||||
mknod /dev/net/tun c 10 200
|
if mknod /dev/net/tun c 10 200; then
|
||||||
chmod 666 /dev/net/tun
|
chmod 666 /dev/net/tun
|
||||||
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
[ ! -c /dev/net/tun ] && error "TUN network interface not available..." && exit 85
|
if [ ! -c /dev/net/tun ]; then
|
||||||
|
error "Please add the following docker settings to your container: --device=/dev/net/tun" && exit 25
|
||||||
|
fi
|
||||||
|
|
||||||
# Create the necessary file structure for /dev/vhost-net
|
# Create the necessary file structure for /dev/vhost-net
|
||||||
if [ ! -c /dev/vhost-net ]; then
|
if [ ! -c /dev/vhost-net ]; then
|
||||||
mknod /dev/vhost-net c 10 238
|
if mknod /dev/vhost-net c 10 238; then
|
||||||
chmod 660 /dev/vhost-net
|
chmod 660 /dev/vhost-net
|
||||||
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
update-alternatives --set iptables /usr/sbin/iptables-legacy > /dev/null
|
update-alternatives --set iptables /usr/sbin/iptables-legacy > /dev/null
|
||||||
@@ -223,21 +204,19 @@ update-alternatives --set ip6tables /usr/sbin/ip6tables-legacy > /dev/null
|
|||||||
|
|
||||||
VM_NET_MAC="${VM_NET_MAC//-/:}"
|
VM_NET_MAC="${VM_NET_MAC//-/:}"
|
||||||
GATEWAY=$(ip r | grep default | awk '{print $3}')
|
GATEWAY=$(ip r | grep default | awk '{print $3}')
|
||||||
|
IP=$(ip address show dev "$VM_NET_DEV" | grep inet | awk '/inet / { print $2 }' | cut -f1 -d/)
|
||||||
|
|
||||||
if [[ "${DEBUG}" == [Yy1]* ]]; then
|
if [[ "$DEBUG" == [Yy1]* ]]; then
|
||||||
|
info "Container IP is $IP with gateway $GATEWAY" && echo
|
||||||
IP=$(ip address show dev "${VM_NET_DEV}" | grep inet | awk '/inet / { print $2 }' | cut -f1 -d/)
|
|
||||||
info "Container IP is ${IP} with gateway ${GATEWAY}" && echo
|
|
||||||
|
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if [[ "${DHCP}" == [Yy1]* ]]; then
|
if [[ "$DHCP" == [Yy1]* ]]; then
|
||||||
|
|
||||||
if [[ "$GATEWAY" == "172."* ]]; then
|
if [[ "$GATEWAY" == "172."* ]]; then
|
||||||
if [[ "${DEBUG}" == [Yy1]* ]]; then
|
if [[ "$DEBUG" == [Yy1]* ]]; then
|
||||||
info "Warning: Are you sure the container is on a macvlan network?"
|
info "Warning: Are you sure the container is on a macvlan network?"
|
||||||
else
|
else
|
||||||
error "You can only enable DHCP while the container is on a macvlan network!" && exit 86
|
error "You can only enable DHCP while the container is on a macvlan network!" && exit 26
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
@@ -254,6 +233,6 @@ else
|
|||||||
|
|
||||||
fi
|
fi
|
||||||
|
|
||||||
NET_OPTS="${NET_OPTS} -device virtio-net-pci,romfile=,netdev=hostnet0,mac=${VM_NET_MAC},id=net0"
|
NET_OPTS="$NET_OPTS -device virtio-net-pci,romfile=,netdev=hostnet0,mac=$VM_NET_MAC,id=net0"
|
||||||
|
|
||||||
return 0
|
return 0
|
||||||
87
src/power.sh
Normal file
87
src/power.sh
Normal file
@@ -0,0 +1,87 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -Eeuo pipefail
|
||||||
|
|
||||||
|
# Configure QEMU for graceful shutdown
|
||||||
|
|
||||||
|
QEMU_PORT=7100
|
||||||
|
QEMU_TIMEOUT=55
|
||||||
|
QEMU_PID="/run/qemu.pid"
|
||||||
|
QEMU_COUNT="/run/qemu.count"
|
||||||
|
|
||||||
|
rm -f "$QEMU_PID"
|
||||||
|
rm -f "$QEMU_COUNT"
|
||||||
|
|
||||||
|
_trap(){
|
||||||
|
func="$1" ; shift
|
||||||
|
for sig ; do
|
||||||
|
trap "$func $sig" "$sig"
|
||||||
|
done
|
||||||
|
}
|
||||||
|
|
||||||
|
_graceful_shutdown() {
|
||||||
|
|
||||||
|
set +e
|
||||||
|
local cnt response
|
||||||
|
|
||||||
|
[ ! -f "$QEMU_PID" ] && exit 130
|
||||||
|
[ -f "$QEMU_COUNT" ] && return
|
||||||
|
|
||||||
|
echo 0 > "$QEMU_COUNT"
|
||||||
|
echo && info "Received $1 signal, sending shutdown command..."
|
||||||
|
|
||||||
|
# Don't send the powerdown signal because vDSM ignores ACPI signals
|
||||||
|
# echo 'system_powerdown' | nc -q 1 -w 1 localhost "${QEMU_PORT}" > /dev/null
|
||||||
|
|
||||||
|
# Send shutdown command to guest agent via serial port
|
||||||
|
url="http://127.0.0.1:2210/read?command=6&timeout=50"
|
||||||
|
response=$(curl -sk -m 52 -S "$url" 2>&1)
|
||||||
|
|
||||||
|
if [[ "$response" =~ "\"success\"" ]]; then
|
||||||
|
|
||||||
|
echo && info "Virtual DSM is now ready to shutdown..."
|
||||||
|
|
||||||
|
else
|
||||||
|
|
||||||
|
response="${response#*message\"\: \"}"
|
||||||
|
echo && error "Failed to send shutdown command: ${response%%\"*}"
|
||||||
|
|
||||||
|
kill -15 "$(cat "$QEMU_PID")"
|
||||||
|
pkill -f qemu-system-x86_64 || true
|
||||||
|
|
||||||
|
fi
|
||||||
|
|
||||||
|
while [ "$(cat $QEMU_COUNT)" -lt "$QEMU_TIMEOUT" ]; do
|
||||||
|
|
||||||
|
# Try to connect to qemu
|
||||||
|
if ! echo 'info version'| nc -q 1 -w 1 localhost "$QEMU_PORT" >/dev/null 2>&1 ; then
|
||||||
|
break
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Increase the counter
|
||||||
|
cnt=$(($(cat $QEMU_COUNT)+1))
|
||||||
|
echo $cnt > "$QEMU_COUNT"
|
||||||
|
|
||||||
|
[[ "$DEBUG" == [Yy1]* ]] && info "Shutting down, waiting... ($cnt/$QEMU_TIMEOUT)"
|
||||||
|
|
||||||
|
done
|
||||||
|
|
||||||
|
if [ "$(cat $QEMU_COUNT)" -ge "$QEMU_TIMEOUT" ]; then
|
||||||
|
echo && error "Shutdown timeout reached, forcefully quitting.."
|
||||||
|
else
|
||||||
|
echo && echo "❯ Quitting..."
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo 'quit' | nc -q 1 -w 1 localhost "$QEMU_PORT" >/dev/null 2>&1 || true
|
||||||
|
|
||||||
|
{ pkill -f print.sh || true; } 2>/dev/null
|
||||||
|
{ pkill -f host.bin || true; } 2>/dev/null
|
||||||
|
|
||||||
|
closeNetwork
|
||||||
|
sleep 1
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
_trap _graceful_shutdown SIGTERM SIGHUP SIGINT SIGABRT SIGQUIT
|
||||||
|
|
||||||
|
MON_OPTS="-monitor telnet:localhost:$QEMU_PORT,server,nowait,nodelay"
|
||||||
86
src/print.sh
Normal file
86
src/print.sh
Normal file
@@ -0,0 +1,86 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -Eeuo pipefail
|
||||||
|
|
||||||
|
: ${DHCP:='N'}
|
||||||
|
: ${VM_NET_DEV:='eth0'}
|
||||||
|
|
||||||
|
info () { printf "%b%s%b" "\E[1;34m❯ \E[1;36m" "$1" "\E[0m\n" >&2; }
|
||||||
|
error () { printf "%b%s%b" "\E[1;31m❯ " "ERROR: $1" "\E[0m\n" >&2; }
|
||||||
|
|
||||||
|
file="/run/dsm.url"
|
||||||
|
shutdown="/run/qemu.count"
|
||||||
|
url="http://127.0.0.1:2210/read?command=10"
|
||||||
|
|
||||||
|
resp_err="Guest returned an invalid response:"
|
||||||
|
jq_err="Failed to parse response from guest: jq error"
|
||||||
|
|
||||||
|
while [ ! -f "$file" ]
|
||||||
|
do
|
||||||
|
|
||||||
|
# Check if not shutting down
|
||||||
|
[ -f "$shutdown" ] && exit 1
|
||||||
|
|
||||||
|
sleep 3
|
||||||
|
|
||||||
|
[ -f "$shutdown" ] && exit 1
|
||||||
|
[ -f "$file" ] && break
|
||||||
|
|
||||||
|
# Retrieve network info from guest VM
|
||||||
|
{ json=$(curl -m 20 -sk "$url"); rc=$?; } || :
|
||||||
|
|
||||||
|
[ -f "$shutdown" ] && exit 1
|
||||||
|
(( rc != 0 )) && error "Failed to connect to guest: curl error $rc" && continue
|
||||||
|
|
||||||
|
{ result=$(echo "$json" | jq -r '.status'); rc=$?; } || :
|
||||||
|
(( rc != 0 )) && error "$jq_err $rc ( $json )" && continue
|
||||||
|
[[ "$result" == "null" ]] && error "$resp_err $json" && continue
|
||||||
|
|
||||||
|
if [[ "$result" != "success" ]] ; then
|
||||||
|
{ msg=$(echo "$json" | jq -r '.message'); rc=$?; } || :
|
||||||
|
error "Guest replied $result: $msg" && continue
|
||||||
|
fi
|
||||||
|
|
||||||
|
{ port=$(echo "$json" | jq -r '.data.data.dsm_setting.data.http_port'); rc=$?; } || :
|
||||||
|
(( rc != 0 )) && error "$jq_err $rc ( $json )" && continue
|
||||||
|
[[ "$port" == "null" ]] && error "$resp_err $json" && continue
|
||||||
|
[ -z "$port" ] && continue
|
||||||
|
|
||||||
|
{ ip=$(echo "$json" | jq -r '.data.data.ip.data[] | select((.name=="eth0") and has("ip")).ip'); rc=$?; } || :
|
||||||
|
(( rc != 0 )) && error "$jq_err $rc ( $json )" && continue
|
||||||
|
[[ "$ip" == "null" ]] && error "$resp_err $json" && continue
|
||||||
|
|
||||||
|
if [ -z "$ip" ]; then
|
||||||
|
[[ "$DHCP" == [Yy1]* ]] && continue
|
||||||
|
ip="20.20.20.21"
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "$ip:$port" > $file
|
||||||
|
|
||||||
|
done
|
||||||
|
|
||||||
|
[ -f "$shutdown" ] && exit 1
|
||||||
|
|
||||||
|
location=$(cat "$file")
|
||||||
|
|
||||||
|
if [[ "$location" != "20.20"* ]]; then
|
||||||
|
|
||||||
|
msg="http://$location"
|
||||||
|
|
||||||
|
else
|
||||||
|
|
||||||
|
ip=$(ip address show dev "$VM_NET_DEV" | grep inet | awk '/inet / { print $2 }' | cut -f1 -d/)
|
||||||
|
port="${location##*:}"
|
||||||
|
|
||||||
|
if [[ "$ip" == "172."* ]]; then
|
||||||
|
msg="port $port"
|
||||||
|
else
|
||||||
|
msg="http://$ip:$port"
|
||||||
|
fi
|
||||||
|
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "" >&2
|
||||||
|
info "-----------------------------------------------------------"
|
||||||
|
info " You can now login to DSM at $msg"
|
||||||
|
info "-----------------------------------------------------------"
|
||||||
|
echo "" >&2
|
||||||
115
src/reset.sh
Normal file
115
src/reset.sh
Normal file
@@ -0,0 +1,115 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -Eeuo pipefail
|
||||||
|
|
||||||
|
info () { printf "%b%s%b" "\E[1;34m❯ \E[1;36m" "$1" "\E[0m\n"; }
|
||||||
|
error () { printf "%b%s%b" "\E[1;31m❯ " "ERROR: $1" "\E[0m\n" >&2; }
|
||||||
|
|
||||||
|
trap 'error "Status $? while: $BASH_COMMAND (line $LINENO/$BASH_LINENO)"' ERR
|
||||||
|
|
||||||
|
[ ! -f "/run/entry.sh" ] && error "Script must run inside Docker container!" && exit 11
|
||||||
|
[ "$(id -u)" -ne "0" ] && error "Script must be executed with root privileges." && exit 12
|
||||||
|
|
||||||
|
# Docker environment variables
|
||||||
|
|
||||||
|
: ${TZ:=''} # System local timezone
|
||||||
|
: ${GPU:='N'} # Disable GPU passthrough
|
||||||
|
: ${KVM:='Y'} # Enable KVM acceleration
|
||||||
|
: ${DEBUG:='N'} # Disable debugging mode
|
||||||
|
: ${COUNTRY:=''} # Country code for mirror
|
||||||
|
: ${CONSOLE:='N'} # Disable console mode
|
||||||
|
: ${ALLOCATE:='Y'} # Preallocate diskspace
|
||||||
|
: ${ARGUMENTS:=''} # Extra QEMU parameters
|
||||||
|
: ${CPU_CORES:='1'} # Amount of CPU cores
|
||||||
|
: ${RAM_SIZE:='1G'} # Maximum RAM amount
|
||||||
|
: ${DISK_SIZE:='16G'} # Initial data disk size
|
||||||
|
|
||||||
|
# Helper variables
|
||||||
|
|
||||||
|
KERNEL=$(uname -r | cut -b 1)
|
||||||
|
MINOR=$(uname -r | cut -d '.' -f2)
|
||||||
|
ARCH=$(dpkg --print-architecture)
|
||||||
|
VERS=$(qemu-system-x86_64 --version | head -n 1 | cut -d '(' -f 1)
|
||||||
|
|
||||||
|
# Check folder
|
||||||
|
|
||||||
|
STORAGE="/storage"
|
||||||
|
[ ! -d "$STORAGE" ] && error "Storage folder ($STORAGE) not found!" && exit 13
|
||||||
|
|
||||||
|
# Cleanup files
|
||||||
|
|
||||||
|
rm -f /run/dsm.url
|
||||||
|
rm -f /run/qemu.pid
|
||||||
|
rm -f /run/qemu.count
|
||||||
|
|
||||||
|
# Cleanup dirs
|
||||||
|
|
||||||
|
rm -rf /tmp/dsm
|
||||||
|
rm -f /tmp/server.*
|
||||||
|
rm -rf "$STORAGE/tmp"
|
||||||
|
|
||||||
|
# Helper functions
|
||||||
|
|
||||||
|
getCountry () {
|
||||||
|
|
||||||
|
local url=$1
|
||||||
|
local query=$2
|
||||||
|
local rc json result
|
||||||
|
|
||||||
|
{ json=$(curl -m 5 -H "Accept: application/json" -sfk "$url"); rc=$?; } || :
|
||||||
|
(( rc != 0 )) && return 0
|
||||||
|
|
||||||
|
{ result=$(echo "$json" | jq -r "$query" 2> /dev/null); rc=$?; } || :
|
||||||
|
(( rc != 0 )) && return 0
|
||||||
|
|
||||||
|
[[ ${#result} -ne 2 ]] && return 0
|
||||||
|
[[ "${result^^}" == "XX" ]] && return 0
|
||||||
|
|
||||||
|
COUNTRY="${result^^}"
|
||||||
|
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
setCountry () {
|
||||||
|
|
||||||
|
[[ "${TZ,,}" == "asia/harbin" ]] && COUNTRY="CN"
|
||||||
|
[[ "${TZ,,}" == "asia/beijing" ]] && COUNTRY="CN"
|
||||||
|
[[ "${TZ,,}" == "asia/urumqi" ]] && COUNTRY="CN"
|
||||||
|
[[ "${TZ,,}" == "asia/kashgar" ]] && COUNTRY="CN"
|
||||||
|
[[ "${TZ,,}" == "asia/shanghai" ]] && COUNTRY="CN"
|
||||||
|
[[ "${TZ,,}" == "asia/chongqing" ]] && COUNTRY="CN"
|
||||||
|
|
||||||
|
[ -z "$COUNTRY" ] && getCountry "https://api.ipapi.is" ".location.country_code"
|
||||||
|
[ -z "$COUNTRY" ] && getCountry "https://ifconfig.co/json" ".country_iso"
|
||||||
|
[ -z "$COUNTRY" ] && getCountry "https://ipinfo.io/json" ".country"
|
||||||
|
[ -z "$COUNTRY" ] && getCountry "https://api.myip.com" ".cc"
|
||||||
|
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
addPackage () {
|
||||||
|
|
||||||
|
local pkg=$1
|
||||||
|
local desc=$2
|
||||||
|
|
||||||
|
if apt-mark showinstall | grep -qx "$pkg"; then
|
||||||
|
return 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
info "Installing $desc..."
|
||||||
|
|
||||||
|
export DEBCONF_NOWARNINGS="yes"
|
||||||
|
export DEBIAN_FRONTEND="noninteractive"
|
||||||
|
|
||||||
|
[ -z "$COUNTRY" ] && setCountry
|
||||||
|
|
||||||
|
if [[ "${COUNTRY^^}" == "CN" ]]; then
|
||||||
|
sed -i 's/deb.debian.org/mirrors.ustc.edu.cn/g' /etc/apt/sources.list.d/debian.sources
|
||||||
|
fi
|
||||||
|
|
||||||
|
apt-get -qq update
|
||||||
|
apt-get -qq --no-install-recommends -y install "$pkg" > /dev/null
|
||||||
|
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0
|
||||||
57
src/serial.sh
Normal file
57
src/serial.sh
Normal file
@@ -0,0 +1,57 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -Eeuo pipefail
|
||||||
|
|
||||||
|
# Docker environment variables
|
||||||
|
|
||||||
|
: ${HOST_MAC:=''}
|
||||||
|
: ${HOST_DEBUG:=''}
|
||||||
|
: ${HOST_SERIAL:=''}
|
||||||
|
: ${HOST_MODEL:=''}
|
||||||
|
: ${GUEST_SERIAL:=''}
|
||||||
|
|
||||||
|
HOST_ARGS=()
|
||||||
|
HOST_ARGS+=("-cpu=$CPU_CORES")
|
||||||
|
HOST_ARGS+=("-cpu_arch=$HOST_CPU")
|
||||||
|
|
||||||
|
[ -n "$HOST_MAC" ] && HOST_ARGS+=("-mac=$HOST_MAC")
|
||||||
|
[ -n "$HOST_MODEL" ] && HOST_ARGS+=("-model=$HOST_MODEL")
|
||||||
|
[ -n "$HOST_SERIAL" ] && HOST_ARGS+=("-hostsn=$HOST_SERIAL")
|
||||||
|
[ -n "$GUEST_SERIAL" ] && HOST_ARGS+=("-guestsn=$GUEST_SERIAL")
|
||||||
|
|
||||||
|
if [[ "$HOST_DEBUG" == [Yy1]* ]]; then
|
||||||
|
set -x
|
||||||
|
./host.bin "${HOST_ARGS[@]}" &
|
||||||
|
{ set +x; } 2>/dev/null
|
||||||
|
echo
|
||||||
|
else
|
||||||
|
./host.bin "${HOST_ARGS[@]}" >/dev/null &
|
||||||
|
fi
|
||||||
|
|
||||||
|
cnt=0
|
||||||
|
sleep 0.2
|
||||||
|
|
||||||
|
while ! nc -z -w2 127.0.0.1 2210 > /dev/null 2>&1; do
|
||||||
|
sleep 0.1
|
||||||
|
cnt=$((cnt + 1))
|
||||||
|
(( cnt > 50 )) && error "Failed to connect to qemu-host.." && exit 58
|
||||||
|
done
|
||||||
|
|
||||||
|
cnt=0
|
||||||
|
|
||||||
|
while ! nc -z -w2 127.0.0.1 12345 > /dev/null 2>&1; do
|
||||||
|
sleep 0.1
|
||||||
|
cnt=$((cnt + 1))
|
||||||
|
(( cnt > 50 )) && error "Failed to connect to qemu-host.." && exit 59
|
||||||
|
done
|
||||||
|
|
||||||
|
# Configure serial ports
|
||||||
|
|
||||||
|
SERIAL_OPTS="\
|
||||||
|
-serial mon:stdio \
|
||||||
|
-device virtio-serial-pci,id=virtio-serial0,bus=pcie.0,addr=0x3 \
|
||||||
|
-chardev pty,id=charserial0 \
|
||||||
|
-device isa-serial,chardev=charserial0,id=serial0 \
|
||||||
|
-chardev socket,id=charchannel0,host=127.0.0.1,port=12345,reconnect=10 \
|
||||||
|
-device virtserialport,bus=virtio-serial0.0,nr=1,chardev=charchannel0,id=channel0,name=vchannel"
|
||||||
|
|
||||||
|
return 0
|
||||||
@@ -14,9 +14,9 @@ trap 'stop' EXIT SIGINT SIGTERM SIGHUP
|
|||||||
html()
|
html()
|
||||||
{
|
{
|
||||||
local h="<!DOCTYPE html><HTML><HEAD><TITLE>VirtualDSM</TITLE>"
|
local h="<!DOCTYPE html><HTML><HEAD><TITLE>VirtualDSM</TITLE>"
|
||||||
h="${h} <STYLE>body { color: white; background-color: #125bdb; font-family: Verdana,"
|
h="$h<STYLE>body { color: white; background-color: #125bdb; font-family: Verdana,"
|
||||||
h="${h} Arial,sans-serif; } a, a:hover, a:active, a:visited { color: white; }</STYLE></HEAD>"
|
h="$h Arial,sans-serif; } a, a:hover, a:active, a:visited { color: white; }</STYLE></HEAD>"
|
||||||
h="${h}<BODY><BR><BR><H1><CENTER>$1</CENTER></H1></BODY></HTML>"
|
h="$h<BODY><BR><BR><H1><CENTER>$1</CENTER></H1></BODY></HTML>"
|
||||||
|
|
||||||
echo "$h"
|
echo "$h"
|
||||||
}
|
}
|
||||||
@@ -33,8 +33,8 @@ if [[ "$2" != "/"* ]]; then
|
|||||||
HTML=$(html "$BODY")
|
HTML=$(html "$BODY")
|
||||||
printf '%b' "HTTP/1.1 200 OK\nContent-Length: ${#HTML}\nConnection: close\n\n$HTML" > "$TMP_FILE"
|
printf '%b' "HTTP/1.1 200 OK\nContent-Length: ${#HTML}\nConnection: close\n\n$HTML" > "$TMP_FILE"
|
||||||
|
|
||||||
socat TCP4-LISTEN:80,reuseaddr,fork,crlf SYSTEM:"cat ${TMP_FILE}" 2> /dev/null &
|
socat TCP4-LISTEN:80,reuseaddr,fork,crlf SYSTEM:"cat $TMP_FILE" 2> /dev/null &
|
||||||
socat TCP4-LISTEN:"${1:-5000}",reuseaddr,fork,crlf SYSTEM:"cat ${TMP_FILE}" 2> /dev/null & wait $!
|
socat TCP4-LISTEN:"${1:-5000}",reuseaddr,fork,crlf SYSTEM:"cat $TMP_FILE" 2> /dev/null & wait $!
|
||||||
|
|
||||||
exit
|
exit
|
||||||
|
|
||||||
@@ -46,15 +46,15 @@ if [[ "$2" != "/run/ip.sh" ]]; then
|
|||||||
|
|
||||||
else
|
else
|
||||||
|
|
||||||
BODY="The location of DSM is <a href='http://\${LOCATION}'>http://\${LOCATION}</a><script>"
|
BODY="The location of DSM is <a href='http://\$LOCATION'>http://\$LOCATION</a><script>"
|
||||||
BODY="${BODY}setTimeout(function(){ window.location.assign('http://\${LOCATION}'); }, 3000);</script>"
|
BODY="$BODY setTimeout(function(){ window.location.assign('http://\$LOCATION'); }, 3000);</script>"
|
||||||
WAIT="Please wait while discovering IP...<script>setTimeout(() => { document.location.reload(); }, 4999);</script>"
|
WAIT="Please wait while discovering IP...<script>setTimeout(() => { document.location.reload(); }, 4999);</script>"
|
||||||
|
|
||||||
HTML=$(html "xxx")
|
HTML=$(html "xxx")
|
||||||
|
|
||||||
{ echo "#!/bin/bash"
|
{ echo "#!/bin/bash"
|
||||||
echo "[ -f \"/run/dsm.url\" ] && LOCATION=\$(cat \"/run/dsm.url\")"
|
echo "[ -f \"/run/dsm.url\" ] && LOCATION=\$(cat \"/run/dsm.url\")"
|
||||||
echo "HTML=\"$HTML\"; [ -z \"\${LOCATION}\" ] && BODY=\"$WAIT\" || BODY=\"$BODY\"; HTML=\${HTML/xxx/\$BODY}"
|
echo "HTML=\"$HTML\"; [ -z \"\$LOCATION\" ] && BODY=\"$WAIT\" || BODY=\"$BODY\"; HTML=\${HTML/xxx/\$BODY}"
|
||||||
echo "printf '%b' \"HTTP/1.1 200 OK\\nContent-Length: \${#HTML}\\nConnection: close\\n\\n\$HTML\""
|
echo "printf '%b' \"HTTP/1.1 200 OK\\nContent-Length: \${#HTML}\\nConnection: close\\n\\n\$HTML\""
|
||||||
} > "$TMP_FILE"
|
} > "$TMP_FILE"
|
||||||
|
|
||||||
Reference in New Issue
Block a user