mirror of
https://github.com/vdsm/virtual-dsm.git
synced 2025-11-07 02:23:42 +08:00
Compare commits
294 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
fcd7b8a825 | ||
|
|
575da1f574 | ||
|
|
3a507f5bf6 | ||
|
|
a3d6e3740c | ||
|
|
53e0330e21 | ||
|
|
f935c1e28a | ||
|
|
6a0c708224 | ||
|
|
a407d2d94d | ||
|
|
53605bd6e8 | ||
|
|
944faaa927 | ||
|
|
fb7cfc09de | ||
|
|
b9ae73e322 | ||
|
|
c28dbb6b9b | ||
|
|
dcc26b67a6 | ||
|
|
2627d320f3 | ||
|
|
fd4bc28835 | ||
|
|
63e4d588a2 | ||
|
|
feb1680909 | ||
|
|
9d21489b8e | ||
|
|
e70ed1900f | ||
|
|
d65b5a089a | ||
|
|
84643647cc | ||
|
|
57f487db6d | ||
|
|
0862cad2ce | ||
|
|
5ae4f59315 | ||
|
|
3fc3005ba5 | ||
|
|
20f48edd00 | ||
|
|
b854aad830 | ||
|
|
6cbe03f656 | ||
|
|
63cac9a75e | ||
|
|
08616f1057 | ||
|
|
e6193b1020 | ||
|
|
f28b9903f3 | ||
|
|
7bf2d119ea | ||
|
|
527bded1b2 | ||
|
|
1208c53ebb | ||
|
|
973efa2d27 | ||
|
|
d09588b915 | ||
|
|
19aa313753 | ||
|
|
9db12cd25f | ||
|
|
69e785e6ee | ||
|
|
159fce6839 | ||
|
|
08e4084458 | ||
|
|
06f210846c | ||
|
|
74629e4b55 | ||
|
|
6e8af6e52f | ||
|
|
38611a7af2 | ||
|
|
f089acc01a | ||
|
|
5a7ecb48d6 | ||
|
|
5b3880aa5e | ||
|
|
4653aafbee | ||
|
|
281f2992ff | ||
|
|
4bdcf8bfe1 | ||
|
|
62acaa95bf | ||
|
|
369bff339d | ||
|
|
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 | ||
|
|
0cd1ddf0a1 | ||
|
|
98245a1efe | ||
|
|
f425c869c6 | ||
|
|
f5b8c2a2ef | ||
|
|
f44584261e | ||
|
|
4134d9e3d3 | ||
|
|
db47f561d3 | ||
|
|
895bc04a57 | ||
|
|
d9e882fce4 | ||
|
|
478b6af755 | ||
|
|
fecd4052fc | ||
|
|
0b8306d827 | ||
|
|
c9d0688424 | ||
|
|
413a089e02 | ||
|
|
01e23f22fb | ||
|
|
308a764bb0 | ||
|
|
5ad5f8a8ef | ||
|
|
3c342a05aa | ||
|
|
d793921bcf | ||
|
|
5365a9ed4e | ||
|
|
7a55c650d0 | ||
|
|
ede42b3647 | ||
|
|
8e41b4e567 | ||
|
|
899687d3f2 | ||
|
|
2d97bc1cef | ||
|
|
538d7f0195 | ||
|
|
dcf95a8591 | ||
|
|
7f7272b7c8 | ||
|
|
c30248f93e | ||
|
|
cf90c9da1f | ||
|
|
461b5598a9 | ||
|
|
47ed2e8bac | ||
|
|
1c78e3c8b1 | ||
|
|
9f17dfa949 | ||
|
|
3de29b6c00 | ||
|
|
037d52957a | ||
|
|
c7ccc912b5 | ||
|
|
72cbf87986 | ||
|
|
317024d327 | ||
|
|
121c487383 | ||
|
|
fd92b60218 | ||
|
|
771429c5d3 | ||
|
|
674b6e5bda | ||
|
|
7c65e2740f | ||
|
|
d462c6d7a2 | ||
|
|
d5637e8da0 | ||
|
|
1b75bd3c44 | ||
|
|
2551413040 | ||
|
|
44d3e23545 | ||
|
|
d340361320 | ||
|
|
f8f8c16200 | ||
|
|
47fd7931eb | ||
|
|
7ac9c242da | ||
|
|
78d330055f | ||
|
|
06cf0a4edc | ||
|
|
e26c208cc7 | ||
|
|
c2f4823d8b | ||
|
|
ca01bf8cb9 | ||
|
|
a5b2b655dd | ||
|
|
0f2f3b2ea8 | ||
|
|
0705c1c21b | ||
|
|
e4de05ce88 | ||
|
|
a24b62ae67 | ||
|
|
b5a9361b68 | ||
|
|
0e35e4a6d9 | ||
|
|
208a6e6636 | ||
|
|
3fec9bf5ef | ||
|
|
8d8efeb341 | ||
|
|
e9f27899a6 | ||
|
|
0c1c422758 | ||
|
|
f56c523627 | ||
|
|
66879b5a6a | ||
|
|
b4eef6161b | ||
|
|
cf38b1f237 | ||
|
|
2c09811365 | ||
|
|
761f5babfc | ||
|
|
26be942a74 | ||
|
|
9c6aeca709 | ||
|
|
81b0bba667 | ||
|
|
e2c00a2e44 | ||
|
|
7bb33f26cd | ||
|
|
63b2d703a4 | ||
|
|
a97dfbfdf2 | ||
|
|
b255cb03e8 | ||
|
|
7a79ff1d2d | ||
|
|
06fda133ed | ||
|
|
acedd1cdcf | ||
|
|
96083fddb3 | ||
|
|
5d7604a205 | ||
|
|
24d44924fe | ||
|
|
db840db76f | ||
|
|
38148641c0 | ||
|
|
5941eae237 | ||
|
|
5fa117c4d8 | ||
|
|
28fc911f00 | ||
|
|
c8eb659c3c | ||
|
|
16ce7c07cf | ||
|
|
f745e5e09d |
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"
|
|
||||||
]
|
|
||||||
}
|
}
|
||||||
|
|||||||
107
.github/workflows/build.yml
vendored
107
.github/workflows/build.yml
vendored
@@ -7,12 +7,15 @@ on:
|
|||||||
- master
|
- master
|
||||||
paths-ignore:
|
paths-ignore:
|
||||||
- '**/*.md'
|
- '**/*.md'
|
||||||
- '**/*.yml'
|
- '**/*.yml'
|
||||||
- '.gitignore'
|
- '.gitignore'
|
||||||
- '.dockerignore'
|
- '.dockerignore'
|
||||||
- '.github/**'
|
- '.github/**'
|
||||||
- '.github/workflows/**'
|
- '.github/workflows/**'
|
||||||
- 'Dockerfile'
|
|
||||||
|
concurrency:
|
||||||
|
group: build
|
||||||
|
cancel-in-progress: false
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
shellcheck:
|
shellcheck:
|
||||||
@@ -23,35 +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+=("ghcr.io/${{ github.repository }}:latest")
|
type=raw,value=${{ vars.MAJOR }}.${{ vars.MINOR }}
|
||||||
TAGS+=("ghcr.io/${{ github.repository }}:${VERSION}")
|
labels: |
|
||||||
|
org.opencontainers.image.title=${{ vars.NAME }}
|
||||||
echo "tags=${TAGS[@]}" >> $GITHUB_OUTPUT
|
env:
|
||||||
echo "version=${VERSION}" >> $GITHUB_OUTPUT
|
DOCKER_METADATA_ANNOTATIONS_LEVELS: manifest,index
|
||||||
echo "docker_platforms=${PLATFORMS}" >> $GITHUB_OUTPUT
|
-
|
||||||
echo "build_date=$(date -u +'%Y-%m-%dT%H:%M:%SZ')" >> $GITHUB_OUTPUT
|
name: Set up Docker Buildx
|
||||||
|
uses: docker/setup-buildx-action@v3
|
||||||
-
|
-
|
||||||
name: Login into Docker Hub
|
name: Login into Docker Hub
|
||||||
uses: docker/login-action@v3
|
uses: docker/login-action@v3
|
||||||
@@ -65,59 +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
|
name: Build Docker image
|
||||||
run: |
|
uses: docker/build-push-action@v5
|
||||||
|
|
||||||
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
|
|
||||||
run: |
|
|
||||||
rm -f ${HOME}/.docker/config.json
|
|
||||||
-
|
|
||||||
name: Get previous tag
|
|
||||||
id: previousTag
|
|
||||||
run: |
|
|
||||||
name=$(git --no-pager tag --sort=creatordate --merged ${{ github.ref_name }} | tail -1)
|
|
||||||
echo "previousTag: $name"
|
|
||||||
echo "previousTag=$name" >> $GITHUB_ENV
|
|
||||||
-
|
|
||||||
name: Generate changelog
|
|
||||||
id: changelog
|
|
||||||
uses: requarks/changelog-action@v1
|
|
||||||
with:
|
with:
|
||||||
token: ${{ github.token }}
|
context: .
|
||||||
fromTag: ${{ github.ref_name }}
|
push: true
|
||||||
toTag: ${{ env.previousTag }}
|
provenance: false
|
||||||
writeToFile: false
|
platforms: linux/amd64,linux/arm64,linux/arm
|
||||||
reverseOrder: true
|
tags: ${{ steps.meta.outputs.tags }}
|
||||||
includeInvalidCommits: true
|
labels: ${{ steps.meta.outputs.labels }}
|
||||||
excludeTypes: "docs,build,chore"
|
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 }}"
|
||||||
body: |
|
token: ${{ secrets.REPO_ACCESS_TOKEN }}
|
||||||
${{ steps.changelog.outputs.changes }}
|
|
||||||
|
|
||||||
**Full Changelog**: https://github.com/${{ github.repository }}/compare//${{ env.previousTag }}...v${{ steps.prepare.outputs.version }}
|
|
||||||
-
|
-
|
||||||
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
|
SHELLCHECK_OPTS: -x --source-path=src -e SC2001 -e SC2002 -e SC2223 -e SC2034 -e SC2064 -e SC2317 -e SC2153 -e SC2028
|
||||||
|
|||||||
2
.github/workflows/hub.yml
vendored
2
.github/workflows/hub.yml
vendored
@@ -19,6 +19,6 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
username: ${{ secrets.DOCKERHUB_USERNAME }}
|
username: ${{ secrets.DOCKERHUB_USERNAME }}
|
||||||
password: ${{ secrets.DOCKERHUB_TOKEN }}
|
password: ${{ secrets.DOCKERHUB_TOKEN }}
|
||||||
repository: ${{ github.repository }}
|
repository: ${{ secrets.DOCKERHUB_REPO }}
|
||||||
short-description: ${{ github.event.repository.description }}
|
short-description: ${{ github.event.repository.description }}
|
||||||
readme-filepath: ./readme.md
|
readme-filepath: ./readme.md
|
||||||
|
|||||||
86
Dockerfile
86
Dockerfile
@@ -2,69 +2,59 @@ FROM qemux/qemu-host as builder
|
|||||||
|
|
||||||
# FROM golang as builder
|
# FROM golang as builder
|
||||||
# WORKDIR /
|
# WORKDIR /
|
||||||
# RUN git clone https://github.com/qemu-tools/qemu-host.git
|
# RUN git clone https://github.com/qemus/qemu-host.git
|
||||||
# WORKDIR /qemu-host/src
|
# WORKDIR /qemu-host/src
|
||||||
# RUN go mod download
|
# RUN go mod download
|
||||||
# RUN CGO_ENABLED=0 GOOS=linux go build -a -installsuffix cgo -o /qemu-host.bin .
|
# RUN CGO_ENABLED=0 GOOS=linux go build -a -installsuffix cgo -o /qemu-host.bin .
|
||||||
|
|
||||||
FROM debian:bookworm-slim
|
FROM debian:trixie-slim
|
||||||
|
|
||||||
ARG DEBCONF_NOWARNINGS="yes"
|
ARG TARGETPLATFORM
|
||||||
ARG DEBIAN_FRONTEND noninteractive
|
ARG DEBCONF_NOWARNINGS "yes"
|
||||||
|
ARG DEBIAN_FRONTEND "noninteractive"
|
||||||
|
ARG DEBCONF_NONINTERACTIVE_SEEN "true"
|
||||||
|
|
||||||
RUN if [ "$TARGETPLATFORM" = "linux/amd64" ]; then \
|
RUN if [ "$TARGETPLATFORM" != "linux/amd64" ]; then extra="qemu-user"; fi \
|
||||||
ADD_PKG="xserver-xorg-video-intel"; fi && \
|
&& apt-get update \
|
||||||
apt-get update && apt-get -y upgrade && \
|
&& apt-get --no-install-recommends -y install \
|
||||||
apt-get --no-install-recommends -y install \
|
jq \
|
||||||
curl \
|
tini \
|
||||||
cpio \
|
curl \
|
||||||
wget \
|
cpio \
|
||||||
fdisk \
|
wget \
|
||||||
unzip \
|
fdisk \
|
||||||
socat \
|
unzip \
|
||||||
procps \
|
socat \
|
||||||
xz-utils \
|
procps \
|
||||||
iptables \
|
xz-utils \
|
||||||
iproute2 \
|
iptables \
|
||||||
dnsmasq \
|
iproute2 \
|
||||||
net-tools \
|
apt-utils \
|
||||||
ca-certificates \
|
dnsmasq \
|
||||||
netcat-openbsd \
|
fakeroot \
|
||||||
qemu-system-x86 \
|
net-tools \
|
||||||
${ADD_PKG} \
|
qemu-utils \
|
||||||
|
ca-certificates \
|
||||||
|
netcat-openbsd \
|
||||||
|
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 agent/*.sh /agent/
|
|
||||||
|
|
||||||
|
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
|
||||||
|
EXPOSE 22 139 445 5000
|
||||||
|
|
||||||
EXPOSE 22
|
ENV RAM_SIZE "1G"
|
||||||
EXPOSE 80
|
|
||||||
EXPOSE 139
|
|
||||||
EXPOSE 445
|
|
||||||
EXPOSE 5000
|
|
||||||
|
|
||||||
ENV CPU_CORES "1"
|
|
||||||
ENV DISK_SIZE "16G"
|
ENV DISK_SIZE "16G"
|
||||||
ENV RAM_SIZE "512M"
|
ENV CPU_CORES "1"
|
||||||
|
|
||||||
ARG DATE_ARG=""
|
ARG VERSION_ARG "0.0"
|
||||||
ARG BUILD_ARG=0
|
RUN echo "$VERSION_ARG" > /run/version
|
||||||
ARG VERSION_ARG="0.0"
|
|
||||||
ENV VERSION=$VERSION_ARG
|
|
||||||
|
|
||||||
LABEL org.opencontainers.image.created=${DATE_ARG}
|
HEALTHCHECK --interval=60s --start-period=45s --retries=2 CMD /run/check.sh
|
||||||
LABEL org.opencontainers.image.revision=${BUILD_ARG}
|
|
||||||
LABEL org.opencontainers.image.version=${VERSION_ARG}
|
|
||||||
LABEL org.opencontainers.image.source=https://github.com/kroese/virtual-dsm/
|
|
||||||
LABEL org.opencontainers.image.url=https://hub.docker.com/r/kroese/virtual-dsm/
|
|
||||||
|
|
||||||
HEALTHCHECK --interval=60s --retries=2 CMD /run/check.sh
|
ENTRYPOINT ["/usr/bin/tini", "-s", "/run/entry.sh"]
|
||||||
|
|
||||||
ENTRYPOINT ["/run/run.sh"]
|
|
||||||
|
|||||||
159
agent/agent.sh
159
agent/agent.sh
@@ -1,159 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -u
|
|
||||||
|
|
||||||
VERSION="7"
|
|
||||||
HEADER="VirtualDSM Agent"
|
|
||||||
|
|
||||||
# Functions
|
|
||||||
|
|
||||||
error () { echo -e "\E[1;31m❯ ERROR: $1\E[0m" ; }
|
|
||||||
info () { echo -e "\E[1;34m❯\E[1;36m $1\E[0m" ; }
|
|
||||||
|
|
||||||
finish() {
|
|
||||||
|
|
||||||
echo "❯ $HEADER: Shutting down.."
|
|
||||||
exit
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
function checkNMI {
|
|
||||||
|
|
||||||
local nmi
|
|
||||||
nmi=$(cat /proc/interrupts | grep NMI | sed 's/[^1-9]*//g')
|
|
||||||
|
|
||||||
if [ "$nmi" != "" ]; then
|
|
||||||
|
|
||||||
info "Received shutdown request through NMI.."
|
|
||||||
|
|
||||||
/usr/syno/sbin/synoshutdown -s > /dev/null
|
|
||||||
finish
|
|
||||||
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
function downloadUpdate {
|
|
||||||
|
|
||||||
TMP="/tmp/agent.sh"
|
|
||||||
rm -f "${TMP}"
|
|
||||||
|
|
||||||
# Auto update the agent
|
|
||||||
|
|
||||||
URL="https://raw.githubusercontent.com/kroese/virtual-dsm/master/agent/agent.sh"
|
|
||||||
|
|
||||||
remote_size=$(curl -sIk -m 4 "${URL}" | grep -i "content-length:" | tr -d " \t" | cut -d ':' -f 2)
|
|
||||||
remote_size=${remote_size//$'\r'}
|
|
||||||
|
|
||||||
[[ "$remote_size" == "" || "$remote_size" == "0" ]] && return
|
|
||||||
|
|
||||||
SCRIPT=$(readlink -f "${BASH_SOURCE[0]}")
|
|
||||||
local_size=$(stat -c%s "$SCRIPT")
|
|
||||||
|
|
||||||
[[ remote_size -eq local_size ]] && return
|
|
||||||
|
|
||||||
if ! curl -sfk -m 10 -o "${TMP}" "${URL}"; then
|
|
||||||
error "$HEADER: curl error ($?)" && return
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ ! -f "${TMP}" ]; then
|
|
||||||
error "$HEADER: update error, file not found.." && return
|
|
||||||
fi
|
|
||||||
|
|
||||||
line=$(head -1 "${TMP}")
|
|
||||||
|
|
||||||
if [[ "$line" != "#!/usr/bin/env bash" ]]; then
|
|
||||||
error "$HEADER: update error, invalid header: $line" && return
|
|
||||||
fi
|
|
||||||
|
|
||||||
if cmp --silent -- "${TMP}" "${SCRIPT}"; then
|
|
||||||
error "$HEADER: update file is already equal? (${local_size} / ${remote_size})" && return
|
|
||||||
fi
|
|
||||||
|
|
||||||
mv -f "${TMP}" "${SCRIPT}"
|
|
||||||
chmod 755 "${SCRIPT}"
|
|
||||||
|
|
||||||
info "$HEADER: succesfully installed update..."
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
function installPackages {
|
|
||||||
|
|
||||||
for filename in /usr/local/packages/*.spk; do
|
|
||||||
if [ -f "$filename" ]; then
|
|
||||||
|
|
||||||
BASE=$(basename "$filename" .spk)
|
|
||||||
BASE="${BASE%%-*}"
|
|
||||||
|
|
||||||
[[ $BASE == "ActiveInsight" ]] && continue
|
|
||||||
|
|
||||||
info "Installing package ${BASE}.."
|
|
||||||
|
|
||||||
/usr/syno/bin/synopkg install "$filename" > /dev/null
|
|
||||||
/usr/syno/bin/synopkg start "$BASE" > /dev/null &
|
|
||||||
|
|
||||||
rm "$filename"
|
|
||||||
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
trap finish SIGINT SIGTERM
|
|
||||||
|
|
||||||
ts=$(date +%s%N)
|
|
||||||
echo "❯ Started $HEADER v$VERSION..."
|
|
||||||
|
|
||||||
checkNMI
|
|
||||||
|
|
||||||
# Install packages
|
|
||||||
|
|
||||||
first_run=false
|
|
||||||
|
|
||||||
for filename in /usr/local/packages/*.spk; do
|
|
||||||
if [ -f "$filename" ]; then
|
|
||||||
first_run=true
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
|
|
||||||
if [ "$first_run" = true ]; then
|
|
||||||
|
|
||||||
installPackages
|
|
||||||
|
|
||||||
else
|
|
||||||
|
|
||||||
downloadUpdate
|
|
||||||
|
|
||||||
fi
|
|
||||||
|
|
||||||
delay=500
|
|
||||||
elapsed=$((($(date +%s%N) - ts)/1000000))
|
|
||||||
|
|
||||||
if [[ delay -gt elapsed ]]; then
|
|
||||||
difference=$((delay-elapsed))
|
|
||||||
float=$(echo | awk -v diff="${difference}" '{print diff * 0.001}')
|
|
||||||
sleep "$float"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Display message in docker log output
|
|
||||||
|
|
||||||
IP=$(ip address show dev eth0 | grep inet | awk '/inet / { print $2 }' | cut -f1 -d/)
|
|
||||||
|
|
||||||
if [[ "$IP" == "20.20"* ]]; then
|
|
||||||
MSG="port 5000"
|
|
||||||
else
|
|
||||||
MSG="http://${IP}:5000"
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo ""
|
|
||||||
info "--------------------------------------------------------"
|
|
||||||
info " You can now login to DSM at ${MSG}"
|
|
||||||
info "--------------------------------------------------------"
|
|
||||||
echo ""
|
|
||||||
|
|
||||||
# Wait for NMI interrupt as a shutdown signal
|
|
||||||
|
|
||||||
while true; do
|
|
||||||
|
|
||||||
checkNMI
|
|
||||||
sleep 2 & wait $!
|
|
||||||
|
|
||||||
done
|
|
||||||
@@ -1,87 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
|
|
||||||
PIDFILE="/var/run/agent.pid"
|
|
||||||
SCRIPT="/usr/local/bin/agent.sh"
|
|
||||||
|
|
||||||
error () { echo -e "\E[1;31m❯ ERROR: $1\E[0m" ; }
|
|
||||||
info () { echo -e "\E[1;34m❯\E[1;36m $1\E[0m" ; }
|
|
||||||
|
|
||||||
status() {
|
|
||||||
|
|
||||||
if [ -f "$PIDFILE" ] && kill -0 "$(cat "$PIDFILE")"; then
|
|
||||||
echo 'Service running'
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
start() {
|
|
||||||
|
|
||||||
if [ -f "$PIDFILE" ] && kill -0 "$(cat "$PIDFILE")"; then
|
|
||||||
echo 'Service already running'
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo 'Starting agent service...'
|
|
||||||
chmod 666 /dev/ttyS0
|
|
||||||
|
|
||||||
if [ ! -f "$SCRIPT" ]; then
|
|
||||||
|
|
||||||
URL="https://raw.githubusercontent.com/kroese/virtual-dsm/master/agent/agent.sh"
|
|
||||||
|
|
||||||
if ! curl -sfk -m 10 -o "${SCRIPT}" "${URL}"; then
|
|
||||||
error 'Failed to download agent script.' > /dev/ttyS0
|
|
||||||
rm -f "${SCRIPT}"
|
|
||||||
return 1
|
|
||||||
else
|
|
||||||
info 'Agent script was missing?' > /dev/ttyS0
|
|
||||||
fi
|
|
||||||
|
|
||||||
chmod 755 "${SCRIPT}"
|
|
||||||
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo "-" > /var/lock/subsys/agent.sh
|
|
||||||
"$SCRIPT" &> /dev/ttyS0 & echo $! > "$PIDFILE"
|
|
||||||
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
stop() {
|
|
||||||
|
|
||||||
if [ ! -f "$PIDFILE" ] || ! kill -0 "$(cat "$PIDFILE")"; then
|
|
||||||
echo 'Service not running'
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo 'Stopping agent service...'
|
|
||||||
|
|
||||||
chmod 666 /dev/ttyS0
|
|
||||||
info 'Stopping agent service...' > /dev/ttyS0
|
|
||||||
|
|
||||||
kill -15 "$(cat "$PIDFILE")" && rm -f "$PIDFILE"
|
|
||||||
rm -f /var/lock/subsys/agent.sh
|
|
||||||
|
|
||||||
echo 'Service stopped'
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
case "$1" in
|
|
||||||
start)
|
|
||||||
start
|
|
||||||
;;
|
|
||||||
stop)
|
|
||||||
stop
|
|
||||||
;;
|
|
||||||
status)
|
|
||||||
status
|
|
||||||
;;
|
|
||||||
restart)
|
|
||||||
stop
|
|
||||||
start
|
|
||||||
;;
|
|
||||||
*)
|
|
||||||
echo "Usage: $0 {start|stop|restart}"
|
|
||||||
exit 1
|
|
||||||
esac
|
|
||||||
@@ -2,21 +2,20 @@ version: "3"
|
|||||||
services:
|
services:
|
||||||
dsm:
|
dsm:
|
||||||
container_name: dsm
|
container_name: dsm
|
||||||
image: kroese/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/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
|
||||||
|
|||||||
170
readme.md
170
readme.md
@@ -1,6 +1,6 @@
|
|||||||
<h1 align="center">Virtual DSM for Docker<br />
|
<h1 align="center">Virtual DSM for Docker<br />
|
||||||
<div align="center">
|
<div align="center">
|
||||||
<img src="https://github.com/kroese/virtual-dsm/raw/master/.github/screen.jpg" title="Screenshot" style="max-width:100%;" width="432" />
|
<img src="https://github.com/vdsm/virtual-dsm/raw/master/.github/screen.jpg" title="Screenshot" style="max-width:100%;" width="432" />
|
||||||
</div>
|
</div>
|
||||||
<div align="center">
|
<div align="center">
|
||||||
|
|
||||||
@@ -10,15 +10,16 @@
|
|||||||
[![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
|
||||||
- Graceful shutdown
|
- GPU passthrough
|
||||||
- Upgrades supported
|
- Upgrades supported
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
|
|
||||||
Via `docker-compose.yml`
|
Via `docker-compose.yml`
|
||||||
@@ -26,28 +27,27 @@ Via `docker-compose.yml`
|
|||||||
```yaml
|
```yaml
|
||||||
version: "3"
|
version: "3"
|
||||||
services:
|
services:
|
||||||
dsm:
|
dsm:
|
||||||
container_name: dsm
|
container_name: dsm
|
||||||
image: kroese/virtual-dsm:latest
|
image: vdsm/virtual-dsm:latest
|
||||||
environment:
|
environment:
|
||||||
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:
|
- /var/dsm:/storage
|
||||||
- /opt/dsm:/storage
|
restart: on-failure
|
||||||
restart: on-failure
|
stop_grace_period: 2m
|
||||||
stop_grace_period: 1m
|
|
||||||
```
|
```
|
||||||
|
|
||||||
Via `docker run`
|
Via `docker run`
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
docker run -it --rm -p 5000:5000 --device=/dev/kvm --cap-add NET_ADMIN --stop-timeout 60 kroese/virtual-dsm:latest
|
docker run -it --rm -p 5000:5000 --device=/dev/kvm --cap-add NET_ADMIN --stop-timeout 60 vdsm/virtual-dsm:latest
|
||||||
```
|
```
|
||||||
|
|
||||||
## FAQ
|
## FAQ
|
||||||
@@ -58,41 +58,58 @@ 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 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?
|
* ### How do I add multiple disks?
|
||||||
|
|
||||||
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:
|
To create additional disks, modify your compose file like this:
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
environment:
|
||||||
|
DISK2_SIZE: "32G"
|
||||||
|
DISK3_SIZE: "64G"
|
||||||
|
volumes:
|
||||||
|
- /home/example:/storage2
|
||||||
|
- /mnt/data/example:/storage3
|
||||||
|
```
|
||||||
|
|
||||||
|
* ### 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:
|
||||||
ALLOCATE: "N"
|
DISK_FMT: "qcow2"
|
||||||
```
|
```
|
||||||
|
|
||||||
Keep in mind that this will not affect any of your existing disks, it only applies to newly created disks.
|
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:
|
||||||
CPU_CORES: "4"
|
RAM_SIZE: "4G"
|
||||||
RAM_SIZE: "2048M"
|
CPU_CORES: "4"
|
||||||
```
|
```
|
||||||
|
|
||||||
* ### How do I verify if my system supports KVM?
|
* ### How do I verify if my system supports KVM?
|
||||||
@@ -104,7 +121,7 @@ docker run -it --rm -p 5000:5000 --device=/dev/kvm --cap-add NET_ADMIN --stop-ti
|
|||||||
sudo kvm-ok
|
sudo kvm-ok
|
||||||
```
|
```
|
||||||
|
|
||||||
If you receive an error from `kvm-ok` indicating that KVM acceleration can't be used, check your BIOS settings.
|
If you receive an error from `kvm-ok` indicating that KVM acceleration can't be used, check the virtualization settings in the BIOS.
|
||||||
|
|
||||||
* ### How do I assign an individual IP address to the container?
|
* ### How do I assign an individual IP address to the container?
|
||||||
|
|
||||||
@@ -126,16 +143,16 @@ docker run -it --rm -p 5000:5000 --device=/dev/kvm --cap-add NET_ADMIN --stop-ti
|
|||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
services:
|
services:
|
||||||
dsm:
|
dsm:
|
||||||
container_name: dsm
|
container_name: dsm
|
||||||
..<snip>..
|
..<snip>..
|
||||||
networks:
|
networks:
|
||||||
vdsm:
|
vdsm:
|
||||||
ipv4_address: 192.168.0.100
|
ipv4_address: 192.168.0.100
|
||||||
|
|
||||||
networks:
|
networks:
|
||||||
vdsm:
|
vdsm:
|
||||||
external: true
|
external: true
|
||||||
```
|
```
|
||||||
|
|
||||||
An added benefit of this approach is that you won't have to perform any port mapping anymore since all ports will be exposed by default.
|
An added benefit of this approach is that you won't have to perform any port mapping anymore since all ports will be exposed by default.
|
||||||
@@ -150,40 +167,57 @@ docker run -it --rm -p 5000:5000 --device=/dev/kvm --cap-add NET_ADMIN --stop-ti
|
|||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
environment:
|
environment:
|
||||||
DHCP: "Y"
|
DHCP: "Y"
|
||||||
devices:
|
|
||||||
- /dev/vhost-net
|
|
||||||
device_cgroup_rules:
|
device_cgroup_rules:
|
||||||
- '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 passthrough a disk?
|
||||||
|
|
||||||
|
When running the container inside a virtualized environment, it is possible to passthrough disk devices directly, instead of binding a folder containing an image. As these devices are already backed by an image on the host, this removes an extra layer.
|
||||||
|
|
||||||
|
This allows for easier management and higher performance. To do so, you can add those devices to your compose file:
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
environment:
|
||||||
|
DEVICE: "/dev/sda"
|
||||||
|
DEVICE2: "/dev/sdb"
|
||||||
|
devices:
|
||||||
|
- /dev/sda
|
||||||
|
- /dev/sdb
|
||||||
|
```
|
||||||
|
|
||||||
|
Please beware that any existing data on the device will be wiped, as DSM will format its partition table during first use. So do NOT passthrough devices containing valueable data.
|
||||||
|
|
||||||
* ### How do I install a specific version of vDSM?
|
* ### How do I install a specific version of vDSM?
|
||||||
|
|
||||||
By default, version 7.2 will be installed, but if you prefer an older version, you can add its URL to your compose file as follows:
|
By default, version 7.2 will be installed, but if you prefer an older version, you can add its download URL to your compose file as follows:
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
environment:
|
environment:
|
||||||
URL: "https://global.synologydownload.com/download/DSM/release/7.0.1/42218/DSM_VirtualDSM_42218.pat"
|
URL: "https://global.synologydownload.com/download/DSM/release/7.0.1/42218/DSM_VirtualDSM_42218.pat"
|
||||||
```
|
```
|
||||||
|
|
||||||
With this method, you are able to switch between different versions while keeping your file data.
|
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 enable this feature, add the following lines to your compose file:
|
|
||||||
|
|
||||||
```yaml
|
|
||||||
environment:
|
|
||||||
GPU: "Y"
|
|
||||||
devices:
|
|
||||||
- /dev/dri
|
|
||||||
```
|
|
||||||
|
|
||||||
* ### 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 provided, and Surveillance Station doesn't 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.
|
||||||
|
|
||||||
* ### Is this project legal?
|
* ### Is this project legal?
|
||||||
|
|
||||||
@@ -195,11 +229,11 @@ docker run -it --rm -p 5000:5000 --device=/dev/kvm --cap-add NET_ADMIN --stop-ti
|
|||||||
|
|
||||||
Only run this container on Synology hardware, any other use is not permitted by their EULA. The product names, logos, brands, and other trademarks referred to within this project are the property of their respective trademark holders. This project is not affiliated, sponsored, or endorsed by Synology, Inc.
|
Only run this container on Synology hardware, any other use is not permitted by their EULA. The product names, logos, brands, and other trademarks referred to within this project are the property of their respective trademark holders. This project is not affiliated, sponsored, or endorsed by Synology, Inc.
|
||||||
|
|
||||||
[build_url]: https://github.com/kroese/virtual-dsm/
|
[build_url]: https://github.com/vdsm/virtual-dsm/
|
||||||
[hub_url]: https://hub.docker.com/r/kroese/virtual-dsm
|
[hub_url]: https://hub.docker.com/r/vdsm/virtual-dsm
|
||||||
[tag_url]: https://hub.docker.com/r/kroese/virtual-dsm/tags
|
[tag_url]: https://hub.docker.com/r/vdsm/virtual-dsm/tags
|
||||||
|
|
||||||
[Build]: https://github.com/kroese/virtual-dsm/actions/workflows/build.yml/badge.svg
|
[Build]: https://github.com/vdsm/virtual-dsm/actions/workflows/build.yml/badge.svg
|
||||||
[Size]: https://img.shields.io/docker/image-size/kroese/virtual-dsm/latest?color=066da5&label=size
|
[Size]: https://img.shields.io/docker/image-size/vdsm/virtual-dsm/latest?color=066da5&label=size
|
||||||
[Pulls]: https://img.shields.io/docker/pulls/kroese/virtual-dsm.svg?style=flat&label=pulls&logo=docker
|
[Pulls]: https://img.shields.io/docker/pulls/kroese/virtual-dsm.svg?style=flat&label=pulls&logo=docker
|
||||||
[Version]: https://img.shields.io/docker/v/kroese/virtual-dsm/latest?arch=amd64&sort=semver&color=066da5
|
[Version]: https://img.shields.io/docker/v/vdsm/virtual-dsm/latest?arch=amd64&sort=semver&color=066da5
|
||||||
|
|||||||
49
run/check.sh
49
run/check.sh
@@ -1,49 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -u
|
|
||||||
|
|
||||||
[ ! -f "/run/qemu.pid" ] && echo "QEMU not running yet.." && exit 0
|
|
||||||
|
|
||||||
# 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
|
|
||||||
|
|
||||||
if ! curl -m 3 -ILfSs "http://${IP}:${PORT}/" > /dev/null; then
|
|
||||||
echo "Failed to reach ${IP}:${PORT}"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ "$IP" == "20.20"* ]]; then
|
|
||||||
echo "Healthcheck OK"
|
|
||||||
else
|
|
||||||
echo "Healthcheck OK ( $IP )"
|
|
||||||
fi
|
|
||||||
|
|
||||||
exit 0
|
|
||||||
155
run/disk.sh
155
run/disk.sh
@@ -1,155 +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
|
|
||||||
|
|
||||||
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 < 6442450944 )); 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
|
|
||||||
|
|
||||||
# Format as BTRFS filesystem
|
|
||||||
# mkfs.btrfs -q -L data -d single -m dup "${DATA}" > /dev/null
|
|
||||||
|
|
||||||
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
|
|
||||||
|
|
||||||
AGENT="${STORAGE}/${BASE}.agent"
|
|
||||||
[ -f "$AGENT" ] && AGENT_VERSION=$(cat "${AGENT}") || AGENT_VERSION=1
|
|
||||||
|
|
||||||
if ((AGENT_VERSION < 5)); then
|
|
||||||
info "The installed VirtualDSM Agent v${AGENT_VERSION} is an outdated version, please upgrade it."
|
|
||||||
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"
|
|
||||||
274
run/install.sh
274
run/install.sh
@@ -1,274 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -Eeuo pipefail
|
|
||||||
|
|
||||||
# Display wait message
|
|
||||||
/run/server.sh 5000 install &
|
|
||||||
|
|
||||||
# Download the required files from the Synology website
|
|
||||||
DL="https://global.synologydownload.com/download/DSM"
|
|
||||||
|
|
||||||
if [ -z "$URL" ]; then
|
|
||||||
|
|
||||||
if [ "$ARCH" == "amd64" ]; then
|
|
||||||
URL="$DL/release/7.2/64570-1/DSM_VirtualDSM_64570.pat"
|
|
||||||
else
|
|
||||||
URL="$DL/release/7.0.1/42218/DSM_VirtualDSM_42218.pat"
|
|
||||||
fi
|
|
||||||
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Check if output is to interactive TTY
|
|
||||||
if [ -t 1 ]; then
|
|
||||||
PROGRESS="--progress=bar:noscroll"
|
|
||||||
else
|
|
||||||
PROGRESS="--progress=dot:giga"
|
|
||||||
fi
|
|
||||||
|
|
||||||
BASE=$(basename "$URL" .pat)
|
|
||||||
|
|
||||||
rm -f "$STORAGE"/"$BASE".pat
|
|
||||||
rm -f "$STORAGE"/"$BASE".agent
|
|
||||||
rm -f "$STORAGE"/"$BASE".boot.img
|
|
||||||
rm -f "$STORAGE"/"$BASE".system.img
|
|
||||||
|
|
||||||
TMP="/tmp/dsm"
|
|
||||||
FS=$(stat -f -c %T "$STORAGE")
|
|
||||||
[[ "$FS" == "ext"* ]] && TMP="$STORAGE/tmp"
|
|
||||||
rm -rf "$TMP" && mkdir -p "$TMP"
|
|
||||||
|
|
||||||
# Check free diskspace
|
|
||||||
MIN_SPACE=5842450944
|
|
||||||
SPACE=$(df --output=avail -B 1 "$TMP" | tail -n 1)
|
|
||||||
(( MIN_SPACE > SPACE )) && error "Not enough free space for installation." && exit 95
|
|
||||||
|
|
||||||
[[ "${DEBUG}" == [Yy1]* ]] && set -x
|
|
||||||
|
|
||||||
RDC="$STORAGE/dsm.rd"
|
|
||||||
|
|
||||||
if [ ! -f "${RDC}" ]; then
|
|
||||||
|
|
||||||
info "Install: Downloading installer..."
|
|
||||||
|
|
||||||
RD="$TMP/rd.gz"
|
|
||||||
POS="65627648-71021835"
|
|
||||||
VERIFY="b4215a4b213ff5154db0488f92c87864"
|
|
||||||
LOC="$DL/release/7.0.1/42218/DSM_VirtualDSM_42218.pat"
|
|
||||||
|
|
||||||
{ curl -r "$POS" -sfk -o "$RD" "$LOC"; rc=$?; } || :
|
|
||||||
(( rc != 0 )) && error "Failed to download $LOC, reason: $rc" && exit 60
|
|
||||||
|
|
||||||
SUM=$(md5sum "$RD" | cut -f 1 -d " ")
|
|
||||||
|
|
||||||
if [ "$SUM" != "$VERIFY" ]; then
|
|
||||||
|
|
||||||
PAT="/install.pat"
|
|
||||||
rm "$RD"
|
|
||||||
rm -f "$PAT"
|
|
||||||
|
|
||||||
{ wget "$LOC" -O "$PAT" -q --no-check-certificate --show-progress "$PROGRESS"; rc=$?; } || :
|
|
||||||
(( rc != 0 )) && error "Failed to download $LOC, reason: $rc" && exit 60
|
|
||||||
|
|
||||||
tar --extract --file="$PAT" --directory="$(dirname "${RD}")"/. "$(basename "${RD}")"
|
|
||||||
rm "$PAT"
|
|
||||||
|
|
||||||
fi
|
|
||||||
|
|
||||||
cp "$RD" "$RDC"
|
|
||||||
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ -f "${RDC}" ]; then
|
|
||||||
|
|
||||||
{ xz -dc <"$RDC" >"$TMP/rd" 2>/dev/null; rc=$?; } || :
|
|
||||||
(( rc != 1 )) && error "Failed to unxz $RDC, reason $rc" && exit 91
|
|
||||||
|
|
||||||
{ (cd "$TMP" && cpio -idm <"$TMP/rd" 2>/dev/null); rc=$?; } || :
|
|
||||||
(( rc != 0 )) && error "Failed to cpio $RDC, reason $rc" && exit 92
|
|
||||||
|
|
||||||
mkdir -p /run/extract
|
|
||||||
for file in $TMP/usr/lib/libcurl.so.4 \
|
|
||||||
$TMP/usr/lib/libmbedcrypto.so.5 \
|
|
||||||
$TMP/usr/lib/libmbedtls.so.13 \
|
|
||||||
$TMP/usr/lib/libmbedx509.so.1 \
|
|
||||||
$TMP/usr/lib/libmsgpackc.so.2 \
|
|
||||||
$TMP/usr/lib/libsodium.so \
|
|
||||||
$TMP/usr/lib/libsynocodesign-ng-virtual-junior-wins.so.7 \
|
|
||||||
$TMP/usr/syno/bin/scemd; do
|
|
||||||
cp "$file" /run/extract/
|
|
||||||
done
|
|
||||||
|
|
||||||
if [ "$ARCH" != "amd64" ]; then
|
|
||||||
mkdir -p /lib64/
|
|
||||||
cp "$TMP/usr/lib/libc.so.6" /lib64/
|
|
||||||
cp "$TMP/usr/lib/libpthread.so.0" /lib64/
|
|
||||||
cp "$TMP/usr/lib/ld-linux-x86-64.so.2" /lib64/
|
|
||||||
fi
|
|
||||||
|
|
||||||
mv /run/extract/scemd /run/extract/syno_extract_system_patch
|
|
||||||
chmod +x /run/extract/syno_extract_system_patch
|
|
||||||
|
|
||||||
fi
|
|
||||||
|
|
||||||
rm -rf "$TMP" && mkdir -p "$TMP"
|
|
||||||
|
|
||||||
info "Install: Downloading $(basename "$URL")..."
|
|
||||||
|
|
||||||
PAT="/$BASE.pat"
|
|
||||||
rm -f "$PAT"
|
|
||||||
|
|
||||||
{ wget "$URL" -O "$PAT" -q --no-check-certificate --show-progress "$PROGRESS"; rc=$?; } || :
|
|
||||||
|
|
||||||
(( rc != 0 )) && error "Failed to download $URL, reason: $rc" && exit 69
|
|
||||||
[ ! -f "$PAT" ] && error "Failed to download $URL" && exit 69
|
|
||||||
|
|
||||||
SIZE=$(stat -c%s "$PAT")
|
|
||||||
|
|
||||||
if ((SIZE<250000000)); then
|
|
||||||
error "The specified PAT file is probably an update pack as it's too small." && exit 62
|
|
||||||
fi
|
|
||||||
|
|
||||||
info "Install: Extracting downloaded image..."
|
|
||||||
|
|
||||||
if { tar tf "$PAT"; } >/dev/null 2>&1; then
|
|
||||||
|
|
||||||
tar xpf "$PAT" -C "$TMP/."
|
|
||||||
|
|
||||||
else
|
|
||||||
|
|
||||||
if [ "$ARCH" != "amd64" ]; then
|
|
||||||
|
|
||||||
export DEBCONF_NOWARNINGS="yes"
|
|
||||||
export DEBIAN_FRONTEND="noninteractive"
|
|
||||||
|
|
||||||
apt-get -qq update
|
|
||||||
apt-get -qq -y upgrade
|
|
||||||
apt-get -qq --no-install-recommends -y install qemu-user > /dev/null
|
|
||||||
|
|
||||||
export DEBIAN_FRONTEND=""
|
|
||||||
export DEBCONF_NOWARNINGS=""
|
|
||||||
|
|
||||||
fi
|
|
||||||
|
|
||||||
export LD_LIBRARY_PATH="/run/extract"
|
|
||||||
|
|
||||||
if [ "$ARCH" == "amd64" ]; then
|
|
||||||
{ /run/extract/syno_extract_system_patch "$PAT" "$TMP/."; rc=$?; } || :
|
|
||||||
else
|
|
||||||
{ qemu-x86_64 /run/extract/syno_extract_system_patch "$PAT" "$TMP/."; rc=$?; } || :
|
|
||||||
fi
|
|
||||||
|
|
||||||
export LD_LIBRARY_PATH=""
|
|
||||||
|
|
||||||
(( rc != 0 )) && error "Failed to extract PAT file, reason $rc" && exit 63
|
|
||||||
|
|
||||||
fi
|
|
||||||
|
|
||||||
HDA="$TMP/hda1"
|
|
||||||
IDB="$TMP/indexdb"
|
|
||||||
PKG="$TMP/packages"
|
|
||||||
HDP="$TMP/synohdpack_img"
|
|
||||||
|
|
||||||
[ ! -f "$HDA.tgz" ] && error "The PAT file contains no OS image." && exit 64
|
|
||||||
|
|
||||||
BOOT=$(find "$TMP" -name "*.bin.zip")
|
|
||||||
[ ! -f "$BOOT" ] && error "The PAT file contains no boot image." && exit 67
|
|
||||||
|
|
||||||
BOOT=$(echo "$BOOT" | head -c -5)
|
|
||||||
unzip -q -o "$BOOT".zip -d "$TMP"
|
|
||||||
|
|
||||||
[[ "${ALLOCATE}" == [Zz]* ]] && info "Install: Allocating diskspace..."
|
|
||||||
|
|
||||||
SYSTEM="$TMP/sys.img"
|
|
||||||
SYSTEM_SIZE=4954537983
|
|
||||||
|
|
||||||
# Check free diskspace
|
|
||||||
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
|
|
||||||
|
|
||||||
if ! fallocate -l "${SYSTEM_SIZE}" "${SYSTEM}"; then
|
|
||||||
if ! truncate -s "${SYSTEM_SIZE}" "${SYSTEM}"; then
|
|
||||||
rm -f "${SYSTEM}" && error "Could not allocate a file for the system disk." && exit 88
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ "${ALLOCATE}" == [Zz]* ]]; then
|
|
||||||
info "Install: Preallocating 4 GB of diskspace..."
|
|
||||||
dd if=/dev/urandom of="${SYSTEM}" count="${SYSTEM_SIZE}" bs=1M iflag=count_bytes status=none
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Check if file exists
|
|
||||||
[ ! -f "${SYSTEM}" ] && error "System disk does not exist ($SYSTEM)" && exit 89
|
|
||||||
|
|
||||||
# Check the filesize
|
|
||||||
SIZE=$(stat -c%s "${SYSTEM}")
|
|
||||||
[[ SIZE -ne SYSTEM_SIZE ]] && rm -f "${SYSTEM}" && error "System disk has the wrong size: ${SIZE}" && exit 90
|
|
||||||
|
|
||||||
PART="$TMP/partition.fdisk"
|
|
||||||
|
|
||||||
{ echo "label: dos"
|
|
||||||
echo "label-id: 0x6f9ee2e9"
|
|
||||||
echo "device: ${SYSTEM}"
|
|
||||||
echo "unit: sectors"
|
|
||||||
echo "sector-size: 512"
|
|
||||||
echo ""
|
|
||||||
echo "${SYSTEM}1 : start= 2048, size= 4980480, type=83"
|
|
||||||
echo "${SYSTEM}2 : start= 4982528, size= 4194304, type=82"
|
|
||||||
} > "$PART"
|
|
||||||
|
|
||||||
sfdisk -q "$SYSTEM" < "$PART"
|
|
||||||
|
|
||||||
info "Install: Extracting system partition..."
|
|
||||||
|
|
||||||
MOUNT="$TMP/system"
|
|
||||||
rm -rf "$MOUNT" && mkdir -p "$MOUNT"
|
|
||||||
|
|
||||||
mv "$HDA.tgz" "$HDA.txz"
|
|
||||||
tar xpfJ "$HDA.txz" --absolute-names -C "$MOUNT/"
|
|
||||||
|
|
||||||
[ -d "$PKG" ] && mv "$PKG/" "$MOUNT/.SynoUpgradePackages/"
|
|
||||||
rm -f "$MOUNT/.SynoUpgradePackages/ActiveInsight-"*
|
|
||||||
|
|
||||||
[ -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/"
|
|
||||||
|
|
||||||
# Install Agent
|
|
||||||
|
|
||||||
LOC="$MOUNT/usr/local/bin"
|
|
||||||
mkdir -p "$LOC"
|
|
||||||
cp /agent/agent.sh "$LOC/agent.sh"
|
|
||||||
chmod 755 "$LOC/agent.sh"
|
|
||||||
|
|
||||||
LOC="$MOUNT/usr/local/etc/rc.d"
|
|
||||||
mkdir -p "$LOC"
|
|
||||||
cp /agent/service.sh "$LOC/agent.sh"
|
|
||||||
chmod 755 "$LOC/agent.sh"
|
|
||||||
|
|
||||||
# Store agent version
|
|
||||||
echo "7" > "$STORAGE"/"$BASE".agent
|
|
||||||
|
|
||||||
info "Install: Installing system partition..."
|
|
||||||
|
|
||||||
LABEL="1.44.1-42218"
|
|
||||||
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"
|
|
||||||
|
|
||||||
rm -rf "$MOUNT"
|
|
||||||
|
|
||||||
echo "$BASE" > "$STORAGE"/dsm.ver
|
|
||||||
|
|
||||||
# Check free diskspace
|
|
||||||
SPACE=$(df --output=avail -B 1 "$STORAGE" | tail -n 1)
|
|
||||||
(( MIN_SPACE > SPACE )) && error "Not enough free space in storage folder." && exit 94
|
|
||||||
|
|
||||||
mv -f "$PAT" "$STORAGE"/"$BASE".pat
|
|
||||||
mv -f "$BOOT" "$STORAGE"/"$BASE".boot.img
|
|
||||||
mv -f "$SYSTEM" "$STORAGE"/"$BASE".system.img
|
|
||||||
|
|
||||||
rm -rf "$TMP"
|
|
||||||
|
|
||||||
{ set +x; } 2>/dev/null
|
|
||||||
[[ "${DEBUG}" == [Yy1]* ]] && echo
|
|
||||||
|
|
||||||
return 0
|
|
||||||
256
run/network.sh
256
run/network.sh
@@ -1,256 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -Eeuo pipefail
|
|
||||||
|
|
||||||
# Docker environment variables
|
|
||||||
|
|
||||||
: ${DHCP:='N'}
|
|
||||||
: ${MAC:='02:11:32:AA:BB:CC'}
|
|
||||||
|
|
||||||
: ${VM_NET_TAP:='dsm'}
|
|
||||||
: ${VM_NET_DEV:='eth0'}
|
|
||||||
: ${VM_NET_MAC:="$MAC"}
|
|
||||||
: ${VM_NET_HOST:='VirtualDSM'}
|
|
||||||
|
|
||||||
: ${DNS_SERVERS:=''}
|
|
||||||
: ${DNSMASQ_OPTS:=''}
|
|
||||||
: ${DNSMASQ:='/usr/sbin/dnsmasq'}
|
|
||||||
: ${DNSMASQ_CONF_DIR:='/etc/dnsmasq.d'}
|
|
||||||
|
|
||||||
# ######################################
|
|
||||||
# Functions
|
|
||||||
# ######################################
|
|
||||||
|
|
||||||
configureDHCP() {
|
|
||||||
|
|
||||||
# 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=$?; } || :
|
|
||||||
|
|
||||||
if (( rc != 0 )); then
|
|
||||||
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
|
|
||||||
fi
|
|
||||||
|
|
||||||
while ! ip link set "${VM_NET_TAP}" up; do
|
|
||||||
info "Waiting for address to become available..."
|
|
||||||
sleep 2
|
|
||||||
done
|
|
||||||
|
|
||||||
TAP_NR=$(</sys/class/net/"${VM_NET_TAP}"/ifindex)
|
|
||||||
TAP_PATH="/dev/tap${TAP_NR}"
|
|
||||||
|
|
||||||
# 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)
|
|
||||||
(( 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}"
|
|
||||||
|
|
||||||
if [[ ! -e "${TAP_PATH}" ]]; then
|
|
||||||
{ mknod "${TAP_PATH}" c "$MAJOR" "$MINOR" ; rc=$?; } || :
|
|
||||||
(( rc != 0 )) && error "Cannot mknod: ${TAP_PATH} ($rc)" && exit 20
|
|
||||||
fi
|
|
||||||
|
|
||||||
{ exec 30>>"$TAP_PATH"; rc=$?; } 2>/dev/null || :
|
|
||||||
|
|
||||||
if (( rc != 0 )); then
|
|
||||||
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
|
|
||||||
fi
|
|
||||||
|
|
||||||
{ exec 40>>/dev/vhost-net; rc=$?; } 2>/dev/null || :
|
|
||||||
|
|
||||||
if (( rc != 0 )); then
|
|
||||||
error "VHOST can not be found ($rc). Please add the following "
|
|
||||||
error "docker setting to your container: --device=/dev/vhost-net" && exit 22
|
|
||||||
fi
|
|
||||||
|
|
||||||
NET_OPTS="-netdev tap,id=hostnet0,vhost=on,vhostfd=40,fd=30"
|
|
||||||
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
configureDNS () {
|
|
||||||
|
|
||||||
# dnsmasq configuration:
|
|
||||||
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
|
|
||||||
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
|
|
||||||
|
|
||||||
# Build DNS options from container /etc/resolv.conf
|
|
||||||
|
|
||||||
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/^ *//')
|
|
||||||
|
|
||||||
[[ "${DEBUG}" == [Yy1]* ]] && set -x
|
|
||||||
$DNSMASQ ${DNSMASQ_OPTS:+ $DNSMASQ_OPTS}
|
|
||||||
{ set +x; } 2>/dev/null
|
|
||||||
[[ "${DEBUG}" == [Yy1]* ]] && echo
|
|
||||||
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
configureNAT () {
|
|
||||||
|
|
||||||
# Create a bridge with a static IP for the VM guest
|
|
||||||
|
|
||||||
VM_NET_IP='20.20.20.21'
|
|
||||||
[[ "${DEBUG}" == [Yy1]* ]] && set -x
|
|
||||||
|
|
||||||
{ ip link add dev dockerbridge type bridge ; rc=$?; } || :
|
|
||||||
|
|
||||||
if (( rc != 0 )); then
|
|
||||||
error "Capability NET_ADMIN has not been set most likely. Please add the "
|
|
||||||
error "following docker setting to your container: --cap-add NET_ADMIN" && exit 23
|
|
||||||
fi
|
|
||||||
|
|
||||||
ip address add ${VM_NET_IP%.*}.1/24 broadcast ${VM_NET_IP%.*}.255 dev dockerbridge
|
|
||||||
|
|
||||||
while ! ip link set dockerbridge up; do
|
|
||||||
info "Waiting for address to become available..."
|
|
||||||
sleep 2
|
|
||||||
done
|
|
||||||
|
|
||||||
# QEMU Works with taps, set tap to the bridge created
|
|
||||||
ip tuntap add dev "${VM_NET_TAP}" mode tap
|
|
||||||
|
|
||||||
while ! ip link set "${VM_NET_TAP}" up promisc on; do
|
|
||||||
info "Waiting for tap to become available..."
|
|
||||||
sleep 2
|
|
||||||
done
|
|
||||||
|
|
||||||
ip link set dev "${VM_NET_TAP}" master dockerbridge
|
|
||||||
|
|
||||||
|
|
||||||
# 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 PREROUTING -i "${VM_NET_DEV}" -d "${IP}" -p udp -j DNAT --to $VM_NET_IP
|
|
||||||
|
|
||||||
if (( KERNEL > 4 )); then
|
|
||||||
# Hack for guest VMs complaining about "bad udp checksums in 5 packets"
|
|
||||||
iptables -A POSTROUTING -t mangle -p udp --dport bootpc -j CHECKSUM --checksum-fill || true
|
|
||||||
fi
|
|
||||||
|
|
||||||
{ set +x; } 2>/dev/null
|
|
||||||
[[ "${DEBUG}" == [Yy1]* ]] && echo
|
|
||||||
|
|
||||||
# Check port forwarding flag
|
|
||||||
if [[ $(< /proc/sys/net/ipv4/ip_forward) -eq 0 ]]; then
|
|
||||||
{ sysctl -w net.ipv4.ip_forward=1 ; rc=$?; } || :
|
|
||||||
if (( rc != 0 )); then
|
|
||||||
error "Please add the following docker setting to your container: --sysctl net.ipv4.ip_forward=1" && exit 24
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
NET_OPTS="-netdev tap,ifname=${VM_NET_TAP},script=no,downscript=no,id=hostnet0"
|
|
||||||
|
|
||||||
{ exec 40>>/dev/vhost-net; rc=$?; } 2>/dev/null || :
|
|
||||||
(( rc == 0 )) && NET_OPTS="$NET_OPTS,vhost=on,vhostfd=40"
|
|
||||||
|
|
||||||
configureDNS
|
|
||||||
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
closeNetwork () {
|
|
||||||
|
|
||||||
if [[ "${DHCP}" == [Yy1]* ]]; then
|
|
||||||
|
|
||||||
ip link set "${VM_NET_TAP}" down || true
|
|
||||||
ip link delete "${VM_NET_TAP}" || true
|
|
||||||
|
|
||||||
else
|
|
||||||
|
|
||||||
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 delete dockerbridge || true
|
|
||||||
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
# ######################################
|
|
||||||
# Configure Network
|
|
||||||
# ######################################
|
|
||||||
|
|
||||||
{ pkill -f server.sh || true; } 2>/dev/null
|
|
||||||
|
|
||||||
# Create the necessary file structure for /dev/net/tun
|
|
||||||
if [ ! -c /dev/net/tun ]; then
|
|
||||||
[ ! -d /dev/net ] && mkdir -m 755 /dev/net
|
|
||||||
mknod /dev/net/tun c 10 200
|
|
||||||
chmod 666 /dev/net/tun
|
|
||||||
fi
|
|
||||||
|
|
||||||
[ ! -c /dev/net/tun ] && error "TUN network interface not available..." && exit 85
|
|
||||||
|
|
||||||
# Create the necessary file structure for /dev/vhost-net
|
|
||||||
if [ ! -c /dev/vhost-net ]; then
|
|
||||||
mknod /dev/vhost-net c 10 238
|
|
||||||
chmod 660 /dev/vhost-net
|
|
||||||
fi
|
|
||||||
|
|
||||||
update-alternatives --set iptables /usr/sbin/iptables-legacy > /dev/null
|
|
||||||
update-alternatives --set ip6tables /usr/sbin/ip6tables-legacy > /dev/null
|
|
||||||
|
|
||||||
VM_NET_MAC="${VM_NET_MAC//-/:}"
|
|
||||||
GATEWAY=$(ip r | grep default | awk '{print $3}')
|
|
||||||
|
|
||||||
if [[ "${DEBUG}" == [Yy1]* ]]; then
|
|
||||||
|
|
||||||
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
|
|
||||||
|
|
||||||
if [[ "${DHCP}" == [Yy1]* ]]; then
|
|
||||||
|
|
||||||
if [[ "$GATEWAY" == "172."* ]]; then
|
|
||||||
error "You can only enable DHCP while the container is on a macvlan network!" && exit 86
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Configuration for DHCP IP
|
|
||||||
configureDHCP
|
|
||||||
|
|
||||||
# Display IP on port 80 and 5000
|
|
||||||
/run/server.sh 5000 /run/ip.sh &
|
|
||||||
|
|
||||||
else
|
|
||||||
|
|
||||||
# Configuration for static IP
|
|
||||||
configureNAT
|
|
||||||
|
|
||||||
fi
|
|
||||||
|
|
||||||
NET_OPTS="${NET_OPTS} -device virtio-net-pci,romfile=,netdev=hostnet0,mac=${VM_NET_MAC},id=net0"
|
|
||||||
|
|
||||||
return 0
|
|
||||||
91
run/power.sh
91
run/power.sh
@@ -1,91 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -Eeuo pipefail
|
|
||||||
|
|
||||||
# Configure QEMU for graceful shutdown
|
|
||||||
|
|
||||||
QEMU_MONPORT=7100
|
|
||||||
QEMU_POWERDOWN_TIMEOUT=50
|
|
||||||
|
|
||||||
_QEMU_PID=/run/qemu.pid
|
|
||||||
_QEMU_SHUTDOWN_COUNTER=/run/qemu.counter
|
|
||||||
|
|
||||||
rm -f "${_QEMU_PID}"
|
|
||||||
rm -f "${_QEMU_SHUTDOWN_COUNTER}"
|
|
||||||
|
|
||||||
_trap(){
|
|
||||||
func="$1" ; shift
|
|
||||||
for sig ; do
|
|
||||||
trap "$func $sig" "$sig"
|
|
||||||
done
|
|
||||||
}
|
|
||||||
|
|
||||||
_graceful_shutdown() {
|
|
||||||
|
|
||||||
set +e
|
|
||||||
|
|
||||||
[ ! -f "${_QEMU_PID}" ] && return
|
|
||||||
[ -f "${_QEMU_SHUTDOWN_COUNTER}" ] && return
|
|
||||||
|
|
||||||
echo && info "Received $1 signal, shutting down..."
|
|
||||||
echo 0 > "${_QEMU_SHUTDOWN_COUNTER}"
|
|
||||||
|
|
||||||
# Don't send the powerdown signal because vDSM ignores ACPI signals
|
|
||||||
# echo 'system_powerdown' | nc -q 1 -w 1 localhost "${QEMU_MONPORT}" > /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)"
|
|
||||||
|
|
||||||
# If we cannot shutdown the usual way, fallback to the NMI method
|
|
||||||
|
|
||||||
AGENT="${STORAGE}/${BASE}.agent"
|
|
||||||
[ -f "$AGENT" ] && AGENT_VERSION=$(cat "${AGENT}") || AGENT_VERSION=1
|
|
||||||
|
|
||||||
if ((AGENT_VERSION > 1)); then
|
|
||||||
|
|
||||||
# Send a NMI interrupt which will be detected by the kernel
|
|
||||||
if ! echo 'nmi' | nc -q 1 -w 1 localhost "${QEMU_MONPORT}" > /dev/null ; then
|
|
||||||
AGENT_VERSION=0
|
|
||||||
fi
|
|
||||||
|
|
||||||
fi
|
|
||||||
|
|
||||||
if ((AGENT_VERSION < 2)); then
|
|
||||||
|
|
||||||
echo && info "Please update the VirtualDSM Agent to allow for gracefull shutdowns..."
|
|
||||||
|
|
||||||
kill -15 "$(cat "${_QEMU_PID}")"
|
|
||||||
pkill -f qemu-system-x86_64 || true
|
|
||||||
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
while [ "$(cat ${_QEMU_SHUTDOWN_COUNTER})" -lt "${QEMU_POWERDOWN_TIMEOUT}" ]; do
|
|
||||||
|
|
||||||
# Increase the counter
|
|
||||||
echo $(($(cat ${_QEMU_SHUTDOWN_COUNTER})+1)) > ${_QEMU_SHUTDOWN_COUNTER}
|
|
||||||
|
|
||||||
# Try to connect to qemu
|
|
||||||
if echo 'info version'| nc -q 1 -w 1 localhost "${QEMU_MONPORT}" >/dev/null 2>&1 ; then
|
|
||||||
|
|
||||||
sleep 1
|
|
||||||
#info "Shutting down, waiting... ($(cat ${_QEMU_SHUTDOWN_COUNTER})/${QEMU_POWERDOWN_TIMEOUT})"
|
|
||||||
|
|
||||||
fi
|
|
||||||
|
|
||||||
done
|
|
||||||
|
|
||||||
echo && echo "❯ Quitting..."
|
|
||||||
echo 'quit' | nc -q 1 -w 1 localhost "${QEMU_MONPORT}" >/dev/null 2>&1 || true
|
|
||||||
|
|
||||||
closeNetwork
|
|
||||||
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
_trap _graceful_shutdown SIGTERM SIGHUP SIGINT SIGABRT SIGQUIT
|
|
||||||
|
|
||||||
MON_OPTS="-monitor telnet:localhost:${QEMU_MONPORT},server,nowait,nodelay"
|
|
||||||
107
run/run.sh
107
run/run.sh
@@ -1,107 +0,0 @@
|
|||||||
#!/usr/bin/env bash
|
|
||||||
set -Eeuo pipefail
|
|
||||||
|
|
||||||
# Docker environment variables
|
|
||||||
|
|
||||||
: ${URL:=''} # URL of the PAT file
|
|
||||||
: ${GPU:='N'} # Enable GPU passthrough
|
|
||||||
: ${DEBUG:='N'} # Enable debug 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
|
|
||||||
|
|
||||||
echo "❯ Starting Virtual DSM for Docker v${VERSION}..."
|
|
||||||
|
|
||||||
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
|
|
||||||
|
|
||||||
STORAGE="/storage"
|
|
||||||
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)
|
|
||||||
|
|
||||||
[ ! -d "$STORAGE" ] && error "Storage folder (${STORAGE}) not found!" && exit 13
|
|
||||||
|
|
||||||
if [ -f "$STORAGE"/dsm.ver ]; then
|
|
||||||
BASE=$(cat "${STORAGE}/dsm.ver")
|
|
||||||
else
|
|
||||||
# Fallback for old installs
|
|
||||||
BASE="DSM_VirtualDSM_42962"
|
|
||||||
fi
|
|
||||||
|
|
||||||
[ -n "$URL" ] && BASE=$(basename "$URL" .pat)
|
|
||||||
|
|
||||||
if [[ ! -f "$STORAGE/$BASE.boot.img" ]] || [[ ! -f "$STORAGE/$BASE.system.img" ]]; then
|
|
||||||
. /run/install.sh
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Initialize disks
|
|
||||||
. /run/disk.sh
|
|
||||||
|
|
||||||
# Initialize network
|
|
||||||
. /run/network.sh
|
|
||||||
|
|
||||||
# Initialize serialport
|
|
||||||
. /run/serial.sh
|
|
||||||
|
|
||||||
# Configure shutdown
|
|
||||||
. /run/power.sh
|
|
||||||
|
|
||||||
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]* ]]; 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 ' ')
|
|
||||||
|
|
||||||
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
|
|
||||||
|
|
||||||
#if (( KERNEL > 5 )) || ( (( KERNEL == 5 )) && (( MINOR > 2 )) ); then
|
|
||||||
# pidwait -F "${_QEMU_PID}" & wait $!
|
|
||||||
#else
|
|
||||||
|
|
||||||
tail --pid "$(cat "${_QEMU_PID}")" --follow /dev/null & wait $!
|
|
||||||
@@ -1,59 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
set -Eeuo pipefail
|
|
||||||
|
|
||||||
# Docker environment variables
|
|
||||||
|
|
||||||
: ${HOST_CPU:=''}
|
|
||||||
: ${HOST_MAC:=''}
|
|
||||||
: ${HOST_BUILD:=''}
|
|
||||||
: ${HOST_DEBUG:=''}
|
|
||||||
: ${HOST_SERIAL:=''}
|
|
||||||
: ${GUEST_SERIAL:=''}
|
|
||||||
: ${HOST_MODEL:=''}
|
|
||||||
: ${HOST_VERSION:=''}
|
|
||||||
: ${HOST_TIMESTAMP:=''}
|
|
||||||
|
|
||||||
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_BUILD" ] && HOST_ARGS+=("-build=${HOST_BUILD}")
|
|
||||||
[ -n "$HOST_MODEL" ] && HOST_ARGS+=("-model=${HOST_MODEL}")
|
|
||||||
[ -n "$HOST_SERIAL" ] && HOST_ARGS+=("-hostsn=${HOST_SERIAL}")
|
|
||||||
[ -n "$GUEST_SERIAL" ] && HOST_ARGS+=("-guestsn=${GUEST_SERIAL}")
|
|
||||||
[ -n "$HOST_VERSION" ] && HOST_ARGS+=("-version=${HOST_VERSION}")
|
|
||||||
[ -n "$HOST_TIMESTAMP" ] && HOST_ARGS+=("-ts=${HOST_TIMESTAMP}")
|
|
||||||
|
|
||||||
if [[ "${HOST_DEBUG}" == [Yy1]* ]]; then
|
|
||||||
set -x
|
|
||||||
./run/host.bin "${HOST_ARGS[@]}" &
|
|
||||||
{ set +x; } 2>/dev/null
|
|
||||||
echo
|
|
||||||
else
|
|
||||||
./run/host.bin "${HOST_ARGS[@]}" 2> /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"
|
|
||||||
30
src/check.sh
Normal file
30
src/check.sh
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -Eeuo pipefail
|
||||||
|
|
||||||
|
[ -f "/run/shm/qemu.end" ] && echo "QEMU is shutting down.." && exit 1
|
||||||
|
[ ! -f "/run/shm/qemu.pid" ] && echo "QEMU is not running yet.." && exit 0
|
||||||
|
|
||||||
|
file="/run/shm/dsm.url"
|
||||||
|
address="/run/shm/qemu.ip"
|
||||||
|
|
||||||
|
[ ! -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="$(cat "$address")"
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "You might need to whitelist IP $ip in the DSM firewall." && exit 1
|
||||||
|
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "Healthcheck OK"
|
||||||
|
exit 0
|
||||||
15
src/config.sh
Normal file
15
src/config.sh
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -Eeuo pipefail
|
||||||
|
|
||||||
|
DEF_OPTS="-nodefaults -boot strict=on"
|
||||||
|
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}"
|
||||||
|
DEV_OPTS="-device virtio-balloon-pci,id=balloon0,bus=pcie.0,addr=0x4"
|
||||||
|
DEV_OPTS="$DEV_OPTS -object rng-random,id=objrng0,filename=/dev/urandom"
|
||||||
|
DEV_OPTS="$DEV_OPTS -device virtio-rng-pci,rng=objrng0,id=rng0,bus=pcie.0,addr=0x1c"
|
||||||
|
|
||||||
|
ARGS="$DEF_OPTS $CPU_OPTS $RAM_OPTS $MAC_OPTS $DISPLAY_OPTS $MON_OPTS $SERIAL_OPTS $NET_OPTS $DISK_OPTS $DEV_OPTS $ARGUMENTS"
|
||||||
|
ARGS=$(echo "$ARGS" | sed 's/\t/ /g' | tr -s ' ')
|
||||||
|
|
||||||
|
return 0
|
||||||
508
src/disk.sh
Normal file
508
src/disk.sh
Normal file
@@ -0,0 +1,508 @@
|
|||||||
|
#!/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_FLAGS:=''} # Specifies the options for use with the qcow2 disk format
|
||||||
|
: ${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
|
||||||
|
|
||||||
|
DISK_OPTS="\
|
||||||
|
-object iothread,id=io2 \
|
||||||
|
-device virtio-scsi-pci,id=hw-synoboot,iothread=io2,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=io2,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"; } || :
|
||||||
|
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
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
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
|
||||||
|
|
||||||
|
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 [[ "${FS,,}" == "fuse"* ]]; then
|
||||||
|
info "Warning: the filesystem of $DIR is FUSE, this extra layer will negatively affect performance!"
|
||||||
|
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 $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
|
||||||
|
}
|
||||||
|
|
||||||
|
addDevice () {
|
||||||
|
|
||||||
|
local DISK_ID=$1
|
||||||
|
local DISK_DEV=$2
|
||||||
|
local DISK_DESC=$3
|
||||||
|
local DISK_INDEX=$4
|
||||||
|
local DISK_ADDRESS=$5
|
||||||
|
|
||||||
|
[ -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
|
||||||
|
}
|
||||||
|
|
||||||
|
DISK_EXT="$(fmt2ext "$DISK_FMT")" || exit $?
|
||||||
|
|
||||||
|
if [ -z "$ALLOCATE" ]; then
|
||||||
|
if [[ "${DISK_FMT,,}" == "raw" ]]; then
|
||||||
|
ALLOCATE="Y"
|
||||||
|
else
|
||||||
|
ALLOCATE="N"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
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"
|
||||||
|
|
||||||
|
: ${DISK2_SIZE:=''}
|
||||||
|
: ${DISK3_SIZE:=''}
|
||||||
|
: ${DISK4_SIZE:=''}
|
||||||
|
|
||||||
|
: ${DEVICE:=''} # Docker variables to passthrough a block device, like /dev/vdc1.
|
||||||
|
: ${DEVICE2:=''}
|
||||||
|
: ${DEVICE3:=''}
|
||||||
|
: ${DEVICE4:=''}
|
||||||
|
|
||||||
|
if [ -n "$DEVICE" ]; then
|
||||||
|
addDevice "userdata" "$DEVICE" "device" "3" "0xc" || exit $?
|
||||||
|
else
|
||||||
|
addDisk "userdata" "$DISK1_FILE" "$DISK_EXT" "disk" "$DISK_SIZE" "3" "0xc" "$DISK_FMT" || exit $?
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -n "$DEVICE2" ]; then
|
||||||
|
addDevice "userdata2" "$DEVICE2" "device2" "4" "0xd" || exit $?
|
||||||
|
else
|
||||||
|
addDisk "userdata2" "$DISK2_FILE" "$DISK_EXT" "disk2" "$DISK2_SIZE" "4" "0xd" "$DISK_FMT" || exit $?
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -n "$DEVICE3" ]; then
|
||||||
|
addDevice "userdata3" "$DEVICE3" "device3" "5" "0xe" || exit $?
|
||||||
|
else
|
||||||
|
addDisk "userdata3" "$DISK3_FILE" "$DISK_EXT" "disk3" "$DISK3_SIZE" "5" "0xe" "$DISK_FMT" || exit $?
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -n "$DEVICE4" ]; then
|
||||||
|
addDevice "userdata4" "$DEVICE4" "device4" "6" "0xf" || exit $?
|
||||||
|
else
|
||||||
|
addDisk "userdata4" "$DISK4_FILE" "$DISK_EXT" "disk4" "$DISK4_SIZE" "6" "0xf" "$DISK_FMT" || exit $?
|
||||||
|
fi
|
||||||
|
|
||||||
|
return 0
|
||||||
35
src/display.sh
Normal file
35
src/display.sh
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -Eeuo pipefail
|
||||||
|
|
||||||
|
# Docker environment variables
|
||||||
|
|
||||||
|
: ${GPU:='N'} # GPU passthrough
|
||||||
|
: ${DISPLAY:='none'} # Display type
|
||||||
|
|
||||||
|
if [[ "$GPU" != [Yy1]* ]] || [[ "$ARCH" != "amd64" ]]; then
|
||||||
|
|
||||||
|
DISPLAY_OPTS="-display $DISPLAY -vga none"
|
||||||
|
return 0
|
||||||
|
|
||||||
|
fi
|
||||||
|
|
||||||
|
DISPLAY_OPTS="-display egl-headless,rendernode=/dev/dri/renderD128 -vga virtio"
|
||||||
|
|
||||||
|
[ ! -d /dev/dri ] && mkdir -m 755 /dev/dri
|
||||||
|
|
||||||
|
if [ ! -c /dev/dri/card0 ]; then
|
||||||
|
if mknod /dev/dri/card0 c 226 0; then
|
||||||
|
chmod 666 /dev/dri/card0
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ ! -c /dev/dri/renderD128 ]; then
|
||||||
|
if mknod /dev/dri/renderD128 c 226 128; then
|
||||||
|
chmod 666 /dev/dri/renderD128
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
addPackage "xserver-xorg-video-intel" "Intel GPU drivers"
|
||||||
|
addPackage "qemu-system-modules-opengl" "OpenGL module"
|
||||||
|
|
||||||
|
return 0
|
||||||
33
src/entry.sh
Executable file
33
src/entry.sh
Executable file
@@ -0,0 +1,33 @@
|
|||||||
|
#!/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
|
||||||
|
. display.sh # Initialize graphics
|
||||||
|
. network.sh # Initialize network
|
||||||
|
. proc.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 ${ARGS:+ $ARGS}
|
||||||
|
fi
|
||||||
|
|
||||||
|
[[ "$DEBUG" == [Yy1]* ]] && info "$VERS" && echo "Arguments: $ARGS" && echo
|
||||||
|
{ qemu-system-x86_64 ${ARGS:+ $ARGS} >"$QEMU_OUT" 2>"$QEMU_LOG"; rc=$?; } || :
|
||||||
|
(( rc != 0 )) && error "$(cat "$QEMU_LOG")" && exit 15
|
||||||
|
|
||||||
|
terminal
|
||||||
|
tail -fn +0 "$QEMU_LOG" 2>/dev/null &
|
||||||
|
cat "$QEMU_TERM" 2>/dev/null & wait $! || :
|
||||||
|
|
||||||
|
sleep 1 && finish 0
|
||||||
330
src/install.sh
Normal file
330
src/install.sh
Normal file
@@ -0,0 +1,330 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -Eeuo pipefail
|
||||||
|
|
||||||
|
: ${URL:=''} # URL of the PAT file to be downloaded.
|
||||||
|
|
||||||
|
if [ -f "$STORAGE/dsm.ver" ]; then
|
||||||
|
BASE=$(cat "$STORAGE/dsm.ver")
|
||||||
|
else
|
||||||
|
# Fallback for old installs
|
||||||
|
BASE="DSM_VirtualDSM_42962"
|
||||||
|
fi
|
||||||
|
|
||||||
|
[ -n "$URL" ] && BASE=$(basename "$URL" .pat)
|
||||||
|
|
||||||
|
if [[ -f "$STORAGE/$BASE.boot.img" ]] && [[ -f "$STORAGE/$BASE.system.img" ]]; then
|
||||||
|
return 0 # Previous installation found
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Display wait message
|
||||||
|
/run/server.sh 5000 install &
|
||||||
|
|
||||||
|
DL=""
|
||||||
|
DL_CHINA="https://cndl.synology.cn/download/DSM"
|
||||||
|
DL_GLOBAL="https://global.synologydownload.com/download/DSM"
|
||||||
|
|
||||||
|
[[ "${URL,,}" == *"cndl.synology"* ]] && DL="$DL_CHINA"
|
||||||
|
[[ "${URL,,}" == *"global.synology"* ]] && DL="$DL_GLOBAL"
|
||||||
|
|
||||||
|
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
|
||||||
|
|
||||||
|
[ -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
|
||||||
|
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,,}" == "fuse"* ]]; then
|
||||||
|
info "Warning: the filesystem of $STORAGE is FUSE, this extra layer will negatively affect performance!"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ "${FS,,}" != "fat"* && "${FS,,}" != "vfat"* && "${FS,,}" != "exfat"* && "${FS,,}" != "ntfs"* && "${FS,,}" != "msdos"* ]]; then
|
||||||
|
TMP="$STORAGE/tmp"
|
||||||
|
else
|
||||||
|
TMP="/tmp/dsm"
|
||||||
|
TMP_SPACE=2147483648
|
||||||
|
SPACE=$(df --output=avail -B 1 /tmp | tail -n 1)
|
||||||
|
SPACE_MB=$(( (SPACE + 1048575)/1048576 ))
|
||||||
|
if (( TMP_SPACE > SPACE )); then
|
||||||
|
error "Not enough free space inside the container, have $SPACE_MB MB available but need at least 2 GB." && exit 93
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
rm -rf "$TMP" && mkdir -p "$TMP"
|
||||||
|
|
||||||
|
# Check free diskspace
|
||||||
|
ROOT_SPACE=536870912
|
||||||
|
SPACE=$(df --output=avail -B 1 / | tail -n 1)
|
||||||
|
SPACE_MB=$(( (SPACE + 1048575)/1048576 ))
|
||||||
|
(( ROOT_SPACE > SPACE )) && error "Not enough free space inside the container, have $SPACE_MB MB available but need at least 500 MB." && exit 96
|
||||||
|
|
||||||
|
MIN_SPACE=8589934592
|
||||||
|
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 8 GB." && exit 94
|
||||||
|
|
||||||
|
# Check if output is to interactive TTY
|
||||||
|
if [ -t 1 ]; then
|
||||||
|
PROGRESS="--progress=bar:noscroll"
|
||||||
|
else
|
||||||
|
PROGRESS="--progress=dot:giga"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Download the required files from the Synology website
|
||||||
|
|
||||||
|
ROOT="Y"
|
||||||
|
RDC="$STORAGE/dsm.rd"
|
||||||
|
|
||||||
|
if [ ! -f "$RDC" ]; then
|
||||||
|
|
||||||
|
info "Install: Downloading installer..."
|
||||||
|
|
||||||
|
RD="$TMP/rd.gz"
|
||||||
|
POS="65627648-71021835"
|
||||||
|
VERIFY="b4215a4b213ff5154db0488f92c87864"
|
||||||
|
LOC="$DL/release/7.0.1/42218/DSM_VirtualDSM_42218.pat"
|
||||||
|
|
||||||
|
{ curl -r "$POS" -sfk -S -o "$RD" "$LOC"; rc=$?; } || :
|
||||||
|
(( rc != 0 )) && error "Failed to download $LOC, reason: $rc" && exit 60
|
||||||
|
|
||||||
|
SUM=$(md5sum "$RD" | cut -f 1 -d " ")
|
||||||
|
|
||||||
|
if [ "$SUM" != "$VERIFY" ]; then
|
||||||
|
|
||||||
|
PAT="/install.pat"
|
||||||
|
rm "$RD"
|
||||||
|
rm -f "$PAT"
|
||||||
|
|
||||||
|
{ wget "$LOC" -O "$PAT" -q --no-check-certificate --show-progress "$PROGRESS"; rc=$?; } || :
|
||||||
|
(( rc != 0 )) && error "Failed to download $LOC, reason: $rc" && exit 60
|
||||||
|
|
||||||
|
tar --extract --file="$PAT" --directory="$(dirname "$RD")"/. "$(basename "$RD")"
|
||||||
|
rm "$PAT"
|
||||||
|
|
||||||
|
fi
|
||||||
|
|
||||||
|
cp "$RD" "$RDC"
|
||||||
|
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -f "$RDC" ]; then
|
||||||
|
|
||||||
|
{ xz -dc <"$RDC" >"$TMP/rd" 2>/dev/null; rc=$?; } || :
|
||||||
|
(( rc != 1 )) && error "Failed to unxz $RDC, reason $rc" && exit 91
|
||||||
|
|
||||||
|
{ (cd "$TMP" && cpio -idm <"$TMP/rd" 2>/dev/null); rc=$?; } || :
|
||||||
|
|
||||||
|
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
|
||||||
|
|
||||||
|
rm -rf /run/extract && mkdir -p /run/extract
|
||||||
|
for file in $TMP/usr/lib/libcurl.so.4 \
|
||||||
|
$TMP/usr/lib/libmbedcrypto.so.5 \
|
||||||
|
$TMP/usr/lib/libmbedtls.so.13 \
|
||||||
|
$TMP/usr/lib/libmbedx509.so.1 \
|
||||||
|
$TMP/usr/lib/libmsgpackc.so.2 \
|
||||||
|
$TMP/usr/lib/libsodium.so \
|
||||||
|
$TMP/usr/lib/libsynocodesign-ng-virtual-junior-wins.so.7 \
|
||||||
|
$TMP/usr/syno/bin/scemd; do
|
||||||
|
cp "$file" /run/extract/
|
||||||
|
done
|
||||||
|
|
||||||
|
if [ "$ARCH" != "amd64" ]; then
|
||||||
|
mkdir -p /lib64/
|
||||||
|
cp "$TMP/usr/lib/libc.so.6" /lib64/
|
||||||
|
cp "$TMP/usr/lib/libpthread.so.0" /lib64/
|
||||||
|
cp "$TMP/usr/lib/ld-linux-x86-64.so.2" /lib64/
|
||||||
|
fi
|
||||||
|
|
||||||
|
mv /run/extract/scemd /run/extract/syno_extract_system_patch
|
||||||
|
chmod +x /run/extract/syno_extract_system_patch
|
||||||
|
|
||||||
|
fi
|
||||||
|
|
||||||
|
rm -rf "$TMP" && mkdir -p "$TMP"
|
||||||
|
|
||||||
|
info "Install: Downloading $(basename "$URL")..."
|
||||||
|
|
||||||
|
PAT="/$BASE.pat"
|
||||||
|
rm -f "$PAT"
|
||||||
|
|
||||||
|
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
|
||||||
|
|
||||||
|
[ ! -f "$PAT" ] && error "Failed to download $URL" && exit 69
|
||||||
|
|
||||||
|
SIZE=$(stat -c%s "$PAT")
|
||||||
|
|
||||||
|
if ((SIZE<250000000)); then
|
||||||
|
error "The specified PAT file is probably an update pack as it's too small." && exit 62
|
||||||
|
fi
|
||||||
|
|
||||||
|
info "Install: Extracting downloaded image..."
|
||||||
|
|
||||||
|
if { tar tf "$PAT"; } >/dev/null 2>&1; then
|
||||||
|
|
||||||
|
tar xpf "$PAT" -C "$TMP/."
|
||||||
|
|
||||||
|
else
|
||||||
|
|
||||||
|
export LD_LIBRARY_PATH="/run/extract"
|
||||||
|
|
||||||
|
if [ "$ARCH" == "amd64" ]; then
|
||||||
|
{ /run/extract/syno_extract_system_patch "$PAT" "$TMP/."; rc=$?; } || :
|
||||||
|
else
|
||||||
|
{ qemu-x86_64 /run/extract/syno_extract_system_patch "$PAT" "$TMP/."; rc=$?; } || :
|
||||||
|
fi
|
||||||
|
|
||||||
|
export LD_LIBRARY_PATH=""
|
||||||
|
|
||||||
|
(( rc != 0 )) && error "Failed to extract PAT file, reason $rc" && exit 63
|
||||||
|
|
||||||
|
fi
|
||||||
|
|
||||||
|
rm -rf /run/extract
|
||||||
|
info "Install: Preparing system partition..."
|
||||||
|
|
||||||
|
BOOT=$(find "$TMP" -name "*.bin.zip")
|
||||||
|
[ ! -f "$BOOT" ] && error "The PAT file contains no boot image." && exit 67
|
||||||
|
|
||||||
|
BOOT=$(echo "$BOOT" | head -c -5)
|
||||||
|
unzip -q -o "$BOOT".zip -d "$TMP"
|
||||||
|
|
||||||
|
SYSTEM="$STORAGE/$BASE.system.img"
|
||||||
|
rm -f "$SYSTEM"
|
||||||
|
|
||||||
|
# Check free diskspace
|
||||||
|
SYSTEM_SIZE=4954537983
|
||||||
|
SPACE=$(df --output=avail -B 1 "$STORAGE" | tail -n 1)
|
||||||
|
SPACE_MB=$(( (SPACE + 1048575)/1048576 ))
|
||||||
|
|
||||||
|
if (( SYSTEM_SIZE > SPACE )); then
|
||||||
|
error "Not enough free space in $STORAGE to create a 5 GB system disk, have only $SPACE_MB MB available." && exit 97
|
||||||
|
fi
|
||||||
|
|
||||||
|
if ! touch "$SYSTEM"; then
|
||||||
|
error "Could not create file $SYSTEM for the system disk." && exit 98
|
||||||
|
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
|
||||||
|
|
||||||
|
if ! fallocate -l "$SYSTEM_SIZE" "$SYSTEM"; then
|
||||||
|
if ! truncate -s "$SYSTEM_SIZE" "$SYSTEM"; then
|
||||||
|
rm -f "$SYSTEM"
|
||||||
|
error "Could not allocate file $SYSTEM for the system disk." && exit 98
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
PART="$TMP/partition.fdisk"
|
||||||
|
|
||||||
|
{ echo "label: dos"
|
||||||
|
echo "label-id: 0x6f9ee2e9"
|
||||||
|
echo "device: $SYSTEM"
|
||||||
|
echo "unit: sectors"
|
||||||
|
echo "sector-size: 512"
|
||||||
|
echo ""
|
||||||
|
echo "${SYSTEM}1 : start= 2048, size= 4980480, type=83"
|
||||||
|
echo "${SYSTEM}2 : start= 4982528, size= 4194304, type=82"
|
||||||
|
} > "$PART"
|
||||||
|
|
||||||
|
sfdisk -q "$SYSTEM" < "$PART"
|
||||||
|
|
||||||
|
MOUNT="$TMP/system"
|
||||||
|
rm -rf "$MOUNT" && mkdir -p "$MOUNT"
|
||||||
|
|
||||||
|
info "Install: Extracting system partition..."
|
||||||
|
|
||||||
|
HDA="$TMP/hda1"
|
||||||
|
IDB="$TMP/indexdb"
|
||||||
|
PKG="$TMP/packages"
|
||||||
|
HDP="$TMP/synohdpack_img"
|
||||||
|
|
||||||
|
[ ! -f "$HDA.tgz" ] && error "The PAT file contains no OS image." && exit 64
|
||||||
|
mv "$HDA.tgz" "$HDA.txz"
|
||||||
|
|
||||||
|
[ -d "$PKG" ] && mv "$PKG/" "$MOUNT/.SynoUpgradePackages/"
|
||||||
|
rm -f "$MOUNT/.SynoUpgradePackages/ActiveInsight-"*
|
||||||
|
|
||||||
|
[ -f "$HDP.txz" ] && tar xpfJ "$HDP.txz" --absolute-names -C "$MOUNT/"
|
||||||
|
|
||||||
|
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"
|
||||||
|
OFFSET="1048576" # 2048 * 512
|
||||||
|
NUMBLOCKS="622560" # (4980480 * 512) / 4096
|
||||||
|
|
||||||
|
if [[ "$ROOT" != [Nn]* ]]; then
|
||||||
|
|
||||||
|
tar xpfJ "$HDA.txz" --absolute-names --skip-old-files -C "$MOUNT/"
|
||||||
|
|
||||||
|
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"
|
||||||
|
|
||||||
|
echo "$BASE" > "$STORAGE/dsm.ver"
|
||||||
|
|
||||||
|
if [[ "$URL" == "file://$STORAGE/$BASE.pat" ]]; then
|
||||||
|
rm -f "$PAT"
|
||||||
|
else
|
||||||
|
mv -f "$PAT" "$STORAGE/$BASE.pat"
|
||||||
|
fi
|
||||||
|
|
||||||
|
mv -f "$BOOT" "$STORAGE/$BASE.boot.img"
|
||||||
|
|
||||||
|
rm -rf "$TMP"
|
||||||
|
|
||||||
|
{ set +x; } 2>/dev/null
|
||||||
|
[[ "$DEBUG" == [Yy1]* ]] && echo
|
||||||
|
|
||||||
|
return 0
|
||||||
267
src/network.sh
Normal file
267
src/network.sh
Normal file
@@ -0,0 +1,267 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -Eeuo pipefail
|
||||||
|
|
||||||
|
# Docker environment variables
|
||||||
|
|
||||||
|
: ${DHCP:='N'}
|
||||||
|
: ${MAC:='02:11:32:AA:BB:CC'}
|
||||||
|
|
||||||
|
: ${VM_NET_DEV:=''}
|
||||||
|
: ${VM_NET_TAP:='dsm'}
|
||||||
|
: ${VM_NET_MAC:="$MAC"}
|
||||||
|
: ${VM_NET_HOST:='VirtualDSM'}
|
||||||
|
|
||||||
|
: ${DNSMASQ_OPTS:=''}
|
||||||
|
: ${DNSMASQ:='/usr/sbin/dnsmasq'}
|
||||||
|
: ${DNSMASQ_CONF_DIR:='/etc/dnsmasq.d'}
|
||||||
|
|
||||||
|
ADD_ERR="Please add the following setting to your container:"
|
||||||
|
|
||||||
|
# ######################################
|
||||||
|
# Functions
|
||||||
|
# ######################################
|
||||||
|
|
||||||
|
configureDHCP() {
|
||||||
|
|
||||||
|
# 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=$?; } || :
|
||||||
|
|
||||||
|
if (( rc != 0 )); then
|
||||||
|
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: --cap-add NET_ADMIN" && exit 16
|
||||||
|
fi
|
||||||
|
|
||||||
|
while ! ip link set "$VM_NET_TAP" up; do
|
||||||
|
info "Waiting for address to become available..."
|
||||||
|
sleep 2
|
||||||
|
done
|
||||||
|
|
||||||
|
local TAP_NR TAP_PATH MAJOR MINOR
|
||||||
|
TAP_NR=$(</sys/class/net/"$VM_NET_TAP"/ifindex)
|
||||||
|
TAP_PATH="/dev/tap${TAP_NR}"
|
||||||
|
|
||||||
|
# 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)
|
||||||
|
(( 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"
|
||||||
|
|
||||||
|
if [[ ! -e "$TAP_PATH" ]]; then
|
||||||
|
{ mknod "$TAP_PATH" c "$MAJOR" "$MINOR" ; rc=$?; } || :
|
||||||
|
(( rc != 0 )) && error "Cannot mknod: $TAP_PATH ($rc)" && exit 20
|
||||||
|
fi
|
||||||
|
|
||||||
|
{ exec 30>>"$TAP_PATH"; rc=$?; } 2>/dev/null || :
|
||||||
|
|
||||||
|
if (( rc != 0 )); then
|
||||||
|
error "Cannot create TAP interface ($rc). $ADD_ERR --device-cgroup-rule='c *:* rwm'" && exit 21
|
||||||
|
fi
|
||||||
|
|
||||||
|
{ exec 40>>/dev/vhost-net; rc=$?; } 2>/dev/null || :
|
||||||
|
|
||||||
|
if (( rc != 0 )); then
|
||||||
|
error "VHOST can not be found ($rc). $ADD_ERR --device=/dev/vhost-net" && exit 22
|
||||||
|
fi
|
||||||
|
|
||||||
|
NET_OPTS="-netdev tap,id=hostnet0,vhost=on,vhostfd=40,fd=30"
|
||||||
|
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
configureDNS() {
|
||||||
|
|
||||||
|
# dnsmasq configuration:
|
||||||
|
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
|
||||||
|
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
|
||||||
|
|
||||||
|
# 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"
|
||||||
|
DNSMASQ_OPTS=$(echo "$DNSMASQ_OPTS" | sed 's/\t/ /g' | tr -s ' ' | sed 's/^ *//')
|
||||||
|
|
||||||
|
[[ "$DEBUG" == [Yy1]* ]] && set -x
|
||||||
|
$DNSMASQ ${DNSMASQ_OPTS:+ $DNSMASQ_OPTS}
|
||||||
|
{ set +x; } 2>/dev/null
|
||||||
|
[[ "$DEBUG" == [Yy1]* ]] && echo
|
||||||
|
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
configureNAT() {
|
||||||
|
|
||||||
|
# Create the necessary file structure for /dev/net/tun
|
||||||
|
if [ ! -c /dev/net/tun ]; then
|
||||||
|
[ ! -d /dev/net ] && mkdir -m 755 /dev/net
|
||||||
|
if mknod /dev/net/tun c 10 200; then
|
||||||
|
chmod 666 /dev/net/tun
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ ! -c /dev/net/tun ]; then
|
||||||
|
error "TUN device missing. $ADD_ERR --cap-add NET_ADMIN" && exit 25
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check port forwarding flag
|
||||||
|
if [[ $(< /proc/sys/net/ipv4/ip_forward) -eq 0 ]]; then
|
||||||
|
{ sysctl -w net.ipv4.ip_forward=1 ; rc=$?; } || :
|
||||||
|
if (( rc != 0 )); then
|
||||||
|
error "IP forwarding is disabled. $ADD_ERR --sysctl net.ipv4.ip_forward=1" && exit 24
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Create a bridge with a static IP for the VM guest
|
||||||
|
|
||||||
|
VM_NET_IP='20.20.20.21'
|
||||||
|
[[ "$DEBUG" == [Yy1]* ]] && set -x
|
||||||
|
|
||||||
|
{ ip link add dev dockerbridge type bridge ; rc=$?; } || :
|
||||||
|
|
||||||
|
if (( rc != 0 )); then
|
||||||
|
error "Failed to create bridge. $ADD_ERR --cap-add NET_ADMIN" && exit 23
|
||||||
|
fi
|
||||||
|
|
||||||
|
ip address add ${VM_NET_IP%.*}.1/24 broadcast ${VM_NET_IP%.*}.255 dev dockerbridge
|
||||||
|
|
||||||
|
while ! ip link set dockerbridge up; do
|
||||||
|
info "Waiting for address to become available..."
|
||||||
|
sleep 2
|
||||||
|
done
|
||||||
|
|
||||||
|
# QEMU Works with taps, set tap to the bridge created
|
||||||
|
ip tuntap add dev "$VM_NET_TAP" mode tap
|
||||||
|
|
||||||
|
while ! ip link set "$VM_NET_TAP" up promisc on; do
|
||||||
|
info "Waiting for tap to become available..."
|
||||||
|
sleep 2
|
||||||
|
done
|
||||||
|
|
||||||
|
ip link set dev "$VM_NET_TAP" master dockerbridge
|
||||||
|
|
||||||
|
# Add internet connection to the VM
|
||||||
|
update-alternatives --set iptables /usr/sbin/iptables-legacy > /dev/null
|
||||||
|
update-alternatives --set ip6tables /usr/sbin/ip6tables-legacy > /dev/null
|
||||||
|
|
||||||
|
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 PREROUTING -i "$VM_NET_DEV" -d "$IP" -p udp -j DNAT --to "$VM_NET_IP"
|
||||||
|
|
||||||
|
if (( KERNEL > 4 )); then
|
||||||
|
# Hack for guest VMs complaining about "bad udp checksums in 5 packets"
|
||||||
|
iptables -A POSTROUTING -t mangle -p udp --dport bootpc -j CHECKSUM --checksum-fill || true
|
||||||
|
fi
|
||||||
|
|
||||||
|
{ set +x; } 2>/dev/null
|
||||||
|
[[ "$DEBUG" == [Yy1]* ]] && echo
|
||||||
|
|
||||||
|
NET_OPTS="-netdev tap,ifname=$VM_NET_TAP,script=no,downscript=no,id=hostnet0"
|
||||||
|
|
||||||
|
{ exec 40>>/dev/vhost-net; rc=$?; } 2>/dev/null || :
|
||||||
|
(( rc == 0 )) && NET_OPTS="$NET_OPTS,vhost=on,vhostfd=40"
|
||||||
|
|
||||||
|
configureDNS
|
||||||
|
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
closeNetwork() {
|
||||||
|
|
||||||
|
exec 30<&- || true
|
||||||
|
exec 40<&- || true
|
||||||
|
|
||||||
|
if [[ "$DHCP" == [Yy1]* ]]; then
|
||||||
|
|
||||||
|
fKill "server.sh"
|
||||||
|
|
||||||
|
ip link set "$VM_NET_TAP" down || true
|
||||||
|
ip link delete "$VM_NET_TAP" || true
|
||||||
|
|
||||||
|
else
|
||||||
|
|
||||||
|
fKill "dnsmasq"
|
||||||
|
|
||||||
|
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 delete dockerbridge || true
|
||||||
|
|
||||||
|
fi
|
||||||
|
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
getInfo() {
|
||||||
|
|
||||||
|
if [ -z "$VM_NET_DEV" ]; then
|
||||||
|
# Automaticly detect the default network interface
|
||||||
|
VM_NET_DEV=$(awk '$2 == 00000000 { print $1 }' /proc/net/route)
|
||||||
|
[ -z "$VM_NET_DEV" ] && VM_NET_DEV="eth0"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ ! -d "/sys/class/net/$VM_NET_DEV" ]; then
|
||||||
|
error "Network interface '$VM_NET_DEV' does not exist inside the container!"
|
||||||
|
error "$ADD_ERR -e \"VM_NET_DEV=NAME\" to specify another interface name." && exit 27
|
||||||
|
fi
|
||||||
|
|
||||||
|
VM_NET_MAC="${VM_NET_MAC//-/:}"
|
||||||
|
if [[ ${#VM_NET_MAC} == 12 ]]; then
|
||||||
|
m="$VM_NET_MAC"
|
||||||
|
VM_NET_MAC="${m:0:2}:${m:2:2}:${m:4:2}:${m:6:2}:${m:8:2}:${m:10:2}"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ ${#VM_NET_MAC} != 17 ]]; then
|
||||||
|
error "Invalid mac address: '$VM_NET_MAC', should be 12 or 17 digits long!" && exit 28
|
||||||
|
fi
|
||||||
|
|
||||||
|
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/)
|
||||||
|
echo "$IP" > /run/shm/qemu.ip
|
||||||
|
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
# ######################################
|
||||||
|
# Configure Network
|
||||||
|
# ######################################
|
||||||
|
|
||||||
|
fKill "server.sh"
|
||||||
|
|
||||||
|
if [ ! -c /dev/vhost-net ]; then
|
||||||
|
if mknod /dev/vhost-net c 10 238; then
|
||||||
|
chmod 660 /dev/vhost-net
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
getInfo
|
||||||
|
|
||||||
|
if [[ "$DEBUG" == [Yy1]* ]]; then
|
||||||
|
info "Container IP is $IP with gateway $GATEWAY on interface $VM_NET_DEV" && echo
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ "$DHCP" == [Yy1]* ]]; then
|
||||||
|
|
||||||
|
if [[ "$GATEWAY" == "172."* ]]; then
|
||||||
|
if [[ "$DEBUG" != [Yy1]* ]]; then
|
||||||
|
error "You can only enable DHCP while the container is on a macvlan network!" && exit 26
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Configuration for DHCP IP
|
||||||
|
configureDHCP
|
||||||
|
|
||||||
|
# Display IP on port 80 and 5000
|
||||||
|
/run/server.sh 5000 /run/ip.sh &
|
||||||
|
|
||||||
|
else
|
||||||
|
|
||||||
|
# Configuration for static IP
|
||||||
|
configureNAT
|
||||||
|
|
||||||
|
fi
|
||||||
|
|
||||||
|
NET_OPTS="$NET_OPTS -device virtio-net-pci,romfile=,netdev=hostnet0,mac=$VM_NET_MAC,id=net0"
|
||||||
|
|
||||||
|
return 0
|
||||||
180
src/power.sh
Normal file
180
src/power.sh
Normal file
@@ -0,0 +1,180 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -Eeuo pipefail
|
||||||
|
|
||||||
|
# Configure QEMU for graceful shutdown
|
||||||
|
|
||||||
|
API_CMD=6
|
||||||
|
API_TIMEOUT=50
|
||||||
|
API_HOST="127.0.0.1:2210"
|
||||||
|
|
||||||
|
QEMU_TERM=""
|
||||||
|
QEMU_PORT=7100
|
||||||
|
QEMU_TIMEOUT=50
|
||||||
|
QEMU_PID="/run/shm/qemu.pid"
|
||||||
|
QEMU_LOG="/run/shm/qemu.log"
|
||||||
|
QEMU_OUT="/run/shm/qemu.out"
|
||||||
|
QEMU_END="/run/shm/qemu.end"
|
||||||
|
|
||||||
|
if [[ "$KVM" == [Nn]* ]]; then
|
||||||
|
API_TIMEOUT=$(( API_TIMEOUT*2 ))
|
||||||
|
QEMU_TIMEOUT=$(( QEMU_TIMEOUT*2 ))
|
||||||
|
fi
|
||||||
|
|
||||||
|
touch "$QEMU_LOG"
|
||||||
|
|
||||||
|
_trap() {
|
||||||
|
func="$1" ; shift
|
||||||
|
for sig ; do
|
||||||
|
trap "$func $sig" "$sig"
|
||||||
|
done
|
||||||
|
}
|
||||||
|
|
||||||
|
finish() {
|
||||||
|
|
||||||
|
local pid
|
||||||
|
local reason=$1
|
||||||
|
|
||||||
|
if [ -f "$QEMU_PID" ]; then
|
||||||
|
|
||||||
|
pid="$(cat "$QEMU_PID")"
|
||||||
|
echo && error "Forcefully terminating QEMU process, reason: $reason..."
|
||||||
|
{ kill -15 "$pid" || true; } 2>/dev/null
|
||||||
|
|
||||||
|
while isAlive "$pid"; do
|
||||||
|
sleep 1
|
||||||
|
# Workaround for zombie pid
|
||||||
|
[ ! -f "$QEMU_PID" ] && break
|
||||||
|
done
|
||||||
|
fi
|
||||||
|
|
||||||
|
fKill "print.sh"
|
||||||
|
fKill "host.bin"
|
||||||
|
|
||||||
|
closeNetwork
|
||||||
|
|
||||||
|
sleep 1
|
||||||
|
echo && echo "❯ Shutdown completed!"
|
||||||
|
|
||||||
|
exit "$reason"
|
||||||
|
}
|
||||||
|
|
||||||
|
terminal() {
|
||||||
|
|
||||||
|
local dev=""
|
||||||
|
|
||||||
|
if [ -f "$QEMU_OUT" ]; then
|
||||||
|
|
||||||
|
local msg
|
||||||
|
msg="$(cat "$QEMU_OUT")"
|
||||||
|
|
||||||
|
if [ -n "$msg" ]; then
|
||||||
|
|
||||||
|
if [[ "${msg,,}" != "char"* || "$msg" != *"serial0)" ]]; then
|
||||||
|
echo "$msg"
|
||||||
|
fi
|
||||||
|
|
||||||
|
dev="${msg#*/dev/p}"
|
||||||
|
dev="/dev/p${dev%% *}"
|
||||||
|
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ ! -c "$dev" ]; then
|
||||||
|
dev=$(echo 'info chardev' | nc -q 1 -w 1 localhost "$QEMU_PORT" | tr -d '\000')
|
||||||
|
dev="${dev#*serial0}"
|
||||||
|
dev="${dev#*pty:}"
|
||||||
|
dev="${dev%%$'\n'*}"
|
||||||
|
dev="${dev%%$'\r'*}"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ ! -c "$dev" ]; then
|
||||||
|
error "Device '$dev' not found!"
|
||||||
|
finish 34 && return 34
|
||||||
|
fi
|
||||||
|
|
||||||
|
QEMU_TERM="$dev"
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
_graceful_shutdown() {
|
||||||
|
|
||||||
|
local cnt=0
|
||||||
|
local code=$?
|
||||||
|
local pid url response
|
||||||
|
|
||||||
|
set +e
|
||||||
|
|
||||||
|
if [ -f "$QEMU_END" ]; then
|
||||||
|
echo && info "Received $1 signal while already shutting down..."
|
||||||
|
return
|
||||||
|
fi
|
||||||
|
|
||||||
|
touch "$QEMU_END"
|
||||||
|
echo && info "Received $1 signal, sending shutdown command..."
|
||||||
|
|
||||||
|
if [ ! -f "$QEMU_PID" ]; then
|
||||||
|
echo && error "QEMU PID file does not exist?"
|
||||||
|
finish "$code" && return "$code"
|
||||||
|
fi
|
||||||
|
|
||||||
|
pid="$(cat "$QEMU_PID")"
|
||||||
|
|
||||||
|
if ! isAlive "$pid"; then
|
||||||
|
echo && error "QEMU process does not exist?"
|
||||||
|
finish "$code" && return "$code"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# 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://$API_HOST/read?command=$API_CMD&timeout=$API_TIMEOUT"
|
||||||
|
response=$(curl -sk -m "$(( API_TIMEOUT+2 ))" -S "$url" 2>&1)
|
||||||
|
|
||||||
|
if [[ "$response" =~ "\"success\"" ]]; then
|
||||||
|
|
||||||
|
echo && info "Virtual DSM is now ready to shutdown..."
|
||||||
|
|
||||||
|
else
|
||||||
|
|
||||||
|
response="${response#*message\"\: \"}"
|
||||||
|
[ -z "$response" ] && response="second signal"
|
||||||
|
echo && error "Forcefully terminating because of: ${response%%\"*}"
|
||||||
|
{ kill -15 "$pid" || true; } 2>/dev/null
|
||||||
|
|
||||||
|
fi
|
||||||
|
|
||||||
|
while [ "$cnt" -lt "$QEMU_TIMEOUT" ]; do
|
||||||
|
|
||||||
|
! isAlive "$pid" && break
|
||||||
|
|
||||||
|
sleep 1
|
||||||
|
cnt=$((cnt+1))
|
||||||
|
|
||||||
|
[[ "$DEBUG" == [Yy1]* ]] && info "Shutting down, waiting... ($cnt/$QEMU_TIMEOUT)"
|
||||||
|
|
||||||
|
# Workaround for zombie pid
|
||||||
|
[ ! -f "$QEMU_PID" ] && break
|
||||||
|
|
||||||
|
done
|
||||||
|
|
||||||
|
if [ "$cnt" -ge "$QEMU_TIMEOUT" ]; then
|
||||||
|
echo && error "Shutdown timeout reached, aborting..."
|
||||||
|
fi
|
||||||
|
|
||||||
|
finish "$code" && return "$code"
|
||||||
|
}
|
||||||
|
|
||||||
|
MON_OPTS="\
|
||||||
|
-pidfile $QEMU_PID \
|
||||||
|
-monitor telnet:localhost:$QEMU_PORT,server,nowait,nodelay"
|
||||||
|
|
||||||
|
if [[ "$CONSOLE" != [Yy]* ]]; then
|
||||||
|
|
||||||
|
MON_OPTS="$MON_OPTS -daemonize -D $QEMU_LOG"
|
||||||
|
|
||||||
|
_trap _graceful_shutdown SIGTERM SIGHUP SIGINT SIGABRT SIGQUIT
|
||||||
|
|
||||||
|
fi
|
||||||
|
|
||||||
|
return 0
|
||||||
87
src/print.sh
Normal file
87
src/print.sh
Normal file
@@ -0,0 +1,87 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -Eeuo pipefail
|
||||||
|
|
||||||
|
: ${DHCP:='N'}
|
||||||
|
|
||||||
|
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/shm/dsm.url"
|
||||||
|
address="/run/shm/qemu.ip"
|
||||||
|
shutdown="/run/shm/qemu.end"
|
||||||
|
url="http://127.0.0.1:2210/read?command=10"
|
||||||
|
|
||||||
|
resp_err="Guest returned an invalid response:"
|
||||||
|
curl_err="Failed to connect to guest: curl error"
|
||||||
|
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 "$curl_err $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="$(cat "$address")"
|
||||||
|
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
|
||||||
77
src/proc.sh
Normal file
77
src/proc.sh
Normal file
@@ -0,0 +1,77 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -Eeuo pipefail
|
||||||
|
|
||||||
|
# Docker environment variables
|
||||||
|
|
||||||
|
: ${KVM:='Y'}
|
||||||
|
: ${HOST_CPU:=''}
|
||||||
|
: ${CPU_MODEL:='host'}
|
||||||
|
: ${CPU_FEATURES:='+ssse3,+sse4.1,+sse4.2'}
|
||||||
|
|
||||||
|
[ "$ARCH" != "amd64" ] && KVM="N"
|
||||||
|
|
||||||
|
if [[ "$KVM" != [Nn]* ]]; then
|
||||||
|
|
||||||
|
KVM_ERR=""
|
||||||
|
|
||||||
|
if [ ! -e /dev/kvm ]; then
|
||||||
|
KVM_ERR="(device file missing)"
|
||||||
|
else
|
||||||
|
if ! sh -c 'echo -n > /dev/kvm' &> /dev/null; then
|
||||||
|
KVM_ERR="(no write access)"
|
||||||
|
else
|
||||||
|
if ! grep -q -e vmx -e svm /proc/cpuinfo; then
|
||||||
|
KVM_ERR="(vmx/svm disabled)"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -n "$KVM_ERR" ]; then
|
||||||
|
KVM="N"
|
||||||
|
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 continue without KVM by setting KVM=N (not recommended)."
|
||||||
|
[[ "$DEBUG" != [Yy1]* ]] && exit 88
|
||||||
|
fi
|
||||||
|
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ "$KVM" != [Nn]* ]]; then
|
||||||
|
|
||||||
|
KVM_OPTS=",accel=kvm -enable-kvm"
|
||||||
|
|
||||||
|
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
|
||||||
|
|
||||||
|
else
|
||||||
|
|
||||||
|
KVM_OPTS=""
|
||||||
|
|
||||||
|
if [[ "$CPU_MODEL" == "host"* ]]; then
|
||||||
|
if [[ "$ARCH" == "amd64" ]]; then
|
||||||
|
CPU_MODEL="max,$CPU_FEATURES"
|
||||||
|
else
|
||||||
|
CPU_MODEL="qemu64,$CPU_FEATURES"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
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
|
||||||
150
src/reset.sh
Normal file
150
src/reset.sh
Normal file
@@ -0,0 +1,150 @@
|
|||||||
|
#!/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
|
||||||
|
: ${DEBUG:='N'} # Disable debugging mode
|
||||||
|
: ${COUNTRY:=''} # Country code for mirror
|
||||||
|
: ${CONSOLE:='N'} # Disable console mode
|
||||||
|
: ${ALLOCATE:=''} # 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"
|
||||||
|
if [ ! -d "$STORAGE" ]; then
|
||||||
|
error "Storage folder ($STORAGE) not found!" && exit 13
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ ! -d "/run/shm" ]; then
|
||||||
|
if [ -d "/dev/shm" ]; then
|
||||||
|
ln -s /dev/shm /run/shm
|
||||||
|
else
|
||||||
|
error "Folder /dev/shm not found!" && exit 14
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Cleanup files
|
||||||
|
rm -f /tmp/server.*
|
||||||
|
rm -f /run/shm/qemu.*
|
||||||
|
rm -f /run/shm/dsm.url
|
||||||
|
|
||||||
|
# Cleanup dirs
|
||||||
|
rm -rf /tmp/dsm
|
||||||
|
rm -rf "$STORAGE/tmp"
|
||||||
|
|
||||||
|
# Helper functions
|
||||||
|
|
||||||
|
isAlive() {
|
||||||
|
local pid=$1
|
||||||
|
|
||||||
|
if kill -0 "$pid" 2>/dev/null; then
|
||||||
|
return 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
return 1
|
||||||
|
}
|
||||||
|
|
||||||
|
pKill() {
|
||||||
|
local pid=$1
|
||||||
|
|
||||||
|
{ kill -15 "$pid" || true; } 2>/dev/null
|
||||||
|
|
||||||
|
while isAlive "$pid"; do
|
||||||
|
sleep 0.1
|
||||||
|
done
|
||||||
|
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
fKill() {
|
||||||
|
local name=$1
|
||||||
|
|
||||||
|
{ pkill -f "$name" || true; } 2>/dev/null
|
||||||
|
|
||||||
|
while pgrep -f -l "$name" >/dev/null; do
|
||||||
|
sleep 0.1
|
||||||
|
done
|
||||||
|
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
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://api.ip2location.io" ".country_code"
|
||||||
|
[ -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..."
|
||||||
|
|
||||||
|
[ -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
|
||||||
|
|
||||||
|
DEBIAN_FRONTEND=noninteractive apt-get -qq update
|
||||||
|
DEBIAN_FRONTEND=noninteractive apt-get -qq --no-install-recommends -y install "$pkg" > /dev/null
|
||||||
|
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0
|
||||||
60
src/serial.sh
Normal file
60
src/serial.sh
Normal file
@@ -0,0 +1,60 @@
|
|||||||
|
#!/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
|
||||||
|
|
||||||
|
if [[ "$CONSOLE" != [Yy]* ]]; then
|
||||||
|
SERIAL_OPTS="-serial pty"
|
||||||
|
else
|
||||||
|
SERIAL_OPTS="-serial mon:stdio"
|
||||||
|
fi
|
||||||
|
|
||||||
|
SERIAL_OPTS="$SERIAL_OPTS \
|
||||||
|
-device virtio-serial-pci,id=virtio-serial0,bus=pcie.0,addr=0x3 \
|
||||||
|
-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,11 +33,11 @@ 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
|
||||||
|
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if [[ "$2" != "/run/ip.sh" ]]; then
|
if [[ "$2" != "/run/ip.sh" ]]; then
|
||||||
@@ -46,23 +46,21 @@ if [[ "$2" != "/run/ip.sh" ]]; then
|
|||||||
|
|
||||||
else
|
else
|
||||||
|
|
||||||
BODY="The location of DSM is <a href='http://\${IP}:\${PORT}'>http://\${IP}:\${PORT}</a><script>"
|
BODY="The location of DSM is <a href='http://\$LOCATION'>http://\$LOCATION</a><script>"
|
||||||
BODY="${BODY}setTimeout(function(){ window.location.assign('http://\${IP}:\${PORT}'); }, 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 "INFO=\$(curl -s -m 2 -S http://127.0.0.1:2210/read?command=10 2>/dev/null)"
|
echo "[ -f \"/run/shm/dsm.url\" ] && LOCATION=\$(cat \"/run/shm/dsm.url\")"
|
||||||
echo "rest=\${INFO#*http_port}; rest=\${rest#*:}; rest=\${rest%%,*}; PORT=\${rest%%\\\"*}"
|
echo "HTML=\"$HTML\"; [ -z \"\$LOCATION\" ] && BODY=\"$WAIT\" || BODY=\"$BODY\"; HTML=\${HTML/xxx/\$BODY}"
|
||||||
echo "rest=\${INFO#*eth0}; rest=\${rest#*ip}; rest=\${rest#*:}; rest=\${rest#*\\\"}; IP=\${rest%%\\\"*}"
|
|
||||||
echo "HTML=\"$HTML\"; [ -z \"\${IP}\" ] && 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"
|
||||||
|
|
||||||
fi
|
fi
|
||||||
|
|
||||||
chmod +x "$TMP_FILE"
|
chmod +x "$TMP_FILE"
|
||||||
|
|
||||||
socat TCP4-LISTEN:80,reuseaddr,fork,crlf SYSTEM:"$TMP_FILE" 2> /dev/null &
|
socat TCP4-LISTEN:80,reuseaddr,fork,crlf SYSTEM:"$TMP_FILE" 2> /dev/null &
|
||||||
socat TCP4-LISTEN:"${1:-5000}",reuseaddr,fork,crlf SYSTEM:"$TMP_FILE" 2> /dev/null & wait $!
|
socat TCP4-LISTEN:"${1:-5000}",reuseaddr,fork,crlf SYSTEM:"$TMP_FILE" 2> /dev/null & wait $!
|
||||||
Reference in New Issue
Block a user