mirror of
https://github.com/factoriotools/factorio-docker.git
synced 2025-10-24 06:39:06 +02:00
Compare commits
516 Commits
0.18.21
...
fix/addgro
Author | SHA1 | Date | |
---|---|---|---|
|
35221ac441 | ||
|
19ca8d8ceb | ||
|
15bc6ef848 | ||
|
bca045c3d1 | ||
|
b0c8562021 | ||
|
2fe6a0b3c7 | ||
|
b8cd293fdd | ||
|
bd774f0ad8 | ||
|
b3099102cf | ||
|
e8adbf55c1 | ||
|
533789470f | ||
|
5b6e0cde8b | ||
|
00038b5184 | ||
|
72c3590cd6 | ||
|
23942e3117 | ||
|
15d38ea739 | ||
|
9464758a7b | ||
|
50f04fb096 | ||
|
15d31c9a2e | ||
|
8784845385 | ||
|
9f6c781331 | ||
|
8a718705b7 | ||
|
60277e89f1 | ||
|
4562f675ea | ||
|
22d02c42fa | ||
|
6b3dd77a54 | ||
|
6d3be9aef1 | ||
|
bf97066b9c | ||
|
c0c235a28d | ||
|
d5db1b1281 | ||
|
0c0349b5d6 | ||
|
0c1b2f4164 | ||
|
b36cd87194 | ||
|
e82cead5f4 | ||
|
d9ff25fa10 | ||
|
b7fdff9b94 | ||
|
04f20de96c | ||
|
fcd2c2e40e | ||
|
f02c02f38e | ||
|
b4be6993c6 | ||
|
a2c11ade50 | ||
|
b334d27cf0 | ||
|
ede09dc402 | ||
|
0929686ad3 | ||
|
317d3731e8 | ||
|
080a70f6be | ||
|
83bffe4b13 | ||
|
14df05cd6d | ||
|
c0e1bae277 | ||
|
370059f2f6 | ||
|
7e8c89c02a | ||
|
1e52dc48c9 | ||
|
14aa945717 | ||
|
57bbf46196 | ||
|
090e771c0c | ||
|
9e89a7930e | ||
|
20e6176a85 | ||
|
e5e1993c97 | ||
|
6e0ea49985 | ||
|
f757e538b7 | ||
|
a94f8a0d4e | ||
|
2b76185f94 | ||
|
3c45811e15 | ||
|
8990021fa1 | ||
|
50e5c868e5 | ||
|
89aebaba4b | ||
|
d19c97c075 | ||
|
19e5dc7532 | ||
|
6a3b160d76 | ||
|
a3d5456e58 | ||
|
8d5096e28b | ||
|
a1135b6a55 | ||
|
192f686b6b | ||
|
66ce43c0ca | ||
|
ebadd1d8b8 | ||
|
c9989d3a07 | ||
|
2dfaac7ec4 | ||
|
c2d7c5696e | ||
|
f14bedbc63 | ||
|
d91c45fa62 | ||
|
2092e86aec | ||
|
ecb89b0c11 | ||
|
42a08a95d4 | ||
|
61d52eaa7c | ||
|
6916fcb150 | ||
|
8c9dfd004e | ||
|
867635db16 | ||
|
b7413607c4 | ||
|
5d6f70b285 | ||
|
a24e42b281 | ||
|
d1142e3408 | ||
|
4eb53b6e48 | ||
|
f986f1c870 | ||
|
bc760a9599 | ||
|
d153e30f1c | ||
|
cc5061d339 | ||
|
14ae422463 | ||
|
88ca95a1ae | ||
|
64980b9db3 | ||
|
85f8011105 | ||
|
cc8d89df84 | ||
|
956ade4f0b | ||
|
049d34ea9a | ||
|
d01580b5a5 | ||
|
25a283ad79 | ||
|
b172b16ff4 | ||
|
eb150cff0d | ||
|
b20d93e229 | ||
|
9562212254 | ||
|
6060c7b722 | ||
|
06baf6a186 | ||
|
6eb0cd96d8 | ||
|
bbbfef343c | ||
|
44a58b056c | ||
|
e37bcf1ed0 | ||
|
3defd74c46 | ||
|
dde2761a8c | ||
|
0ec91c3391 | ||
|
f9040c7450 | ||
|
c9957b7d76 | ||
|
89ae20012e | ||
|
d137f32c2f | ||
|
4f3c5f2e93 | ||
|
f7e9dea263 | ||
|
748dec4ad5 | ||
|
3ce89a714b | ||
|
a8ecb01f4c | ||
|
6692bb514d | ||
|
2d805feb4f | ||
|
bc2dc44cad | ||
|
04085ef23d | ||
|
fbb787e535 | ||
|
2ba59b4f95 | ||
|
c269289a96 | ||
|
ee0388e163 | ||
|
a1cb15b709 | ||
|
890489fb15 | ||
|
c756ffe827 | ||
|
4474b52473 | ||
|
84a808102f | ||
|
016c5cf3b5 | ||
|
159b14bd8e | ||
|
c1b0f6ca83 | ||
|
965fef7ac2 | ||
|
88fed07073 | ||
|
4ab4c889f4 | ||
|
bc28f4a917 | ||
|
d792589c56 | ||
|
486c0f1d0a | ||
|
80d8761520 | ||
|
f4e0e640c5 | ||
|
51b601b8b6 | ||
|
1d88d18b38 | ||
|
7d085d7fe5 | ||
|
630c5a1864 | ||
|
98d3e475d3 | ||
|
bd76f1c541 | ||
|
afde97243d | ||
|
81b476ac97 | ||
|
30e56a197c | ||
|
effd07f84c | ||
|
23d40f36de | ||
|
250a90de53 | ||
|
89f601e5ce | ||
|
73fe111bea | ||
|
18b455ba43 | ||
|
1c5e98f289 | ||
|
0a3d923c5b | ||
|
19f52a6ce0 | ||
|
f6461d99be | ||
|
f5a9224756 | ||
|
79eebfb0f9 | ||
|
b75bce7f91 | ||
|
ef449c4763 | ||
|
42c2d0f0b9 | ||
|
e2f732a61c | ||
|
d53f7900d8 | ||
|
abf3166142 | ||
|
f255b89a5e | ||
|
fd717ee1b1 | ||
|
551b748297 | ||
|
87403fdb1b | ||
|
882ad42893 | ||
|
f9c256e0e4 | ||
|
8f0e42ec25 | ||
|
6b92576644 | ||
|
2f51b1ae60 | ||
|
21fbb3f294 | ||
|
d5e3f1580b | ||
|
e3c1e34fe7 | ||
|
0fc0e18cb0 | ||
|
d634a84e0f | ||
|
ba5670692e | ||
|
ceea4e5dcb | ||
|
0f5d60d61c | ||
|
9171b1fd5d | ||
|
ced1340b11 | ||
|
d7e6952db8 | ||
|
24218ec31a | ||
|
5491f01acb | ||
|
3e8c2f5cb7 | ||
|
d306b82fbe | ||
|
bd1ba3cf9d | ||
|
dfd442dfda | ||
|
f262181b4e | ||
|
f7282c516c | ||
|
4fe8b3d0bd | ||
|
69e7d8ee55 | ||
|
d90480712d | ||
|
d74b635be5 | ||
|
0b5695e55a | ||
|
34bf552c2c | ||
|
bd048b90e2 | ||
|
771eb142f0 | ||
|
492b0378a3 | ||
|
2ac5f259fa | ||
|
5bf1b045e1 | ||
|
b09c5ab001 | ||
|
c6bf7fef1d | ||
|
626351232a | ||
|
db41eaad31 | ||
|
02695f0f00 | ||
|
5aef1b9f6f | ||
|
2f28ee6626 | ||
|
9912dbd9bb | ||
|
420fecc788 | ||
|
67bea4df99 | ||
|
b12e652e4d | ||
|
d0cfb691ec | ||
|
88757eb5d1 | ||
|
e365e2cd5e | ||
|
d7db07e265 | ||
|
e5aac5fc9e | ||
|
0f969e517a | ||
|
96d38bc3e8 | ||
|
4e43fdd73b | ||
|
b20ff166e7 | ||
|
e0165b4b2d | ||
|
bbfbb86308 | ||
|
0aae38e6a0 | ||
|
991d3e2765 | ||
|
e4ce42d9f3 | ||
|
157f9484db | ||
|
a8326cd88a | ||
|
1ebd9ced8d | ||
|
723160de54 | ||
|
cc96cf2294 | ||
|
79e22b8db5 | ||
|
5a03444010 | ||
|
43ce3dc8e1 | ||
|
97e1433fde | ||
|
f6aec0a546 | ||
|
039a8626ba | ||
|
634e7267a9 | ||
|
d1e5ac742e | ||
|
9eec6b9dd5 | ||
|
c8b079caf0 | ||
|
bce3ff95ba | ||
|
faa84dbfa6 | ||
|
8003142235 | ||
|
9863a4cd74 | ||
|
c5d668ac09 | ||
|
87a4ba87e5 | ||
|
9bdbd90322 | ||
|
4812b5b970 | ||
|
65e3f84c4f | ||
|
02cc66c3ba | ||
|
c09257d28a | ||
|
769617edc4 | ||
|
d6154359d9 | ||
|
765d2bb23d | ||
|
50d60680e0 | ||
|
0d5f77f0da | ||
|
0a56912cfd | ||
|
136329f45e | ||
|
f3dc109e7e | ||
|
f84fee3624 | ||
|
118f149aee | ||
|
60165855ff | ||
|
89b9851edb | ||
|
f2a5d6d075 | ||
|
5e018ca2ae | ||
|
0160889854 | ||
|
bc9276737b | ||
|
5e3943362b | ||
|
3778d6bb66 | ||
|
fbf430373d | ||
|
a495ac72cd | ||
|
17e84504cf | ||
|
cea55cda67 | ||
|
b0ad3fa036 | ||
|
1a6e902c6b | ||
|
dadf88d5e3 | ||
|
b065ecb257 | ||
|
f0b1ff342f | ||
|
9a563a35b8 | ||
|
75543ebbe6 | ||
|
d17a6e648e | ||
|
b96fb634b1 | ||
|
9248c57549 | ||
|
8641b6d34b | ||
|
aefd5339c6 | ||
|
504aa3a05a | ||
|
ccc7f5624a | ||
|
0336fbe3cf | ||
|
452aab7ffb | ||
|
a953759b96 | ||
|
f61abdc8b5 | ||
|
bb4d2d0154 | ||
|
f44f690b6a | ||
|
eadada76b9 | ||
|
45eb323d17 | ||
|
cad67f5d0b | ||
|
18fe336fb8 | ||
|
a7c9b3e1c8 | ||
|
2026cce01a | ||
|
7b7f47f464 | ||
|
1f83234834 | ||
|
bc06b94f38 | ||
|
3abd2d69fc | ||
|
63fc1066dc | ||
|
7f4e005b1a | ||
|
5222d392b1 | ||
|
be8c1b28a3 | ||
|
e5a70d4ada | ||
|
31448a0c06 | ||
|
6d2fcd5696 | ||
|
65b909310c | ||
|
e8fca26f7a | ||
|
6ce248eacb | ||
|
a683f67f86 | ||
|
b253a086cc | ||
|
36ecbb3de8 | ||
|
3bd6929ee7 | ||
|
cb4f36fc91 | ||
|
06433d2ba1 | ||
|
31a0c78ba6 | ||
|
29f5222c35 | ||
|
13ebf33867 | ||
|
3947458623 | ||
|
cb6eb6e519 | ||
|
9896b03e0a | ||
|
5176be56db | ||
|
72fdb9fc42 | ||
|
b0b62d5ba7 | ||
|
4f73862221 | ||
|
5b83699e5c | ||
|
511520c459 | ||
|
99db93d6f8 | ||
|
24825309c3 | ||
|
db25aab203 | ||
|
d10ec1e309 | ||
|
14851be061 | ||
|
1212e00fa3 | ||
|
c537410871 | ||
|
068b96ae2b | ||
|
9c3ec3c817 | ||
|
229f3d7fda | ||
|
248dab3db1 | ||
|
c25e2c8a1d | ||
|
4a84743920 | ||
|
cc4f8b192d | ||
|
acd9c559fb | ||
|
ff7074822b | ||
|
99100f8495 | ||
|
939827eb81 | ||
|
7c1406b38c | ||
|
5f7a32e492 | ||
|
a11cbbbf41 | ||
|
dac8e07016 | ||
|
63fcb292a8 | ||
|
2a313aa6d6 | ||
|
b5a6059386 | ||
|
5904849dbc | ||
|
1987d7b43a | ||
|
54b1a97074 | ||
|
ec1aef13d6 | ||
|
bc9ec68e87 | ||
|
51762ba589 | ||
|
dffc09f0ff | ||
|
cc7f772024 | ||
|
cef0016bf4 | ||
|
5b5a658116 | ||
|
3e3dfb7c05 | ||
|
e150272201 | ||
|
228cc517e0 | ||
|
850bb9aef0 | ||
|
caafc2cbef | ||
|
3f3fa0a008 | ||
|
af51b87df7 | ||
|
78b45f2cca | ||
|
257229118f | ||
|
6aadd6053a | ||
|
db54b89dae | ||
|
64cad3e345 | ||
|
c88e79fd5b | ||
|
d0610c47d6 | ||
|
851507c3ef | ||
|
3e5afe32ab | ||
|
2359a1c5d8 | ||
|
d99ef45240 | ||
|
2b26cd1b69 | ||
|
7dddaf62bd | ||
|
3f034d0754 | ||
|
7720e32081 | ||
|
65049c8120 | ||
|
c94ab998b4 | ||
|
6e4ec92a95 | ||
|
7dcac4cdd7 | ||
|
cc70895c33 | ||
|
89b54579e0 | ||
|
c1e1a1c4ca | ||
|
429c88f290 | ||
|
074bf11633 | ||
|
5765088f7e | ||
|
316f790530 | ||
|
c91852a196 | ||
|
9ba7c5c8d6 | ||
|
f2a4223c7b | ||
|
2a5e5b48ef | ||
|
57e979bca1 | ||
|
88ca3293a6 | ||
|
018269377a | ||
|
846e102ac8 | ||
|
4551f6ee03 | ||
|
09db1e2141 | ||
|
cb9a9812bd | ||
|
d03e4063e7 | ||
|
21c5c85ac3 | ||
|
a10642d3eb | ||
|
2856c1cf76 | ||
|
3e736968d1 | ||
|
ac6d647ddd | ||
|
02439750d1 | ||
|
3efae7b161 | ||
|
49a258c830 | ||
|
3ffa3610ef | ||
|
56984b2ef1 | ||
|
653c87a32e | ||
|
07bd20120c | ||
|
bd7347b42c | ||
|
cfbef2aedb | ||
|
8436dfe8a0 | ||
|
82892d8ad8 | ||
|
34132ddbd6 | ||
|
b52af78919 | ||
|
13964c7a50 | ||
|
e5e5dc27d2 | ||
|
8ad8f81eed | ||
|
b0e855bb8a | ||
|
31a60e976b | ||
|
c108d48f00 | ||
|
fa741f635f | ||
|
5258ed7843 | ||
|
d31d7f5ae6 | ||
|
0efbfb5c2d | ||
|
57f867aa80 | ||
|
ded7d36fa8 | ||
|
93822d930c | ||
|
5474bb842d | ||
|
ff432200f0 | ||
|
310214eb46 | ||
|
bc3b08e470 | ||
|
1ef555ab24 | ||
|
c8c6cb62ea | ||
|
ac21591ee9 | ||
|
bef01206d1 | ||
|
a7131499e4 | ||
|
2a2092755c | ||
|
7d50408b87 | ||
|
9a7174ddaf | ||
|
fffc38615f | ||
|
7b31cc044a | ||
|
0331df2774 | ||
|
1405a9be04 | ||
|
b630de0efa | ||
|
a2aec76419 | ||
|
c112d3969f | ||
|
624c42a831 | ||
|
8c9adbbbc6 | ||
|
e057dd5b22 | ||
|
a5b2e042a9 | ||
|
9960cc856d | ||
|
1bb5f2610e | ||
|
8ee8fafe38 | ||
|
ab33e69fb1 | ||
|
d14fb33575 | ||
|
3f4e823fbc | ||
|
e439c21591 | ||
|
c7bb5393ad | ||
|
54a9b4c5c4 | ||
|
775ed4b228 | ||
|
5550f9b707 | ||
|
0c924152a7 | ||
|
cb36026261 | ||
|
ef6bbde78d | ||
|
44d4898862 | ||
|
46b94cb19f | ||
|
e47431c7af | ||
|
d03fb43532 | ||
|
bd6d956518 | ||
|
afd636fc3f | ||
|
a2849b016a | ||
|
4e261043cd | ||
|
4c47dbc4ee | ||
|
07264e626f | ||
|
64a1f8d9bf | ||
|
15a2d493f8 | ||
|
3777041d3e | ||
|
ccd0707d21 | ||
|
a5b03e6625 | ||
|
03b4d9eb5e | ||
|
af48272100 | ||
|
cc63e10cd1 | ||
|
f90cd27bcd | ||
|
1ab01d0277 |
11
.github/dependabot.yml
vendored
Normal file
11
.github/dependabot.yml
vendored
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
version: 2
|
||||||
|
updates:
|
||||||
|
- package-ecosystem: "github-actions"
|
||||||
|
directory: "/"
|
||||||
|
schedule:
|
||||||
|
interval: "daily"
|
||||||
|
|
||||||
|
- package-ecosystem: docker
|
||||||
|
directory: "/docker"
|
||||||
|
schedule:
|
||||||
|
interval: "daily"
|
29
.github/workflows/docker-build.yml
vendored
Normal file
29
.github/workflows/docker-build.yml
vendored
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
name: Docker build & push
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- master
|
||||||
|
paths:
|
||||||
|
- buildinfo.json
|
||||||
|
# workaround for #526
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v5
|
||||||
|
|
||||||
|
- name: Set up QEMU
|
||||||
|
uses: docker/setup-qemu-action@v3
|
||||||
|
|
||||||
|
- name: build and push all images
|
||||||
|
if: ${{ env.DOCKER_USERNAME != '' && env.DOCKER_PASSWORD != '' }}
|
||||||
|
env:
|
||||||
|
DOCKER_PASSWORD: ${{ secrets.DOCKER_PASSWORD }}
|
||||||
|
DOCKER_USERNAME: ${{ secrets.DOCKER_USERNAME }}
|
||||||
|
run: |
|
||||||
|
./build.py --push-tags --multiarch --both
|
28
.github/workflows/docker-description.yml
vendored
Normal file
28
.github/workflows/docker-description.yml
vendored
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
name: Docker Hub Description
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- master
|
||||||
|
paths:
|
||||||
|
- README.md
|
||||||
|
# workaround for #526
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
docker-description:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v5
|
||||||
|
|
||||||
|
- name: Docker Hub Description
|
||||||
|
uses: peter-evans/dockerhub-description@v4.0.2
|
||||||
|
if: ${{ env.DOCKER_USERNAME != '' && env.DOCKER_PASSWORD != '' }}
|
||||||
|
env:
|
||||||
|
DOCKER_PASSWORD: ${{ secrets.DOCKER_PASSWORD }}
|
||||||
|
DOCKER_USERNAME: ${{ secrets.DOCKER_USERNAME }}
|
||||||
|
with:
|
||||||
|
username: ${{ secrets.DOCKER_USERNAME }}
|
||||||
|
password: ${{ secrets.DOCKER_PASSWORD }}
|
||||||
|
repository: factoriotools/factorio
|
32
.github/workflows/lint.yml
vendored
Normal file
32
.github/workflows/lint.yml
vendored
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
name: 'Linter'
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- master
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
shellcheck:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v5
|
||||||
|
|
||||||
|
- name: shellcheck
|
||||||
|
uses: reviewdog/action-shellcheck@v1
|
||||||
|
with:
|
||||||
|
github_token: ${{ secrets.github_token }}
|
||||||
|
reporter: github-pr-review
|
||||||
|
|
||||||
|
hadolint:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v5
|
||||||
|
|
||||||
|
- name: hadolint
|
||||||
|
uses: reviewdog/action-hadolint@v1
|
||||||
|
with:
|
||||||
|
github_token: ${{ secrets.github_token }}
|
||||||
|
reporter: github-pr-review
|
21
.github/workflows/update.yml
vendored
Normal file
21
.github/workflows/update.yml
vendored
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
name: Check Update
|
||||||
|
|
||||||
|
on:
|
||||||
|
schedule:
|
||||||
|
- cron: "0 * * * *"
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
check:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
if: github.event_name != 'schedule' || (github.event_name == 'schedule' && github.repository == 'factoriotools/factorio-docker')
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v5
|
||||||
|
with:
|
||||||
|
token: ${{ secrets.REPO_TOKEN }}
|
||||||
|
|
||||||
|
- name: Run update script
|
||||||
|
run: ./update.sh
|
||||||
|
shell: bash
|
7
.gitignore
vendored
Normal file
7
.gitignore
vendored
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
# IDE
|
||||||
|
.idea
|
||||||
|
|
||||||
|
# Python
|
||||||
|
__pycache__/
|
||||||
|
*.py[cod]
|
||||||
|
*$py.class
|
34
.travis.yml
34
.travis.yml
@@ -1,34 +0,0 @@
|
|||||||
dist: xenial
|
|
||||||
language: bash
|
|
||||||
services:
|
|
||||||
- docker
|
|
||||||
|
|
||||||
addons:
|
|
||||||
apt:
|
|
||||||
packages:
|
|
||||||
- jq
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
include:
|
|
||||||
- stage: test
|
|
||||||
script:
|
|
||||||
- git ls-files --exclude='*Dockerfile' --ignored | xargs --max-lines=1 -I{} sh -c 'docker run --rm -i -v ${PWD}/.hadolint.yaml:/.hadolint.yaml hadolint/hadolint < "$1"' -- {}
|
|
||||||
- bash -c 'shopt -s globstar; shellcheck **/*.sh'
|
|
||||||
|
|
||||||
- &build
|
|
||||||
stage: Build & update Docker HUB description
|
|
||||||
env: VERSION_SHORT=0.18 EXTRA_TAG=latest
|
|
||||||
script:
|
|
||||||
- ./build.sh $VERSION_SHORT
|
|
||||||
- <<: *build
|
|
||||||
env: VERSION_SHORT=0.17 EXTRA_TAG=stable
|
|
||||||
- <<: *build
|
|
||||||
env: VERSION_SHORT=0.16
|
|
||||||
- <<: *build
|
|
||||||
env: VERSION_SHORT=0.15
|
|
||||||
- <<: *build
|
|
||||||
env: VERSION_SHORT=0.14
|
|
||||||
- <<: *build
|
|
||||||
if: branch = master AND type != pull_request
|
|
||||||
env:
|
|
||||||
script: docker run -v $PWD:/workspace -e DOCKERHUB_USERNAME=$DOCKER_USERNAME -e DOCKERHUB_PASSWORD=$DOCKER_PASSWORD -e DOCKERHUB_REPOSITORY='factoriotools/factorio' -e README_FILEPATH='/workspace/README.md' peterevans/dockerhub-description:2.1.0
|
|
@@ -1,26 +0,0 @@
|
|||||||
FROM frolvlad/alpine-glibc:alpine-3.9
|
|
||||||
|
|
||||||
LABEL maintainer="https://github.com/factoriotools/factorio-docker"
|
|
||||||
|
|
||||||
ENV VERSION=0.14.23 \
|
|
||||||
SHA1=6ef84341c6fc1cf45cfdd6acc8468aaa117b9e8a
|
|
||||||
|
|
||||||
RUN mkdir -p /opt \
|
|
||||||
&& apk --no-cache add curl tini pwgen \
|
|
||||||
&& curl -sSL https://www.factorio.com/get-download/$VERSION/headless/linux64 \
|
|
||||||
-o /tmp/factorio_headless_x64_$VERSION.tar.gz \
|
|
||||||
&& echo "$SHA1 /tmp/factorio_headless_x64_$VERSION.tar.gz" | sha1sum -c \
|
|
||||||
&& tar xzf /tmp/factorio_headless_x64_$VERSION.tar.gz --directory /opt \
|
|
||||||
&& rm /tmp/factorio_headless_x64_$VERSION.tar.gz \
|
|
||||||
&& apk del curl \
|
|
||||||
&& ln -s /factorio/saves /opt/factorio/saves \
|
|
||||||
&& ln -s /factorio/mods /opt/factorio/mods
|
|
||||||
|
|
||||||
VOLUME /factorio
|
|
||||||
|
|
||||||
EXPOSE 34197/udp 27015/tcp
|
|
||||||
|
|
||||||
COPY ./docker-entrypoint.sh /
|
|
||||||
|
|
||||||
ENTRYPOINT ["/sbin/tini", "--"]
|
|
||||||
CMD ["/docker-entrypoint.sh"]
|
|
@@ -1,2 +0,0 @@
|
|||||||
#!/bin/sh
|
|
||||||
sudo docker build --no-cache -t factorio .
|
|
@@ -1,2 +0,0 @@
|
|||||||
#!/bin/sh
|
|
||||||
sudo rm -rf /tmp/factorio
|
|
@@ -1,35 +0,0 @@
|
|||||||
#!/bin/sh -x
|
|
||||||
|
|
||||||
set -euo pipefail
|
|
||||||
|
|
||||||
SAVES=/factorio/saves
|
|
||||||
CONFIG=/factorio/config
|
|
||||||
|
|
||||||
mkdir -p "$SAVES"
|
|
||||||
mkdir -p /factorio/mods
|
|
||||||
mkdir -p "$CONFIG"
|
|
||||||
|
|
||||||
if [ ! -f $CONFIG/rconpw ]; then
|
|
||||||
pwgen 15 1 >"$CONFIG/rconpw"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ ! -f "$CONFIG/server-settings.json" ]; then
|
|
||||||
cp /opt/factorio/data/server-settings.example.json "$CONFIG/server-settings.json"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ ! -f "$CONFIG/map-gen-settings.json" ]; then
|
|
||||||
cp /opt/factorio/data/map-gen-settings.example.json "$CONFIG/map-gen-settings.json"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if ! find -L "$SAVES" -iname \*.zip -mindepth 1 -print | grep -q .; then
|
|
||||||
/opt/factorio/bin/x64/factorio \
|
|
||||||
--create "$SAVES/_autosave1.zip" \
|
|
||||||
--map-gen-settings "$CONFIG/map-gen-settings.json"
|
|
||||||
fi
|
|
||||||
|
|
||||||
exec /opt/factorio/bin/x64/factorio \
|
|
||||||
--port 34197 \
|
|
||||||
--start-server-load-latest \
|
|
||||||
--server-settings "$CONFIG/server-settings.json" \
|
|
||||||
--rcon-port 27015 \
|
|
||||||
--rcon-password "$(cat "$CONFIG/rconpw")"
|
|
@@ -1,6 +0,0 @@
|
|||||||
#!/bin/sh
|
|
||||||
sudo docker run --rm -it \
|
|
||||||
-v /tmp/factorio:/factorio \
|
|
||||||
--name factorio \
|
|
||||||
factorio "$@"
|
|
||||||
find /tmp/factorio -type f
|
|
@@ -1,30 +0,0 @@
|
|||||||
FROM frolvlad/alpine-glibc:alpine-3.9
|
|
||||||
|
|
||||||
LABEL maintainer="https://github.com/factoriotools/factorio-docker"
|
|
||||||
|
|
||||||
ENV PORT=34197 \
|
|
||||||
RCON_PORT=27015 \
|
|
||||||
VERSION=0.15.40 \
|
|
||||||
SHA1=f79a975f6b8c0ee87e2fa60f7d1f7133f332c3ec
|
|
||||||
|
|
||||||
RUN mkdir -p /opt \
|
|
||||||
&& apk add --update --no-cache tini pwgen \
|
|
||||||
&& apk add --update --no-cache --virtual .build-deps curl \
|
|
||||||
&& curl -sSL https://www.factorio.com/get-download/$VERSION/headless/linux64 \
|
|
||||||
-o /tmp/factorio_headless_x64_$VERSION.tar.xz \
|
|
||||||
&& echo "$SHA1 /tmp/factorio_headless_x64_$VERSION.tar.xz" | sha1sum -c \
|
|
||||||
&& tar xf /tmp/factorio_headless_x64_$VERSION.tar.xz --directory /opt \
|
|
||||||
&& chmod -R ugo=rwx /opt/factorio \
|
|
||||||
&& rm /tmp/factorio_headless_x64_$VERSION.tar.xz \
|
|
||||||
&& ln -s /factorio/saves /opt/factorio/saves \
|
|
||||||
&& ln -s /factorio/mods /opt/factorio/mods \
|
|
||||||
&& apk del .build-deps
|
|
||||||
|
|
||||||
VOLUME /factorio
|
|
||||||
|
|
||||||
EXPOSE $PORT/udp $RCON_PORT/tcp
|
|
||||||
|
|
||||||
COPY ./docker-entrypoint.sh /
|
|
||||||
|
|
||||||
ENTRYPOINT ["/sbin/tini", "--"]
|
|
||||||
CMD ["/docker-entrypoint.sh"]
|
|
@@ -1,8 +0,0 @@
|
|||||||
version: '2'
|
|
||||||
services:
|
|
||||||
factorio:
|
|
||||||
build: .
|
|
||||||
ports:
|
|
||||||
- "34197:34197"
|
|
||||||
volumes:
|
|
||||||
- /tmp/factorio:/factorio
|
|
@@ -1,43 +0,0 @@
|
|||||||
#!/bin/sh -x
|
|
||||||
|
|
||||||
set -euo pipefail
|
|
||||||
|
|
||||||
SAVES=/factorio/saves
|
|
||||||
CONFIG=/factorio/config
|
|
||||||
|
|
||||||
mkdir -p "$SAVES"
|
|
||||||
mkdir -p /factorio/mods
|
|
||||||
mkdir -p "$CONFIG"
|
|
||||||
|
|
||||||
if [ ! -f "$CONFIG/rconpw" ]; then
|
|
||||||
pwgen 15 1 >"$CONFIG/rconpw"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ ! -f "$CONFIG/server-settings.json" ]; then
|
|
||||||
cp /opt/factorio/data/server-settings.example.json "$CONFIG/server-settings.json"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ ! -f "$CONFIG/map-gen-settings.json" ]; then
|
|
||||||
cp /opt/factorio/data/map-gen-settings.example.json "$CONFIG/map-gen-settings.json"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ ! -f "$CONFIG/map-settings.json" ]; then
|
|
||||||
cp /opt/factorio/data/map-settings.example.json "$CONFIG/map-settings.json"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if ! find -L "$SAVES" -iname \*.zip -mindepth 1 -print | grep -q .; then
|
|
||||||
/opt/factorio/bin/x64/factorio \
|
|
||||||
--create "$SAVES/_autosave1.zip" \
|
|
||||||
--map-gen-settings "$CONFIG/map-gen-settings.json" \
|
|
||||||
--map-settings "$CONFIG/map-settings.json"
|
|
||||||
fi
|
|
||||||
|
|
||||||
exec /opt/factorio/bin/x64/factorio \
|
|
||||||
--port "$PORT" \
|
|
||||||
--start-server-load-latest \
|
|
||||||
--server-settings "$CONFIG/server-settings.json" \
|
|
||||||
--server-whitelist "$CONFIG/server-whitelist.json" \
|
|
||||||
--server-banlist "$CONFIG/server-banlist.json" \
|
|
||||||
--rcon-port "$RCON_PORT" \
|
|
||||||
--rcon-password "$(cat "$CONFIG/rconpw")" \
|
|
||||||
--server-id /factorio/config/server-id.json
|
|
@@ -1,46 +0,0 @@
|
|||||||
FROM frolvlad/alpine-glibc:alpine-3.9
|
|
||||||
|
|
||||||
LABEL maintainer="https://github.com/factoriotools/factorio-docker"
|
|
||||||
|
|
||||||
ARG USER=factorio
|
|
||||||
ARG GROUP=factorio
|
|
||||||
ARG PUID=845
|
|
||||||
ARG PGID=845
|
|
||||||
|
|
||||||
ENV PORT=34197 \
|
|
||||||
RCON_PORT=27015 \
|
|
||||||
VERSION=0.16.51 \
|
|
||||||
SHA1=127e7ff484ab263b13615d6114013ce0a66ac929 \
|
|
||||||
SAVES=/factorio/saves \
|
|
||||||
CONFIG=/factorio/config \
|
|
||||||
MODS=/factorio/mods \
|
|
||||||
SCENARIOS=/factorio/scenarios \
|
|
||||||
SCRIPTOUTPUT=/factorio/script-output \
|
|
||||||
PUID="$PUID" \
|
|
||||||
PGID="$PGID"
|
|
||||||
|
|
||||||
RUN mkdir -p /opt /factorio \
|
|
||||||
&& apk add --update --no-cache pwgen su-exec shadow \
|
|
||||||
&& apk add --update --no-cache --virtual .build-deps curl \
|
|
||||||
&& curl -sSL https://www.factorio.com/get-download/$VERSION/headless/linux64 \
|
|
||||||
-o /tmp/factorio_headless_x64_$VERSION.tar.xz \
|
|
||||||
&& echo "$SHA1 /tmp/factorio_headless_x64_$VERSION.tar.xz" | sha1sum -c \
|
|
||||||
&& tar xf /tmp/factorio_headless_x64_$VERSION.tar.xz --directory /opt \
|
|
||||||
&& chmod ugo=rwx /opt/factorio \
|
|
||||||
&& rm /tmp/factorio_headless_x64_$VERSION.tar.xz \
|
|
||||||
&& ln -s $SAVES /opt/factorio/saves \
|
|
||||||
&& ln -s $MODS /opt/factorio/mods \
|
|
||||||
&& ln -s $SCENARIOS /opt/factorio/scenarios \
|
|
||||||
&& ln -s $SCRIPTOUTPUT /opt/factorio/script-output \
|
|
||||||
&& apk del .build-deps \
|
|
||||||
&& addgroup -g $PGID -S $GROUP \
|
|
||||||
&& adduser -u $PUID -G $GROUP -s /bin/sh -SDH $USER \
|
|
||||||
&& chown -R $USER:$GROUP /opt/factorio /factorio
|
|
||||||
|
|
||||||
VOLUME /factorio
|
|
||||||
|
|
||||||
EXPOSE $PORT/udp $RCON_PORT/tcp
|
|
||||||
|
|
||||||
COPY files/ /
|
|
||||||
|
|
||||||
ENTRYPOINT ["/docker-entrypoint.sh"]
|
|
@@ -1,9 +0,0 @@
|
|||||||
version: '2'
|
|
||||||
services:
|
|
||||||
factorio:
|
|
||||||
build: .
|
|
||||||
ports:
|
|
||||||
- "34197:34197/udp"
|
|
||||||
- "27015:27015/tcp"
|
|
||||||
volumes:
|
|
||||||
- /opt/factorio:/factorio
|
|
@@ -1,62 +0,0 @@
|
|||||||
#!/bin/sh -x
|
|
||||||
set -euo pipefail
|
|
||||||
|
|
||||||
id
|
|
||||||
|
|
||||||
FACTORIO_VOL=/factorio
|
|
||||||
mkdir -p "$FACTORIO_VOL"
|
|
||||||
mkdir -p "$SAVES"
|
|
||||||
mkdir -p "$CONFIG"
|
|
||||||
mkdir -p "$MODS"
|
|
||||||
mkdir -p "$SCENARIOS"
|
|
||||||
mkdir -p "$SCRIPTOUTPUT"
|
|
||||||
|
|
||||||
if [ ! -f "$CONFIG/rconpw" ]; then
|
|
||||||
pwgen 15 1>"$CONFIG/rconpw"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ ! -f "$CONFIG/server-settings.json" ]; then
|
|
||||||
cp /opt/factorio/data/server-settings.example.json "$CONFIG/server-settings.json"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ ! -f "$CONFIG/map-gen-settings.json" ]; then
|
|
||||||
cp /opt/factorio/data/map-gen-settings.example.json "$CONFIG/map-gen-settings.json"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ ! -f "$CONFIG/map-settings.json" ]; then
|
|
||||||
cp /opt/factorio/data/map-settings.example.json "$CONFIG/map-settings.json"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if find -L "$SAVES" -iname \*.tmp.zip -mindepth 1 -print | grep -q .; then
|
|
||||||
rm -f "$SAVES"/*.tmp.zip
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ "$(id -u)" = '0' ]; then
|
|
||||||
# Update the User and Group ID based on the PUID/PGID variables
|
|
||||||
usermod -o -u "$PUID" factorio
|
|
||||||
groupmod -o -g "$PGID" factorio
|
|
||||||
# Take ownership of factorio data if running as root
|
|
||||||
chown -R factorio:factorio "$FACTORIO_VOL"
|
|
||||||
# Drop to the factorio user
|
|
||||||
SU_EXEC="su-exec factorio"
|
|
||||||
else
|
|
||||||
SU_EXEC=""
|
|
||||||
fi
|
|
||||||
|
|
||||||
if ! find -L "$SAVES" -iname \*.zip -mindepth 1 -print | grep -q .; then
|
|
||||||
$SU_EXEC /opt/factorio/bin/x64/factorio \
|
|
||||||
--create "$SAVES/_autosave1.zip" \
|
|
||||||
--map-gen-settings "$CONFIG/map-gen-settings.json" \
|
|
||||||
--map-settings "$CONFIG/map-settings.json"
|
|
||||||
fi
|
|
||||||
|
|
||||||
$SU_EXEC /opt/factorio/bin/x64/factorio \
|
|
||||||
--port "$PORT" \
|
|
||||||
--start-server-load-latest \
|
|
||||||
--server-settings "$CONFIG/server-settings.json" \
|
|
||||||
--server-whitelist "$CONFIG/server-whitelist.json" \
|
|
||||||
--server-banlist "$CONFIG/server-banlist.json" \
|
|
||||||
--rcon-port "$RCON_PORT" \
|
|
||||||
--rcon-password "$(cat "$CONFIG/rconpw")" \
|
|
||||||
--server-id /factorio/config/server-id.json \
|
|
||||||
"$@"
|
|
@@ -1,42 +0,0 @@
|
|||||||
#!/bin/sh -x
|
|
||||||
if [ -z "$1" ]; then
|
|
||||||
echo "No argument supplied"
|
|
||||||
fi
|
|
||||||
SERVER_SCENARIO=$1
|
|
||||||
|
|
||||||
set -euo pipefail
|
|
||||||
|
|
||||||
id
|
|
||||||
|
|
||||||
mkdir -p "$SAVES"
|
|
||||||
mkdir -p "$CONFIG"
|
|
||||||
mkdir -p "$MODS"
|
|
||||||
mkdir -p "$SCENARIOS"
|
|
||||||
|
|
||||||
#chown -R factorio /factorio
|
|
||||||
|
|
||||||
if [ ! -f "$CONFIG/rconpw" ]; then
|
|
||||||
pwgen 15 1 >"$CONFIG/rconpw"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ ! -f "$CONFIG/server-settings.json" ]; then
|
|
||||||
cp /opt/factorio/data/server-settings.example.json "$CONFIG/server-settings.json"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ ! -f "$CONFIG/map-gen-settings.json" ]; then
|
|
||||||
cp /opt/factorio/data/map-gen-settings.example.json "$CONFIG/map-gen-settings.json"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ ! -f "$CONFIG/map-settings.json" ]; then
|
|
||||||
cp /opt/factorio/data/map-settings.example.json "$CONFIG/map-settings.json"
|
|
||||||
fi
|
|
||||||
|
|
||||||
exec /opt/factorio/bin/x64/factorio \
|
|
||||||
--port "$PORT" \
|
|
||||||
--start-server-load-scenario "$SERVER_SCENARIO" \
|
|
||||||
--server-settings "$CONFIG/server-settings.json" \
|
|
||||||
--server-whitelist "$CONFIG/server-whitelist.json" \
|
|
||||||
--server-banlist "$CONFIG/server-banlist.json" \
|
|
||||||
--rcon-port "$RCON_PORT" \
|
|
||||||
--rcon-password "$(cat "$CONFIG/rconpw")" \
|
|
||||||
--server-id /factorio/config/server-id.json
|
|
@@ -1,29 +0,0 @@
|
|||||||
#!/bin/sh -x
|
|
||||||
if [ -z "$1" ]; then
|
|
||||||
echo "No argument supplied"
|
|
||||||
fi
|
|
||||||
SERVER_SCENARIO=$1
|
|
||||||
|
|
||||||
set -euo pipefail
|
|
||||||
|
|
||||||
id
|
|
||||||
|
|
||||||
mkdir -p "$SAVES"
|
|
||||||
mkdir -p "$CONFIG"
|
|
||||||
mkdir -p "$MODS"
|
|
||||||
mkdir -p "$SCENARIOS"
|
|
||||||
|
|
||||||
if [ ! -f "$CONFIG/server-settings.json" ]; then
|
|
||||||
cp /opt/factorio/data/server-settings.example.json "$CONFIG/server-settings.json"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ ! -f "$CONFIG/map-gen-settings.json" ]; then
|
|
||||||
cp /opt/factorio/data/map-gen-settings.example.json "$CONFIG/map-gen-settings.json"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ ! -f "$CONFIG/map-settings.json" ]; then
|
|
||||||
cp /opt/factorio/data/map-settings.example.json "$CONFIG/map-settings.json"
|
|
||||||
fi
|
|
||||||
|
|
||||||
exec /opt/factorio/bin/x64/factorio \
|
|
||||||
--scenario2map "$SERVER_SCENARIO"
|
|
@@ -1,46 +0,0 @@
|
|||||||
FROM frolvlad/alpine-glibc:alpine-3.9
|
|
||||||
|
|
||||||
LABEL maintainer="https://github.com/factoriotools/factorio-docker"
|
|
||||||
|
|
||||||
ARG USER=factorio
|
|
||||||
ARG GROUP=factorio
|
|
||||||
ARG PUID=845
|
|
||||||
ARG PGID=845
|
|
||||||
|
|
||||||
ENV PORT=34197 \
|
|
||||||
RCON_PORT=27015 \
|
|
||||||
VERSION=0.17.79 \
|
|
||||||
SHA1=7f127baf3cf01c6e545a9ca376dec1ac37468f8a \
|
|
||||||
SAVES=/factorio/saves \
|
|
||||||
CONFIG=/factorio/config \
|
|
||||||
MODS=/factorio/mods \
|
|
||||||
SCENARIOS=/factorio/scenarios \
|
|
||||||
SCRIPTOUTPUT=/factorio/script-output \
|
|
||||||
PUID="$PUID" \
|
|
||||||
PGID="$PGID"
|
|
||||||
|
|
||||||
RUN set -ox pipefail \
|
|
||||||
&& archive="/tmp/factorio_headless_x64_$VERSION.tar.xz" \
|
|
||||||
&& mkdir -p /opt /factorio \
|
|
||||||
&& apk add --update --no-cache --no-progress bash binutils curl file gettext jq libintl pwgen shadow su-exec \
|
|
||||||
&& curl -sSL "https://www.factorio.com/get-download/$VERSION/headless/linux64" -o "$archive" \
|
|
||||||
&& echo "$SHA1 $archive" | sha1sum -c \
|
|
||||||
|| (sha1sum "$archive" && file "$archive" && exit 1) \
|
|
||||||
&& tar xf "$archive" --directory /opt \
|
|
||||||
&& chmod ugo=rwx /opt/factorio \
|
|
||||||
&& rm "$archive" \
|
|
||||||
&& ln -s "$SAVES" /opt/factorio/saves \
|
|
||||||
&& ln -s "$MODS" /opt/factorio/mods \
|
|
||||||
&& ln -s "$SCENARIOS" /opt/factorio/scenarios \
|
|
||||||
&& ln -s "$SCRIPTOUTPUT" /opt/factorio/script-output \
|
|
||||||
&& addgroup -g "$PGID" -S "$GROUP" \
|
|
||||||
&& adduser -u "$PUID" -G "$GROUP" -s /bin/sh -SDH "$USER" \
|
|
||||||
&& chown -R "$USER":"$GROUP" /opt/factorio /factorio
|
|
||||||
|
|
||||||
VOLUME /factorio
|
|
||||||
|
|
||||||
EXPOSE $PORT/udp $RCON_PORT/tcp
|
|
||||||
|
|
||||||
COPY files/ /
|
|
||||||
|
|
||||||
ENTRYPOINT ["/docker-entrypoint.sh"]
|
|
@@ -1,15 +0,0 @@
|
|||||||
version: '2'
|
|
||||||
services:
|
|
||||||
factorio:
|
|
||||||
build: .
|
|
||||||
ports:
|
|
||||||
- "34197:34197/udp"
|
|
||||||
- "27015:27015/tcp"
|
|
||||||
volumes:
|
|
||||||
- /opt/factorio:/factorio
|
|
||||||
# environment:
|
|
||||||
# - PUID=1000
|
|
||||||
# - PGID=1000
|
|
||||||
# - UPDATE_MODS_ON_START=true
|
|
||||||
# - USERNAME=FactorioUsername
|
|
||||||
# - TOKEN=FactorioToken
|
|
@@ -1,95 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
set -eou pipefail
|
|
||||||
|
|
||||||
FACTORIO_VERSION=$1
|
|
||||||
MOD_DIR=$2
|
|
||||||
USERNAME=$3
|
|
||||||
TOKEN=$4
|
|
||||||
|
|
||||||
MOD_BASE_URL="https://mods.factorio.com"
|
|
||||||
|
|
||||||
print_step()
|
|
||||||
{
|
|
||||||
echo "$1"
|
|
||||||
}
|
|
||||||
|
|
||||||
print_success()
|
|
||||||
{
|
|
||||||
echo "$1"
|
|
||||||
}
|
|
||||||
|
|
||||||
print_failure()
|
|
||||||
{
|
|
||||||
echo "$1"
|
|
||||||
}
|
|
||||||
|
|
||||||
update_mod()
|
|
||||||
{
|
|
||||||
MOD_NAME="${1// /%20}"
|
|
||||||
|
|
||||||
print_step "Checking for update of mod $MOD_NAME..."
|
|
||||||
|
|
||||||
MOD_INFO_URL="$MOD_BASE_URL/api/mods/$MOD_NAME"
|
|
||||||
MOD_INFO_JSON=$(curl --silent "$MOD_INFO_URL")
|
|
||||||
|
|
||||||
if ! echo "$MOD_INFO_JSON" | jq -e .name >/dev/null; then
|
|
||||||
print_success " Custom mod not on $MOD_BASE_URL, skipped."
|
|
||||||
return 0
|
|
||||||
fi
|
|
||||||
|
|
||||||
MOD_INFO=$(echo "$MOD_INFO_JSON" | jq -j --arg version "$FACTORIO_VERSION" ".releases|reverse|map(select(.info_json.factorio_version as \$mod_version | \$version | startswith(\$mod_version)))[0]|.file_name, \";\", .download_url, \";\", .sha1")
|
|
||||||
|
|
||||||
MOD_FILENAME=$(echo "$MOD_INFO" | cut -f1 -d";")
|
|
||||||
MOD_URL=$(echo "$MOD_INFO" | cut -f2 -d";")
|
|
||||||
MOD_SHA1=$(echo "$MOD_INFO" | cut -f3 -d";")
|
|
||||||
|
|
||||||
if [[ $MOD_FILENAME == null ]]; then
|
|
||||||
print_failure " Not compatible with version"
|
|
||||||
return 0
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ -f $MOD_DIR/$MOD_FILENAME ]]; then
|
|
||||||
print_success " Already up-to-date."
|
|
||||||
return 0
|
|
||||||
fi
|
|
||||||
|
|
||||||
print_step "Downloading..."
|
|
||||||
FULL_URL="$MOD_BASE_URL$MOD_URL?username=$USERNAME&token=$TOKEN"
|
|
||||||
HTTP_STATUS=$(curl --silent -L -w "%{http_code}" -o "$MOD_DIR/$MOD_FILENAME" "$FULL_URL")
|
|
||||||
|
|
||||||
if [[ $HTTP_STATUS != 200 ]]; then
|
|
||||||
print_failure " Download failed: Code $HTTP_STATUS."
|
|
||||||
rm -f "$MOD_DIR/$MOD_FILENAME"
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ ! -f $MOD_DIR/$MOD_FILENAME ]]; then
|
|
||||||
print_failure " Downloaded file missing!"
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
if ! [[ $(sha1sum "$MOD_DIR/$MOD_FILENAME") =~ $MOD_SHA1 ]]; then
|
|
||||||
print_failure " SHA1 mismatch!"
|
|
||||||
rm -f "$MOD_DIR/$MOD_FILENAME"
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
print_success " Download complete."
|
|
||||||
|
|
||||||
for file in "$MOD_DIR/${MOD_NAME}_"*".zip"; do # wildcard does usually not work in quotes: https://unix.stackexchange.com/a/67761
|
|
||||||
if [[ $file != $MOD_DIR/$MOD_FILENAME ]]; then
|
|
||||||
print_success " Deleting old version: $file"
|
|
||||||
rm -f "$file"
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
if [[ -f $MOD_DIR/mod-list.json ]]; then
|
|
||||||
jq -r ".mods|map(select(.enabled))|.[].name" "$MOD_DIR/mod-list.json" | while read -r mod; do
|
|
||||||
if [[ $mod != base ]]; then
|
|
||||||
update_mod "$mod"
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
fi
|
|
@@ -1,46 +0,0 @@
|
|||||||
FROM frolvlad/alpine-glibc:alpine-3.9
|
|
||||||
|
|
||||||
LABEL maintainer="https://github.com/factoriotools/factorio-docker"
|
|
||||||
|
|
||||||
ARG USER=factorio
|
|
||||||
ARG GROUP=factorio
|
|
||||||
ARG PUID=845
|
|
||||||
ARG PGID=845
|
|
||||||
|
|
||||||
ENV PORT=34197 \
|
|
||||||
RCON_PORT=27015 \
|
|
||||||
VERSION=0.18.21 \
|
|
||||||
SHA1=21bab03c7bc85fbf9c640ad1f8eba1825ecbcb91 \
|
|
||||||
SAVES=/factorio/saves \
|
|
||||||
CONFIG=/factorio/config \
|
|
||||||
MODS=/factorio/mods \
|
|
||||||
SCENARIOS=/factorio/scenarios \
|
|
||||||
SCRIPTOUTPUT=/factorio/script-output \
|
|
||||||
PUID="$PUID" \
|
|
||||||
PGID="$PGID"
|
|
||||||
|
|
||||||
RUN set -ox pipefail \
|
|
||||||
&& archive="/tmp/factorio_headless_x64_$VERSION.tar.xz" \
|
|
||||||
&& mkdir -p /opt /factorio \
|
|
||||||
&& apk add --update --no-cache --no-progress bash binutils curl file gettext jq libintl pwgen shadow su-exec \
|
|
||||||
&& curl -sSL "https://www.factorio.com/get-download/$VERSION/headless/linux64" -o "$archive" \
|
|
||||||
&& echo "$SHA1 $archive" | sha1sum -c \
|
|
||||||
|| (sha1sum "$archive" && file "$archive" && exit 1) \
|
|
||||||
&& tar xf "$archive" --directory /opt \
|
|
||||||
&& chmod ugo=rwx /opt/factorio \
|
|
||||||
&& rm "$archive" \
|
|
||||||
&& ln -s "$SAVES" /opt/factorio/saves \
|
|
||||||
&& ln -s "$MODS" /opt/factorio/mods \
|
|
||||||
&& ln -s "$SCENARIOS" /opt/factorio/scenarios \
|
|
||||||
&& ln -s "$SCRIPTOUTPUT" /opt/factorio/script-output \
|
|
||||||
&& addgroup -g "$PGID" -S "$GROUP" \
|
|
||||||
&& adduser -u "$PUID" -G "$GROUP" -s /bin/sh -SDH "$USER" \
|
|
||||||
&& chown -R "$USER":"$GROUP" /opt/factorio /factorio
|
|
||||||
|
|
||||||
VOLUME /factorio
|
|
||||||
|
|
||||||
EXPOSE $PORT/udp $RCON_PORT/tcp
|
|
||||||
|
|
||||||
COPY files/ /
|
|
||||||
|
|
||||||
ENTRYPOINT ["/docker-entrypoint.sh"]
|
|
@@ -1,15 +0,0 @@
|
|||||||
version: '2'
|
|
||||||
services:
|
|
||||||
factorio:
|
|
||||||
build: .
|
|
||||||
ports:
|
|
||||||
- "34197:34197/udp"
|
|
||||||
- "27015:27015/tcp"
|
|
||||||
volumes:
|
|
||||||
- /opt/factorio:/factorio
|
|
||||||
# environment:
|
|
||||||
# - PUID=1000
|
|
||||||
# - PGID=1000
|
|
||||||
# - UPDATE_MODS_ON_START=true
|
|
||||||
# - USERNAME=FactorioUsername
|
|
||||||
# - TOKEN=FactorioToken
|
|
@@ -1,28 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
set -eou pipefail
|
|
||||||
|
|
||||||
if [[ -f /run/secrets/username ]]; then
|
|
||||||
USERNAME=$(cat /run/secrets/username)
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ -f /run/secrets/username ]]; then
|
|
||||||
TOKEN=$(cat /run/secrets/token)
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ -z ${USERNAME:-} ]]; then
|
|
||||||
USERNAME="$(jq -j ".username" "$CONFIG/server-settings.json")"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ -z ${TOKEN:-} ]]; then
|
|
||||||
TOKEN="$(jq -j ".token" "$CONFIG/server-settings.json")"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ -z ${USERNAME:-} ]]; then
|
|
||||||
echo "You need to provide your Factorio username to update mods."
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ -z ${TOKEN:-} ]]; then
|
|
||||||
echo "You need to provide your Factorio token to update mods."
|
|
||||||
fi
|
|
||||||
|
|
||||||
./update-mods.sh "$VERSION" "$MODS" "$USERNAME" "$TOKEN"
|
|
@@ -1,42 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
set -eoux pipefail
|
|
||||||
|
|
||||||
if [[ -z ${1:-} ]]; then
|
|
||||||
echo "No argument supplied"
|
|
||||||
fi
|
|
||||||
|
|
||||||
SERVER_SCENARIO="$1"
|
|
||||||
mkdir -p "$SAVES"
|
|
||||||
mkdir -p "$CONFIG"
|
|
||||||
mkdir -p "$MODS"
|
|
||||||
mkdir -p "$SCENARIOS"
|
|
||||||
|
|
||||||
#chown -R factorio /factorio
|
|
||||||
|
|
||||||
if [[ ! -f $CONFIG/rconpw ]]; then
|
|
||||||
pwgen 15 1 >"$CONFIG/rconpw"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ ! -f $CONFIG/server-settings.json ]]; then
|
|
||||||
cp /opt/factorio/data/server-settings.example.json "$CONFIG/server-settings.json"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ ! -f $CONFIG/map-gen-settings.json ]]; then
|
|
||||||
cp /opt/factorio/data/map-gen-settings.example.json "$CONFIG/map-gen-settings.json"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ ! -f $CONFIG/map-settings.json ]]; then
|
|
||||||
cp /opt/factorio/data/map-settings.example.json "$CONFIG/map-settings.json"
|
|
||||||
fi
|
|
||||||
|
|
||||||
exec /opt/factorio/bin/x64/factorio \
|
|
||||||
--port "$PORT" \
|
|
||||||
--start-server-load-scenario "$SERVER_SCENARIO" \
|
|
||||||
--server-settings "$CONFIG/server-settings.json" \
|
|
||||||
--server-banlist "$CONFIG/server-banlist.json" \
|
|
||||||
--server-whitelist "$CONFIG/server-whitelist.json" \
|
|
||||||
--use-server-whitelist \
|
|
||||||
--server-adminlist "$CONFIG/server-adminlist.json" \
|
|
||||||
--rcon-port "$RCON_PORT" \
|
|
||||||
--rcon-password "$(cat "$CONFIG/rconpw")" \
|
|
||||||
--server-id /factorio/config/server-id.json
|
|
@@ -1,27 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
set -eoux pipefail
|
|
||||||
|
|
||||||
if [[ -z ${1:-} ]]; then
|
|
||||||
echo "No argument supplied"
|
|
||||||
fi
|
|
||||||
|
|
||||||
SERVER_SCENARIO="$1"
|
|
||||||
mkdir -p "$SAVES"
|
|
||||||
mkdir -p "$CONFIG"
|
|
||||||
mkdir -p "$MODS"
|
|
||||||
mkdir -p "$SCENARIOS"
|
|
||||||
|
|
||||||
if [[ ! -f $CONFIG/server-settings.json ]]; then
|
|
||||||
cp /opt/factorio/data/server-settings.example.json "$CONFIG/server-settings.json"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ ! -f $CONFIG/map-gen-settings.json ]]; then
|
|
||||||
cp /opt/factorio/data/map-gen-settings.example.json "$CONFIG/map-gen-settings.json"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ ! -f $CONFIG/map-settings.json ]]; then
|
|
||||||
cp /opt/factorio/data/map-settings.example.json "$CONFIG/map-settings.json"
|
|
||||||
fi
|
|
||||||
|
|
||||||
exec /opt/factorio/bin/x64/factorio \
|
|
||||||
--scenario2map "$SERVER_SCENARIO"
|
|
@@ -1,95 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
set -eou pipefail
|
|
||||||
|
|
||||||
FACTORIO_VERSION=$1
|
|
||||||
MOD_DIR=$2
|
|
||||||
USERNAME=$3
|
|
||||||
TOKEN=$4
|
|
||||||
|
|
||||||
MOD_BASE_URL="https://mods.factorio.com"
|
|
||||||
|
|
||||||
print_step()
|
|
||||||
{
|
|
||||||
echo "$1"
|
|
||||||
}
|
|
||||||
|
|
||||||
print_success()
|
|
||||||
{
|
|
||||||
echo "$1"
|
|
||||||
}
|
|
||||||
|
|
||||||
print_failure()
|
|
||||||
{
|
|
||||||
echo "$1"
|
|
||||||
}
|
|
||||||
|
|
||||||
update_mod()
|
|
||||||
{
|
|
||||||
MOD_NAME="${1// /%20}"
|
|
||||||
|
|
||||||
print_step "Checking for update of mod $MOD_NAME..."
|
|
||||||
|
|
||||||
MOD_INFO_URL="$MOD_BASE_URL/api/mods/$MOD_NAME"
|
|
||||||
MOD_INFO_JSON=$(curl --silent "$MOD_INFO_URL")
|
|
||||||
|
|
||||||
if ! echo "$MOD_INFO_JSON" | jq -e .name >/dev/null; then
|
|
||||||
print_success " Custom mod not on $MOD_BASE_URL, skipped."
|
|
||||||
return 0
|
|
||||||
fi
|
|
||||||
|
|
||||||
MOD_INFO=$(echo "$MOD_INFO_JSON" | jq -j --arg version "$FACTORIO_VERSION" ".releases|reverse|map(select(.info_json.factorio_version as \$mod_version | \$version | startswith(\$mod_version)))[0]|.file_name, \";\", .download_url, \";\", .sha1")
|
|
||||||
|
|
||||||
MOD_FILENAME=$(echo "$MOD_INFO" | cut -f1 -d";")
|
|
||||||
MOD_URL=$(echo "$MOD_INFO" | cut -f2 -d";")
|
|
||||||
MOD_SHA1=$(echo "$MOD_INFO" | cut -f3 -d";")
|
|
||||||
|
|
||||||
if [[ $MOD_FILENAME == null ]]; then
|
|
||||||
print_failure " Not compatible with version"
|
|
||||||
return 0
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ -f $MOD_DIR/$MOD_FILENAME ]]; then
|
|
||||||
print_success " Already up-to-date."
|
|
||||||
return 0
|
|
||||||
fi
|
|
||||||
|
|
||||||
print_step "Downloading..."
|
|
||||||
FULL_URL="$MOD_BASE_URL$MOD_URL?username=$USERNAME&token=$TOKEN"
|
|
||||||
HTTP_STATUS=$(curl --silent -L -w "%{http_code}" -o "$MOD_DIR/$MOD_FILENAME" "$FULL_URL")
|
|
||||||
|
|
||||||
if [[ $HTTP_STATUS != 200 ]]; then
|
|
||||||
print_failure " Download failed: Code $HTTP_STATUS."
|
|
||||||
rm -f "$MOD_DIR/$MOD_FILENAME"
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ ! -f $MOD_DIR/$MOD_FILENAME ]]; then
|
|
||||||
print_failure " Downloaded file missing!"
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
if ! [[ $(sha1sum "$MOD_DIR/$MOD_FILENAME") =~ $MOD_SHA1 ]]; then
|
|
||||||
print_failure " SHA1 mismatch!"
|
|
||||||
rm -f "$MOD_DIR/$MOD_FILENAME"
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
print_success " Download complete."
|
|
||||||
|
|
||||||
for file in "$MOD_DIR/${MOD_NAME}_"*".zip"; do # wildcard does usually not work in quotes: https://unix.stackexchange.com/a/67761
|
|
||||||
if [[ $file != $MOD_DIR/$MOD_FILENAME ]]; then
|
|
||||||
print_success " Deleting old version: $file"
|
|
||||||
rm -f "$file"
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
|
|
||||||
return 0
|
|
||||||
}
|
|
||||||
|
|
||||||
if [[ -f $MOD_DIR/mod-list.json ]]; then
|
|
||||||
jq -r ".mods|map(select(.enabled))|.[].name" "$MOD_DIR/mod-list.json" | while read -r mod; do
|
|
||||||
if [[ $mod != base ]]; then
|
|
||||||
update_mod "$mod"
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
fi
|
|
126
CLAUDE.md
Normal file
126
CLAUDE.md
Normal file
@@ -0,0 +1,126 @@
|
|||||||
|
# CLAUDE.md
|
||||||
|
|
||||||
|
This file provides guidance to Claude Code (claude.ai/code) when working with code in this repository.
|
||||||
|
|
||||||
|
## Project Overview
|
||||||
|
|
||||||
|
This is a Docker image for running a Factorio headless server. It provides automated builds for multiple Factorio versions (stable and experimental) and supports both AMD64 and ARM64 architectures.
|
||||||
|
|
||||||
|
## Architecture
|
||||||
|
|
||||||
|
### Key Components
|
||||||
|
|
||||||
|
1. **Docker Image Build System**
|
||||||
|
- `build.py` - Unified Python script that builds both regular and rootless Docker images from `buildinfo.json`
|
||||||
|
- `docker/Dockerfile` - Main Dockerfile that creates the Factorio server image
|
||||||
|
- `docker/Dockerfile.rootless` - Dockerfile for rootless variant (runs as UID 1000)
|
||||||
|
- `buildinfo.json` - Contains version info, SHA256 checksums, and tags for all supported versions
|
||||||
|
- Supports multi-architecture builds (linux/amd64, linux/arm64) using Docker buildx
|
||||||
|
|
||||||
|
2. **Automated Updates**
|
||||||
|
- `update.sh` - Checks for new Factorio releases and updates `buildinfo.json`
|
||||||
|
- Updates README.md with new version tags
|
||||||
|
- Commits changes and tags releases automatically
|
||||||
|
- Run by GitHub Actions to keep images up-to-date
|
||||||
|
|
||||||
|
3. **Container Scripts**
|
||||||
|
- `docker/files/docker-entrypoint.sh` - Main entrypoint that configures and starts the server
|
||||||
|
- `docker/files/docker-update-mods.sh` - Updates mods on server start
|
||||||
|
- `docker/files/docker-dlc.sh` - Manages DLC (Space Age) activation
|
||||||
|
- `docker/files/scenario.sh` - Alternative entrypoint for launching scenarios
|
||||||
|
- `docker/files/players-online.sh` - Checks if players are online (for watchtower integration)
|
||||||
|
|
||||||
|
4. **RCON Client**
|
||||||
|
- `docker/rcon/` - C source for RCON client, built during Docker image creation
|
||||||
|
- Allows sending commands to the running server
|
||||||
|
|
||||||
|
## Common Development Commands
|
||||||
|
|
||||||
|
### Building Images
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Build regular images locally (single architecture)
|
||||||
|
python3 build.py
|
||||||
|
|
||||||
|
# Build rootless images only
|
||||||
|
python3 build.py --rootless
|
||||||
|
|
||||||
|
# Build both regular and rootless images
|
||||||
|
python3 build.py --both
|
||||||
|
|
||||||
|
# Build and push multi-architecture images (regular only)
|
||||||
|
python3 build.py --multiarch --push-tags
|
||||||
|
|
||||||
|
# Build and push both regular and rootless multi-architecture images
|
||||||
|
python3 build.py --multiarch --push-tags --both
|
||||||
|
```
|
||||||
|
|
||||||
|
### Running the Container
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Basic run command
|
||||||
|
docker run -d \
|
||||||
|
-p 34197:34197/udp \
|
||||||
|
-p 27015:27015/tcp \
|
||||||
|
-v /opt/factorio:/factorio \
|
||||||
|
--name factorio \
|
||||||
|
factoriotools/factorio
|
||||||
|
|
||||||
|
# Using docker-compose
|
||||||
|
docker-compose up -d
|
||||||
|
```
|
||||||
|
|
||||||
|
### Linting
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Lint Dockerfiles
|
||||||
|
./lint.sh
|
||||||
|
```
|
||||||
|
|
||||||
|
### Testing Updates
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Check for new Factorio versions and update buildinfo.json
|
||||||
|
./update.sh
|
||||||
|
```
|
||||||
|
|
||||||
|
## Key Configuration
|
||||||
|
|
||||||
|
### Environment Variables
|
||||||
|
- `LOAD_LATEST_SAVE` - Load the most recent save (default: true)
|
||||||
|
- `GENERATE_NEW_SAVE` - Generate a new save if none exists (default: false)
|
||||||
|
- `SAVE_NAME` - Name of the save file to load/create
|
||||||
|
- `UPDATE_MODS_ON_START` - Update mods before starting (requires USERNAME/TOKEN)
|
||||||
|
- `DLC_SPACE_AGE` - Enable/disable Space Age DLC (default: true)
|
||||||
|
- `PORT` - UDP port for game server (default: 34197)
|
||||||
|
- `RCON_PORT` - TCP port for RCON (default: 27015)
|
||||||
|
|
||||||
|
### Volume Structure
|
||||||
|
All data is stored in a single volume mounted at `/factorio`:
|
||||||
|
```
|
||||||
|
/factorio/
|
||||||
|
├── config/ # Server configuration files
|
||||||
|
├── mods/ # Game modifications
|
||||||
|
├── saves/ # Save games
|
||||||
|
├── scenarios/ # Scenario files
|
||||||
|
└── script-output/ # Script output directory
|
||||||
|
```
|
||||||
|
|
||||||
|
## Version Management
|
||||||
|
|
||||||
|
The project maintains compatibility with multiple Factorio versions:
|
||||||
|
- Latest experimental version gets the `latest` tag
|
||||||
|
- Latest stable version gets the `stable` tag
|
||||||
|
- Each version also gets specific tags (e.g., `2.0.55`, `2.0`, `2`)
|
||||||
|
- Legacy versions back to 0.12 are supported
|
||||||
|
|
||||||
|
Version updates are automated via GitHub Actions that run `update.sh` periodically.
|
||||||
|
|
||||||
|
## Testing Changes
|
||||||
|
|
||||||
|
1. Modify `buildinfo.json` to test specific versions
|
||||||
|
2. Run `python3 build.py` to build regular images locally
|
||||||
|
- Use `python3 build.py --rootless` for rootless images
|
||||||
|
- Use `python3 build.py --both` to build both variants
|
||||||
|
3. Test the container with your local data volume
|
||||||
|
4. For production changes, ensure `update.sh` handles version transitions correctly
|
408
PERMISSION_ISSUES_GUIDE.md
Normal file
408
PERMISSION_ISSUES_GUIDE.md
Normal file
@@ -0,0 +1,408 @@
|
|||||||
|
# Factorio Docker Permission Issues - Solutions and Workarounds
|
||||||
|
|
||||||
|
This document provides comprehensive solutions and workarounds for permission-related issues in the Factorio Docker container, based on detailed analysis of issues #558, #556, #555, #549, #496, #501, #492, and #420.
|
||||||
|
|
||||||
|
## Table of Contents
|
||||||
|
- [Root Cause Analysis](#root-cause-analysis)
|
||||||
|
- [Critical Prerequisites](#critical-prerequisites)
|
||||||
|
- [General Solutions](#general-solutions)
|
||||||
|
- [Platform-Specific Issues](#platform-specific-issues)
|
||||||
|
- [Docker System Requirements](#docker-system-requirements)
|
||||||
|
- [Advanced Troubleshooting](#advanced-troubleshooting)
|
||||||
|
- [Known Issues and Limitations](#known-issues-and-limitations)
|
||||||
|
|
||||||
|
## Root Cause Analysis
|
||||||
|
|
||||||
|
Based on detailed investigation by maintainer @Fank and community reports, the permission issues stem from:
|
||||||
|
|
||||||
|
1. **Container Architecture Issues**:
|
||||||
|
- No `USER` directive in Dockerfile despite creating a factorio user
|
||||||
|
- Container starts as root and performs recursive `chown` on every start
|
||||||
|
- The recursive `chown -R factorio:factorio /factorio` can be interrupted, leaving inconsistent permissions
|
||||||
|
- Dynamic UID/GID mapping using PUID/PGID environment variables adds complexity
|
||||||
|
|
||||||
|
2. **Rootless Docker Complications**:
|
||||||
|
- UID namespace remapping (e.g., container UID 845 → host UID 100844)
|
||||||
|
- Rootless Docker daemons cannot change ownership of bind-mounted volumes
|
||||||
|
- Different rootless implementations use different UID mappings
|
||||||
|
|
||||||
|
3. **Host System Dependencies**:
|
||||||
|
- Older Docker versions (especially pre-20.x) have permission handling bugs
|
||||||
|
- Some kernel versions have issues with user namespace operations
|
||||||
|
- SELinux and AppArmor can interfere with volume permissions
|
||||||
|
|
||||||
|
## Critical Prerequisites
|
||||||
|
|
||||||
|
### Update Your System First!
|
||||||
|
Many permission issues are caused by outdated system components:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# For Ubuntu/Debian
|
||||||
|
sudo apt-get update
|
||||||
|
sudo apt-get upgrade
|
||||||
|
|
||||||
|
# Specifically update Docker to 27.x or newer
|
||||||
|
# Follow: https://docs.docker.com/engine/install/ubuntu/#install-docker-engine
|
||||||
|
```
|
||||||
|
|
||||||
|
**Important**: Multiple users reported that updating Docker resolved their "Operation not permitted" errors.
|
||||||
|
|
||||||
|
## General Solutions
|
||||||
|
|
||||||
|
### Solution A: Pre-create Directories with Correct Permissions
|
||||||
|
```bash
|
||||||
|
# Create the directory structure
|
||||||
|
sudo mkdir -p /opt/factorio/{saves,mods,config,scenarios,script-output}
|
||||||
|
|
||||||
|
# Set ownership to factorio user (845:845)
|
||||||
|
sudo chown -R 845:845 /opt/factorio
|
||||||
|
|
||||||
|
# Set appropriate permissions (note the 'u+rwx' for write access)
|
||||||
|
sudo chmod -R u+rwx /opt/factorio
|
||||||
|
```
|
||||||
|
|
||||||
|
### Solution B: Use the Rootless Docker Image (Recommended)
|
||||||
|
The project now provides a rootless variant that runs as UID 1000, which avoids most permission issues:
|
||||||
|
```bash
|
||||||
|
docker run -d \
|
||||||
|
-p 34197:34197/udp \
|
||||||
|
-p 27015:27015/tcp \
|
||||||
|
-v /opt/factorio:/factorio \
|
||||||
|
--name factorio \
|
||||||
|
factoriotools/factorio:latest-rootless
|
||||||
|
```
|
||||||
|
|
||||||
|
**Benefits of rootless images**:
|
||||||
|
- No `chown` operations on startup
|
||||||
|
- No need to pre-create directories with specific permissions
|
||||||
|
- Works seamlessly with rootless Docker installations
|
||||||
|
- Avoids the recursive permission changes that can be interrupted
|
||||||
|
|
||||||
|
**Available rootless tags**:
|
||||||
|
- `latest-rootless`
|
||||||
|
- `stable-rootless`
|
||||||
|
- `2.0.55-rootless` (or any specific version with `-rootless` suffix)
|
||||||
|
|
||||||
|
## Platform-Specific Issues and Solutions
|
||||||
|
|
||||||
|
### NixOS with Rootless Docker
|
||||||
|
|
||||||
|
**Problem**: Permission denied errors when creating directories, even after setting ownership to 845:845. Files show ownership by UID 100844 instead of 845.
|
||||||
|
|
||||||
|
**Solutions**:
|
||||||
|
1. **Find and use your actual rootless Docker user ID**:
|
||||||
|
```bash
|
||||||
|
# Method 1: Check your user ID
|
||||||
|
id -u
|
||||||
|
|
||||||
|
# Method 2: Check existing Docker volumes for the UID Docker is using
|
||||||
|
ls -lan /path/to/other/docker/volumes
|
||||||
|
|
||||||
|
# Common rootless Docker UIDs:
|
||||||
|
# - 100999 (NixOS default)
|
||||||
|
# - 100844 (as reported in issue #558)
|
||||||
|
# - 1000 (some configurations)
|
||||||
|
|
||||||
|
# Apply the correct ownership
|
||||||
|
sudo chown -R 100999:100999 ./factorio
|
||||||
|
```
|
||||||
|
|
||||||
|
2. **Configure NixOS Docker properly**:
|
||||||
|
```nix
|
||||||
|
# In configuration.nix
|
||||||
|
virtualisation.docker.rootless = {
|
||||||
|
enable = true;
|
||||||
|
setSocketVariable = true;
|
||||||
|
};
|
||||||
|
```
|
||||||
|
|
||||||
|
3. **Port Mapping Issues**: Rootless Docker on NixOS has issues with userland-proxy that can cause random port assignments. Consider using host networking if possible.
|
||||||
|
|
||||||
|
### macOS with Colima
|
||||||
|
|
||||||
|
**Problem**: `copy_file` permission denied errors, even with correct ownership. Permission errors when running docker-dlc.sh.
|
||||||
|
|
||||||
|
**Solutions**:
|
||||||
|
1. **Set broader permissions before mounting**:
|
||||||
|
```bash
|
||||||
|
# Create directory structure
|
||||||
|
mkdir -p ./factorio-server/{saves,mods,config,scenarios}
|
||||||
|
|
||||||
|
# Set ownership AND permissions
|
||||||
|
sudo chown -R 845:845 ./factorio-server
|
||||||
|
sudo chmod -R 775 ./factorio-server
|
||||||
|
```
|
||||||
|
|
||||||
|
2. **Use Docker Desktop instead of Colima** if the issues persist, as it has better macOS integration
|
||||||
|
|
||||||
|
3. **Specify PUID/PGID explicitly**:
|
||||||
|
```yaml
|
||||||
|
environment:
|
||||||
|
- PUID=502 # Common macOS user ID
|
||||||
|
- PGID=20 # Common macOS staff group
|
||||||
|
```
|
||||||
|
|
||||||
|
### Windows
|
||||||
|
|
||||||
|
**Problem**: Cannot remove temporary locale files due to Windows-Linux permission translation. Errors like "Permission denied trying to remove /factorio/temp/currently-playing/locale/de".
|
||||||
|
|
||||||
|
**Solutions**:
|
||||||
|
1. **Use WSL2 backend** for Docker Desktop (required for proper Linux filesystem semantics)
|
||||||
|
|
||||||
|
2. **Store volumes in WSL2 filesystem** instead of Windows filesystem:
|
||||||
|
```bash
|
||||||
|
# Inside WSL2 terminal
|
||||||
|
mkdir -p ~/factorio
|
||||||
|
chmod -R 777 ~/factorio
|
||||||
|
```
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
# docker-compose.yml - use WSL2 path
|
||||||
|
volumes:
|
||||||
|
- ~/factorio:/factorio
|
||||||
|
```
|
||||||
|
|
||||||
|
3. **Avoid Windows drive mounts** (like `W:\docker\factorio`) as they have inherent permission translation issues
|
||||||
|
|
||||||
|
4. **Add :Z flag for SELinux context** (some Windows Docker setups benefit from this):
|
||||||
|
```yaml
|
||||||
|
volumes:
|
||||||
|
- ~/factorio:/factorio:Z
|
||||||
|
```
|
||||||
|
|
||||||
|
### Synology NAS
|
||||||
|
|
||||||
|
**Problem**: Permission denied when accessing mounted volumes. Error: "filesystem error: status: Permission denied [/factorio/saves]".
|
||||||
|
|
||||||
|
**Solutions**:
|
||||||
|
1. **Create and set permissions via SSH**:
|
||||||
|
```bash
|
||||||
|
# SSH into Synology
|
||||||
|
sudo mkdir -p /volume1/docker/factorio
|
||||||
|
sudo chown -R 845:845 /volume1/docker/factorio
|
||||||
|
sudo chmod -R u+rwx /volume1/docker/factorio # Important: u+rwx for write access
|
||||||
|
```
|
||||||
|
|
||||||
|
2. **Use the correct volume path in your container**:
|
||||||
|
```bash
|
||||||
|
docker run -d \
|
||||||
|
-p 34197:34197/udp \
|
||||||
|
-p 27015:27015/tcp \
|
||||||
|
-v /volume1/docker/factorio:/factorio \
|
||||||
|
--name factorio \
|
||||||
|
--restart=always \
|
||||||
|
factoriotools/factorio
|
||||||
|
```
|
||||||
|
|
||||||
|
3. **Check DSM Docker permissions** - ensure the Docker package has proper permissions to the shared folder
|
||||||
|
|
||||||
|
## Docker System Requirements
|
||||||
|
|
||||||
|
### Minimum Docker Version
|
||||||
|
Based on community reports, these Docker versions are known to work:
|
||||||
|
- **Docker 27.4.1** - Confirmed working
|
||||||
|
- **Docker 20.x+** - Generally stable
|
||||||
|
- **Docker 19.x and below** - Known permission issues
|
||||||
|
|
||||||
|
**Check your Docker version**:
|
||||||
|
```bash
|
||||||
|
docker --version
|
||||||
|
# If below 20.x, update immediately!
|
||||||
|
```
|
||||||
|
|
||||||
|
### "Operation not permitted" at Util.cpp:81
|
||||||
|
This specific error is often caused by:
|
||||||
|
1. **Outdated Docker version** - Update Docker first!
|
||||||
|
2. **Outdated kernel** - Run system updates
|
||||||
|
3. **Missing kernel capabilities** - Check Docker daemon configuration
|
||||||
|
|
||||||
|
## Docker Compose Best Practices
|
||||||
|
|
||||||
|
### Basic Configuration
|
||||||
|
```yaml
|
||||||
|
version: '3'
|
||||||
|
services:
|
||||||
|
factorio:
|
||||||
|
image: factoriotools/factorio:stable
|
||||||
|
container_name: factorio
|
||||||
|
ports:
|
||||||
|
- "34197:34197/udp"
|
||||||
|
- "27015:27015/tcp"
|
||||||
|
volumes:
|
||||||
|
- ./factorio:/factorio
|
||||||
|
restart: unless-stopped
|
||||||
|
stdin_open: true # For interactive console
|
||||||
|
tty: true
|
||||||
|
```
|
||||||
|
|
||||||
|
### Advanced Configuration for Permission Issues
|
||||||
|
```yaml
|
||||||
|
version: '3'
|
||||||
|
services:
|
||||||
|
factorio:
|
||||||
|
image: factoriotools/factorio:stable
|
||||||
|
container_name: factorio
|
||||||
|
ports:
|
||||||
|
- "34197:34197/udp"
|
||||||
|
- "27015:27015/tcp"
|
||||||
|
volumes:
|
||||||
|
- ./factorio:/factorio:Z # :Z for SELinux systems
|
||||||
|
restart: unless-stopped
|
||||||
|
# user: "845:845" # WARNING: This might break the entrypoint script
|
||||||
|
environment:
|
||||||
|
- PUID=845
|
||||||
|
- PGID=845
|
||||||
|
- UPDATE_MODS_ON_START=false # Disable if having permission issues
|
||||||
|
```
|
||||||
|
|
||||||
|
### Rootless Docker Configuration
|
||||||
|
```yaml
|
||||||
|
version: '3'
|
||||||
|
services:
|
||||||
|
factorio:
|
||||||
|
image: factoriotools/factorio:latest-rootless
|
||||||
|
container_name: factorio
|
||||||
|
ports:
|
||||||
|
- "34197:34197/udp"
|
||||||
|
- "27015:27015/tcp"
|
||||||
|
volumes:
|
||||||
|
- ./factorio:/factorio
|
||||||
|
restart: unless-stopped
|
||||||
|
environment:
|
||||||
|
- PUID=1000 # Rootless default
|
||||||
|
- PGID=1000
|
||||||
|
```
|
||||||
|
|
||||||
|
## Advanced Troubleshooting
|
||||||
|
|
||||||
|
### Step-by-Step Diagnosis
|
||||||
|
|
||||||
|
1. **Check Current Ownership**:
|
||||||
|
```bash
|
||||||
|
ls -lan ./factorio
|
||||||
|
# Look for UIDs like 845, 1000, 100844, 100999
|
||||||
|
```
|
||||||
|
|
||||||
|
2. **Verify Docker User Mapping**:
|
||||||
|
```bash
|
||||||
|
# Check what user the container is running as
|
||||||
|
docker exec factorio id
|
||||||
|
|
||||||
|
# Check file ownership inside container
|
||||||
|
docker exec factorio ls -lan /factorio
|
||||||
|
```
|
||||||
|
|
||||||
|
3. **Test Without Volume Mount** (isolates host permission issues):
|
||||||
|
```bash
|
||||||
|
docker run --rm -it factoriotools/factorio:stable
|
||||||
|
# If this works, the issue is with your host volume permissions
|
||||||
|
```
|
||||||
|
|
||||||
|
4. **Check Security Modules**:
|
||||||
|
```bash
|
||||||
|
# SELinux (Fedora, RHEL, CentOS)
|
||||||
|
getenforce
|
||||||
|
# If "Enforcing", try adding :Z to volume mount
|
||||||
|
|
||||||
|
# AppArmor (Ubuntu, Debian)
|
||||||
|
sudo apparmor_status | grep docker
|
||||||
|
```
|
||||||
|
|
||||||
|
5. **Debug the Entrypoint Script**:
|
||||||
|
```bash
|
||||||
|
# Run with debug output
|
||||||
|
docker run --rm -it \
|
||||||
|
-e DEBUG=true \
|
||||||
|
-v ./factorio:/factorio \
|
||||||
|
factoriotools/factorio:stable
|
||||||
|
```
|
||||||
|
|
||||||
|
### Common Error Messages and Solutions
|
||||||
|
|
||||||
|
| Error | Cause | Solution |
|
||||||
|
|-------|-------|----------|
|
||||||
|
| `Util.cpp:81: Operation not permitted` | Outdated Docker/kernel | Update Docker and system packages |
|
||||||
|
| `chown: Operation not permitted` | Rootless Docker | Use rootless Docker UID for ownership |
|
||||||
|
| `Permission denied [/factorio/saves]` | Wrong directory permissions | `chmod -R u+rwx` on host directory |
|
||||||
|
| `Couldn't create lock file /factorio/.lock` | Container can't write to volume | Check volume mount and permissions |
|
||||||
|
| `Map version X cannot be loaded` | Version mismatch | Use correct Docker image version |
|
||||||
|
|
||||||
|
## Known Issues and Limitations
|
||||||
|
|
||||||
|
### Interrupted chown Operations
|
||||||
|
The container performs `chown -R factorio:factorio /factorio` on every start. If the container is killed during this operation:
|
||||||
|
- Files will have inconsistent ownership
|
||||||
|
- Some files owned by 845, others by different UIDs
|
||||||
|
- Solution: Let the container complete startup before stopping
|
||||||
|
|
||||||
|
### Rootless Docker Port Mapping
|
||||||
|
**Issue #496**: Rootless Docker with userland-proxy causes random port assignments instead of the configured 34197.
|
||||||
|
- **Workaround**: Use host networking mode if possible
|
||||||
|
- **Note**: This is a Docker limitation, not specific to this image
|
||||||
|
|
||||||
|
### Map Version Compatibility
|
||||||
|
**Problem**: "Map version 2.0.23-0 cannot be loaded because it is higher than the game version".
|
||||||
|
|
||||||
|
**Solution**:
|
||||||
|
```bash
|
||||||
|
# Use a version that matches or exceeds your save
|
||||||
|
docker pull factoriotools/factorio:2.0.23
|
||||||
|
# Or always use latest for newest features
|
||||||
|
docker pull factoriotools/factorio:latest
|
||||||
|
```
|
||||||
|
|
||||||
|
## Recommended Approach
|
||||||
|
|
||||||
|
### For New Installations
|
||||||
|
1. **Update your system first** - Many issues are caused by old Docker versions
|
||||||
|
2. **Try the rootless image first** - It avoids most permission issues entirely
|
||||||
|
3. **Pre-create directories** with correct permissions if using the standard image
|
||||||
|
4. **Test without volumes** first to ensure the image works
|
||||||
|
|
||||||
|
### For Existing Installations with Issues
|
||||||
|
1. **Stop the container** and let it shut down cleanly
|
||||||
|
2. **Backup your data** before making changes
|
||||||
|
3. **Check Docker version** - update if below 20.x
|
||||||
|
4. **Fix permissions** using the platform-specific solution
|
||||||
|
5. **Consider rootless variant** for easier permission management
|
||||||
|
|
||||||
|
### Best Practices
|
||||||
|
- **Let the container start fully** before stopping (avoid interrupted chown)
|
||||||
|
- **Use named volumes** instead of bind mounts when possible
|
||||||
|
- **Monitor first startup** to ensure permissions are set correctly
|
||||||
|
- **Keep Docker updated** to avoid known bugs
|
||||||
|
|
||||||
|
## Community Solutions
|
||||||
|
|
||||||
|
### Proposed Improvements (from @Fank)
|
||||||
|
1. **Add USER directive** in Dockerfile after creating directories
|
||||||
|
2. **Optimize chown logic** to only run when ownership is wrong
|
||||||
|
3. **Implement fixuid** for better UID/GID mapping
|
||||||
|
4. **Add health checks** to ensure permissions are correct before starting
|
||||||
|
|
||||||
|
### Alternative Images
|
||||||
|
Some users have tried other Factorio Docker images (e.g., goofball222/factorio) but report the same Util.cpp:81 errors, suggesting this is a broader ecosystem issue related to Docker versions and system configurations.
|
||||||
|
|
||||||
|
## Quick Reference
|
||||||
|
|
||||||
|
| Platform | Common UID | Recommended Approach |
|
||||||
|
|----------|-----------|---------------------|
|
||||||
|
| Standard Docker | 845 | Update Docker, use `chown 845:845` |
|
||||||
|
| Rootless Docker (NixOS) | 100999, 100844 | Find actual UID, chown to that |
|
||||||
|
| macOS (Docker Desktop) | 502 (user), 20 (staff) | Use PUID/PGID env vars |
|
||||||
|
| Windows | N/A | Use WSL2 filesystem |
|
||||||
|
| Synology NAS | varies | Check DSM user, ensure Docker has folder access |
|
||||||
|
|
||||||
|
## Getting Help
|
||||||
|
|
||||||
|
If these solutions don't work:
|
||||||
|
1. **Update everything first** (Docker, kernel, system packages)
|
||||||
|
2. **Provide full details** when reporting issues:
|
||||||
|
- Docker version (`docker --version`)
|
||||||
|
- OS and version
|
||||||
|
- Full error messages
|
||||||
|
- Output of `ls -lan` on your volume
|
||||||
|
3. **Try the rootless image** as an alternative
|
||||||
|
4. **Check issue #558** for ongoing discussions
|
||||||
|
|
||||||
|
Remember: The vast majority of permission issues are resolved by updating Docker to version 20.x or newer!
|
212
README.md
212
README.md
@@ -1,15 +1,19 @@
|
|||||||
# Factorio [](https://travis-ci.org/factoriotools/factorio-docker)  [](https://hub.docker.com/r/factoriotools/factorio/) [](https://hub.docker.com/r/factoriotools/factorio/) [](https://hub.docker.com/r/factoriotools/factorio/) [](https://microbadger.com/images/factoriotools/factorio "Get your own image badge on microbadger.com")
|
# Factorio [](https://hub.docker.com/r/factoriotools/factorio/) [](https://hub.docker.com/r/factoriotools/factorio/) [](https://hub.docker.com/r/factoriotools/factorio/)
|
||||||
|
|
||||||
* `0.18.21`, `0.18`, `latest` [(0.18/Dockerfile)](https://github.com/factoriotools/factorio-docker/blob/master/0.18/Dockerfile)
|
> [!NOTE]
|
||||||
* `0.17.79`, `0.17`, `stable` [(0.17/Dockerfile)](https://github.com/factoriotools/factorio-docker/blob/master/0.17/Dockerfile)
|
> Support for ARM is experimental. Expect crashes and lag if you try to run this on a raspberry pi.
|
||||||
* `0.16.51`, `0.16` [(0.16/Dockerfile)](https://github.com/factoriotools/factorio-docker/blob/master/0.16/Dockerfile)
|
|
||||||
* `0.15.40`, `0.15` [(0.15/Dockerfile)](https://github.com/factoriotools/factorio-docker/blob/master/0.15/Dockerfile)
|
[中文](./README_zh_CN.md)
|
||||||
* `0.14.23`, `0.14` [(0.14/Dockerfile)](https://github.com/factoriotools/factorio-docker/blob/master/0.14/Dockerfile)
|
|
||||||
|
<!-- start autogeneration tags -->
|
||||||
|
* `latest, 2.0.64`
|
||||||
|
* `2, 2.0, 2.0.60, stable, stable-2.0.60`
|
||||||
|
<!-- end autogeneration tags -->
|
||||||
|
|
||||||
## Tag descriptions
|
## Tag descriptions
|
||||||
|
|
||||||
* `latest` - most up-to-date version (may be experimental).
|
* `latest` - most up-to-date version (may be experimental).
|
||||||
* `stable` - version declared stable on [factorio.com](https://www.factorio.com).
|
* `stable` - version declared stable on [factorio.com](https://www.factorio.com) ([FFF-435 Since 2.0 versions gets released as experimental first, once stable it will be marked as stable](https://factorio.com/blog/post/fff-435)).
|
||||||
* `0.x` - latest version in a branch.
|
* `0.x` - latest version in a branch.
|
||||||
* `0.x.y` - a specific version.
|
* `0.x.y` - a specific version.
|
||||||
* `0.x-z` - incremental fix for that version.
|
* `0.x-z` - incremental fix for that version.
|
||||||
@@ -38,7 +42,7 @@ sudo docker run -d \
|
|||||||
-p 27015:27015/tcp \
|
-p 27015:27015/tcp \
|
||||||
-v /opt/factorio:/factorio \
|
-v /opt/factorio:/factorio \
|
||||||
--name factorio \
|
--name factorio \
|
||||||
--restart=always \
|
--restart=unless-stopped \
|
||||||
factoriotools/factorio
|
factoriotools/factorio
|
||||||
```
|
```
|
||||||
|
|
||||||
@@ -83,11 +87,20 @@ docker run -d -it \
|
|||||||
docker attach factorio
|
docker attach factorio
|
||||||
```
|
```
|
||||||
|
|
||||||
|
### RCON (2.0.18+)
|
||||||
|
|
||||||
|
Alternativly (e.g. for scripting) the RCON connection can be used to send commands to the running factorio server.
|
||||||
|
This does not require the RCON connection to be exposed.
|
||||||
|
|
||||||
|
```shell
|
||||||
|
docker exec factorio rcon /h
|
||||||
|
```
|
||||||
|
|
||||||
### Upgrading
|
### Upgrading
|
||||||
|
|
||||||
Before upgrading backup the save. It's easy to make a save in the client.
|
Before upgrading backup the save. It's easy to make a save in the client.
|
||||||
|
|
||||||
Ensure `-v` was used to run the server so the save is outside of the Docker container. The `docker rm` command completely destroys the container, which includes the save if it isn't stored in an data volume.
|
Ensure `-v` was used to run the server so the save is outside of the Docker container. The `docker rm` command completely destroys the container, which includes the save if it isn't stored in a data volume.
|
||||||
|
|
||||||
Delete the container and refresh the image:
|
Delete the container and refresh the image:
|
||||||
|
|
||||||
@@ -118,25 +131,41 @@ sudo docker run -d \
|
|||||||
-p 34197:34197/udp \
|
-p 34197:34197/udp \
|
||||||
-p 27015:27015/tcp \
|
-p 27015:27015/tcp \
|
||||||
-v /opt/factorio:/factorio \
|
-v /opt/factorio:/factorio \
|
||||||
-e ENABLE_SERVER_LOAD_LATEST=false \
|
-e LOAD_LATEST_SAVE=false \
|
||||||
-e SAVE_NAME=replaceme \
|
-e SAVE_NAME=replaceme \
|
||||||
--name factorio \
|
--name factorio \
|
||||||
--restart=always \
|
--restart=unless-stopped \
|
||||||
factoriotools/factorio
|
factoriotools/factorio
|
||||||
```
|
```
|
||||||
|
|
||||||
To generate a new map set `ENABLE_GENERATE_NEW_MAP_SAVE=true` and specify `SAVE_NAME`:
|
To generate a new map set `GENERATE_NEW_SAVE=true` and specify `SAVE_NAME`:
|
||||||
|
|
||||||
```shell
|
```shell
|
||||||
sudo docker run -d \
|
sudo docker run -d \
|
||||||
-p 34197:34197/udp \
|
-p 34197:34197/udp \
|
||||||
-p 27015:27015/tcp \
|
-p 27015:27015/tcp \
|
||||||
-v /opt/factorio:/factorio \
|
-v /opt/factorio:/factorio \
|
||||||
-e ENABLE_SERVER_LOAD_LATEST=false \
|
-e LOAD_LATEST_SAVE=false \
|
||||||
-e ENABLE_GENERATE_NEW_MAP_SAVE=true \
|
-e GENERATE_NEW_SAVE=true \
|
||||||
-e SAVE_NAME=replaceme \
|
-e SAVE_NAME=replaceme \
|
||||||
--name factorio \
|
--name factorio \
|
||||||
--restart=always \
|
--restart=unless-stopped \
|
||||||
|
factoriotools/factorio
|
||||||
|
```
|
||||||
|
|
||||||
|
To generate a new map with a specific preset (e.g., death-world):
|
||||||
|
|
||||||
|
```shell
|
||||||
|
sudo docker run -d \
|
||||||
|
-p 34197:34197/udp \
|
||||||
|
-p 27015:27015/tcp \
|
||||||
|
-v /opt/factorio:/factorio \
|
||||||
|
-e LOAD_LATEST_SAVE=false \
|
||||||
|
-e GENERATE_NEW_SAVE=true \
|
||||||
|
-e SAVE_NAME=replaceme \
|
||||||
|
-e PRESET=death-world \
|
||||||
|
--name factorio \
|
||||||
|
--restart=unless-stopped \
|
||||||
factoriotools/factorio
|
factoriotools/factorio
|
||||||
```
|
```
|
||||||
|
|
||||||
@@ -146,6 +175,8 @@ Copy mods into the mods folder and restart the server.
|
|||||||
|
|
||||||
As of 0.17 a new environment variable was added ``UPDATE_MODS_ON_START`` which if set to ``true`` will cause the mods get to updated on server start. If set a valid [Factorio Username and Token](https://www.factorio.com/profile) must be supplied or else the server will not start. They can either be set as docker secrets, environment variables, or pulled from the server-settings.json file.
|
As of 0.17 a new environment variable was added ``UPDATE_MODS_ON_START`` which if set to ``true`` will cause the mods get to updated on server start. If set a valid [Factorio Username and Token](https://www.factorio.com/profile) must be supplied or else the server will not start. They can either be set as docker secrets, environment variables, or pulled from the server-settings.json file.
|
||||||
|
|
||||||
|
**Note:** When using the Space Age DLC, the built-in mods (`elevated-rails`, `quality`, and `space-age`) are automatically skipped during mod updates to prevent conflicts. These mods are included with the DLC and should not be downloaded separately.
|
||||||
|
|
||||||
### Scenarios
|
### Scenarios
|
||||||
|
|
||||||
If you want to launch a scenario from a clean start (not from a saved map) you'll need to start the docker image from an alternate entrypoint. To do this, use the example entrypoint file stored in the /factorio/entrypoints directory in the volume, and launch the image with the following syntax. Note that this is the normal syntax with the addition of the --entrypoint setting AND the additional argument at the end, which is the name of the Scenario in the Scenarios folder.
|
If you want to launch a scenario from a clean start (not from a saved map) you'll need to start the docker image from an alternate entrypoint. To do this, use the example entrypoint file stored in the /factorio/entrypoints directory in the volume, and launch the image with the following syntax. Note that this is the normal syntax with the addition of the --entrypoint setting AND the additional argument at the end, which is the name of the Scenario in the Scenarios folder.
|
||||||
@@ -156,7 +187,7 @@ docker run -d \
|
|||||||
-p 27015:27015/tcp \
|
-p 27015:27015/tcp \
|
||||||
-v /opt/factorio:/factorio \
|
-v /opt/factorio:/factorio \
|
||||||
--name factorio \
|
--name factorio \
|
||||||
--restart=always \
|
--restart=unless-stopped \
|
||||||
--entrypoint "/scenario.sh" \
|
--entrypoint "/scenario.sh" \
|
||||||
factoriotools/factorio \
|
factoriotools/factorio \
|
||||||
MyScenarioName
|
MyScenarioName
|
||||||
@@ -172,7 +203,7 @@ docker run -d \
|
|||||||
-p 27015:27015/tcp \
|
-p 27015:27015/tcp \
|
||||||
-v /opt/factorio:/factorio \
|
-v /opt/factorio:/factorio \
|
||||||
--name factorio \
|
--name factorio \
|
||||||
--restart=always \
|
--restart=unless-stopped \
|
||||||
--entrypoint "/scenario2map.sh" \
|
--entrypoint "/scenario2map.sh" \
|
||||||
factoriotools/factorio
|
factoriotools/factorio
|
||||||
MyScenarioName
|
MyScenarioName
|
||||||
@@ -203,8 +234,8 @@ Create file `config/server-banlist.json` and add the banlisted users.
|
|||||||
|
|
||||||
```json
|
```json
|
||||||
[
|
[
|
||||||
"bad_person",
|
"bad_person",
|
||||||
"other_bad_person"
|
"other_bad_person"
|
||||||
]
|
]
|
||||||
```
|
```
|
||||||
|
|
||||||
@@ -214,8 +245,8 @@ Create file `config/server-adminlist.json` and add the adminlisted users.
|
|||||||
|
|
||||||
```json
|
```json
|
||||||
[
|
[
|
||||||
"you",
|
"you",
|
||||||
"friend"
|
"friend"
|
||||||
]
|
]
|
||||||
```
|
```
|
||||||
|
|
||||||
@@ -246,6 +277,42 @@ The `server-settings.json` file may then contain the variable references like th
|
|||||||
"description": "${INSTANCE_DESC}",
|
"description": "${INSTANCE_DESC}",
|
||||||
```
|
```
|
||||||
|
|
||||||
|
### Environment Variables
|
||||||
|
|
||||||
|
These are the environment variables which can be specified at container run time.
|
||||||
|
|
||||||
|
| Variable Name | Description | Default | Available in |
|
||||||
|
|----------------------|----------------------------------------------------------------------|----------------|--------------|
|
||||||
|
| GENERATE_NEW_SAVE | Generate a new save if one does not exist before starting the server | false | 0.17+ |
|
||||||
|
| LOAD_LATEST_SAVE | Load latest when true. Otherwise load SAVE_NAME | true | 0.17+ |
|
||||||
|
| PORT | UDP port the server listens on | 34197 | 0.15+ |
|
||||||
|
| BIND | IP address (v4 or v6) the server listens on (IP\[:PORT]) | | 0.15+ |
|
||||||
|
| RCON_PORT | TCP port the rcon server listens on | 27015 | 0.15+ |
|
||||||
|
| SAVE_NAME | Name to use for the save file | _autosave1 | 0.17+ |
|
||||||
|
| PRESET | Map generation preset when GENERATE_NEW_SAVE is true | | 0.17+ |
|
||||||
|
| TOKEN | factorio.com token | | 0.17+ |
|
||||||
|
| UPDATE_MODS_ON_START | If mods should be updated before starting the server | | 0.17+ |
|
||||||
|
| USERNAME | factorio.com username | | 0.17+ |
|
||||||
|
| CONSOLE_LOG_LOCATION | Saves the console log to the specifies location | | |
|
||||||
|
| DLC_SPACE_AGE | Enables or disables the mods for DLC Space Age in mod-list.json[^1] | true | 2.0.8+ |
|
||||||
|
| MODS | Mod directory to use | /factorio/mods | 2.0.8+ |
|
||||||
|
|
||||||
|
**Note:** All environment variables are compared as strings
|
||||||
|
|
||||||
|
#### PRESET Values
|
||||||
|
|
||||||
|
The `PRESET` environment variable is used when generating a new map (when `GENERATE_NEW_SAVE=true`). It corresponds to Factorio's built-in map generation presets. Common values include:
|
||||||
|
|
||||||
|
- `default` - Normal settings
|
||||||
|
- `rich-resources` - Resources are more abundant
|
||||||
|
- `marathon` - Recipes and technologies are more expensive
|
||||||
|
- `death-world` - Biters are more aggressive and numerous
|
||||||
|
- `death-world-marathon` - Combines death-world and marathon settings
|
||||||
|
- `rail-world` - Resources are further apart, encouraging train usage
|
||||||
|
- `ribbon-world` - Map height is limited for a unique challenge
|
||||||
|
|
||||||
|
If PRESET is not specified or left empty, the map will be generated using the settings from `map-gen-settings.json` and `map-settings.json` without a preset.
|
||||||
|
|
||||||
## Container Details
|
## Container Details
|
||||||
|
|
||||||
The philosophy is to [keep it simple](http://wiki.c2.com/?KeepItSimple).
|
The philosophy is to [keep it simple](http://wiki.c2.com/?KeepItSimple).
|
||||||
@@ -280,16 +347,19 @@ The files in this volume should be owned by the factorio user, uid 845.
|
|||||||
|
|
||||||
[Docker Compose](https://docs.docker.com/compose/install/) is an easy way to run Docker containers.
|
[Docker Compose](https://docs.docker.com/compose/install/) is an easy way to run Docker containers.
|
||||||
|
|
||||||
First get a [docker-compose.yml](https://github.com/factoriotools/factorio-docker/blob/master/0.17/docker-compose.yml) file. To get it from this repository:
|
* docker-engine >= 1.10.0 is required
|
||||||
|
* docker-compose >=1.6.0 is required
|
||||||
|
|
||||||
|
First get a [docker-compose.yml](https://github.com/factoriotools/factorio-docker/blob/master/docker/docker-compose.yml) file. To get it from this repository:
|
||||||
|
|
||||||
```shell
|
```shell
|
||||||
git clone https://github.com/factoriotools/factorio-docker.git
|
git clone https://github.com/factoriotools/factorio-docker.git
|
||||||
cd docker_factorio_server/0.17
|
cd factorio-docker/docker
|
||||||
```
|
```
|
||||||
|
|
||||||
Or make your own:
|
Or make your own:
|
||||||
|
|
||||||
```shell
|
```yaml
|
||||||
version: '2'
|
version: '2'
|
||||||
services:
|
services:
|
||||||
factorio:
|
factorio:
|
||||||
@@ -311,14 +381,14 @@ sudo docker-compose up -d
|
|||||||
|
|
||||||
### Ports
|
### Ports
|
||||||
|
|
||||||
* `34197/udp` - Game server (required).
|
* `34197/udp` - Game server (required). This can be changed with the `PORT` environment variable.
|
||||||
* `27015/tcp` - RCON (optional).
|
* `27015/tcp` - RCON (optional).
|
||||||
|
|
||||||
## LAN Games
|
## LAN Games
|
||||||
|
|
||||||
Ensure the `lan` setting in server-settings.json is `true`.
|
Ensure the `lan` setting in server-settings.json is `true`.
|
||||||
|
|
||||||
```shell
|
```json
|
||||||
"visibility":
|
"visibility":
|
||||||
{
|
{
|
||||||
"public": false,
|
"public": false,
|
||||||
@@ -335,7 +405,7 @@ sudo docker run -d \
|
|||||||
-p 27015:27015/tcp \
|
-p 27015:27015/tcp \
|
||||||
-v /opt/factorio:/factorio \
|
-v /opt/factorio:/factorio \
|
||||||
--name factorio \
|
--name factorio \
|
||||||
--restart=always \
|
--restart=unless-stopped \
|
||||||
factoriotools/factorio
|
factoriotools/factorio
|
||||||
```
|
```
|
||||||
|
|
||||||
@@ -356,8 +426,89 @@ For LAN games the VM needs an internal IP in order for clients to connect. One w
|
|||||||
|
|
||||||
If you're looking for a simple way to deploy this to the Amazon Web Services Cloud, check out the [Factorio Server Deployment (CloudFormation) repository](https://github.com/m-chandler/factorio-spot-pricing). This repository contains a CloudFormation template that will get you up and running in AWS in a matter of minutes. Optionally it uses Spot Pricing so the server is very cheap, and you can easily turn it off when not in use.
|
If you're looking for a simple way to deploy this to the Amazon Web Services Cloud, check out the [Factorio Server Deployment (CloudFormation) repository](https://github.com/m-chandler/factorio-spot-pricing). This repository contains a CloudFormation template that will get you up and running in AWS in a matter of minutes. Optionally it uses Spot Pricing so the server is very cheap, and you can easily turn it off when not in use.
|
||||||
|
|
||||||
|
## Using a reverse proxy
|
||||||
|
|
||||||
|
If you need to use a reverse proxy you can use the following nginx snippet:
|
||||||
|
|
||||||
|
```
|
||||||
|
stream {
|
||||||
|
server {
|
||||||
|
listen 34197 udp reuseport;
|
||||||
|
proxy_pass my.upstream.host:34197;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
If your factorio host uses multiple IP addresses (very common with IPv6), you might additionally need to bind Factorio to a single IP (otherwise the UDP proxy might get confused with IP mismatches). To do that pass the `BIND` envvar to the container: `docker run --network=host -e BIND=2a02:1234::5678 ...`
|
||||||
|
|
||||||
|
## Rootless Docker Support (Experimental)
|
||||||
|
|
||||||
|
> **Note**: Rootless support is currently experimental. Please report any issues you encounter.
|
||||||
|
|
||||||
|
If you're experiencing permission issues or want better security, consider using the rootless images. These images are designed to work seamlessly with rootless Docker installations and avoid common permission problems.
|
||||||
|
|
||||||
|
### What are Rootless Images?
|
||||||
|
|
||||||
|
The rootless images differ from regular images in several ways:
|
||||||
|
- Run as UID 1000 (non-root) by default
|
||||||
|
- No dynamic UID/GID mapping (PUID/PGID not supported)
|
||||||
|
- No runtime chown operations
|
||||||
|
- All directories created with open permissions during build
|
||||||
|
|
||||||
|
### Rootless Image Tags
|
||||||
|
|
||||||
|
Each regular tag has a corresponding rootless version with the `-rootless` suffix:
|
||||||
|
- `latest-rootless` (experimental)
|
||||||
|
- `stable-rootless` (experimental)
|
||||||
|
- `2.0.55-rootless` (experimental)
|
||||||
|
|
||||||
|
### Quick Start with Rootless
|
||||||
|
|
||||||
|
```shell
|
||||||
|
docker run -d \
|
||||||
|
-p 34197:34197/udp \
|
||||||
|
-p 27015:27015/tcp \
|
||||||
|
-v ~/factorio:/factorio \
|
||||||
|
--name factorio \
|
||||||
|
--restart=unless-stopped \
|
||||||
|
factoriotools/factorio:stable-rootless
|
||||||
|
```
|
||||||
|
|
||||||
|
Key differences:
|
||||||
|
- No `chown` command needed
|
||||||
|
- No PUID/PGID environment variables
|
||||||
|
- Runs as UID 1000 by default
|
||||||
|
- No permission issues with volumes
|
||||||
|
|
||||||
|
### When to Use Rootless Images
|
||||||
|
|
||||||
|
Consider using rootless images if you:
|
||||||
|
- Are running Docker in rootless mode
|
||||||
|
- Experience permission issues with volume mounts
|
||||||
|
- Want to avoid containers running as root
|
||||||
|
- Don't need dynamic UID/GID mapping via PUID/PGID
|
||||||
|
|
||||||
|
### Limitations
|
||||||
|
|
||||||
|
- PUID/PGID environment variables are not supported
|
||||||
|
- Fixed to UID 1000 (may not match your host user)
|
||||||
|
- Experimental feature - may have undiscovered issues
|
||||||
|
|
||||||
## Troubleshooting
|
## Troubleshooting
|
||||||
|
|
||||||
|
### Permission Issues
|
||||||
|
|
||||||
|
If you're experiencing permission errors such as:
|
||||||
|
- `chown: Operation not permitted`
|
||||||
|
- `Permission denied [/factorio/saves]`
|
||||||
|
- `Util.cpp:81: Operation not permitted`
|
||||||
|
- Files owned by unexpected UIDs (like 100844 instead of 845)
|
||||||
|
|
||||||
|
Please refer to our comprehensive [Permission Issues Guide](./PERMISSION_ISSUES_GUIDE.md) for detailed solutions. Common fixes include:
|
||||||
|
- **Updating Docker** to version 20.x or newer (this resolves many issues)
|
||||||
|
- **Using the rootless image** variants (e.g., `factoriotools/factorio:stable-rootless`)
|
||||||
|
- **Setting correct ownership** for your specific Docker configuration
|
||||||
|
|
||||||
### My server is listed in the server browser, but nobody can connect
|
### My server is listed in the server browser, but nobody can connect
|
||||||
|
|
||||||
Check the logs. If there is the line `Own address is RIGHT IP:WRONG PORT`, then this could be caused by the Docker proxy. If the the IP and port is correct it's probably a port forwarding or firewall issue instead.
|
Check the logs. If there is the line `Own address is RIGHT IP:WRONG PORT`, then this could be caused by the Docker proxy. If the the IP and port is correct it's probably a port forwarding or firewall issue instead.
|
||||||
@@ -378,7 +529,12 @@ Use the `PORT` environment variable to start the server on the a different port,
|
|||||||
* [DBendit](https://github.com/DBendit/docker_factorio_server) - Coded admin list, ban list support and contributed version updates
|
* [DBendit](https://github.com/DBendit/docker_factorio_server) - Coded admin list, ban list support and contributed version updates
|
||||||
* [Zopanix](https://github.com/zopanix/docker_factorio_server) - Original Author
|
* [Zopanix](https://github.com/zopanix/docker_factorio_server) - Original Author
|
||||||
* [Rfvgyhn](https://github.com/Rfvgyhn/docker-factorio) - Coded randomly generated RCON password
|
* [Rfvgyhn](https://github.com/Rfvgyhn/docker-factorio) - Coded randomly generated RCON password
|
||||||
* [gnomus](https://github.com/gnomus/docker_factorio_server) - Coded wite listing support
|
* [gnomus](https://github.com/gnomus/docker_factorio_server) - Coded white listing support
|
||||||
* [bplein](https://github.com/bplein/docker_factorio_server) - Coded scenario support
|
* [bplein](https://github.com/bplein/docker_factorio_server) - Coded scenario support
|
||||||
* [jaredledvina](https://github.com/jaredledvina/docker_factorio_server) - Contributed version updates
|
* [jaredledvina](https://github.com/jaredledvina/docker_factorio_server) - Contributed version updates
|
||||||
* [carlbennett](https://github.com/carlbennett) - Contributed version updates and bugfixes
|
* [carlbennett](https://github.com/carlbennett) - Contributed version updates and bugfixes
|
||||||
|
|
||||||
|
[^1]: Space Age mods can also be individually enabled by using their name separated by space.
|
||||||
|
Example 1: Enable all by using `true`
|
||||||
|
Example 2: Enable all by listing the mod names `space-age elevated-rails quality`
|
||||||
|
Example 3: Enable only Elevated rails `elevated-rails`
|
||||||
|
490
README_zh_CN.md
Normal file
490
README_zh_CN.md
Normal file
@@ -0,0 +1,490 @@
|
|||||||
|
# Factorio [](https://hub.docker.com/r/factoriotools/factorio/) [](https://hub.docker.com/r/factoriotools/factorio/) [](https://hub.docker.com/r/factoriotools/factorio/)
|
||||||
|
|
||||||
|
> [!注意]
|
||||||
|
> ARM 架构支持是实验性的。如果你尝试在 Raspberry Pi 上运行,可能会遇到崩溃和延迟问题。
|
||||||
|
|
||||||
|
[English](./README.md)
|
||||||
|
|
||||||
|
<!-- start autogeneration tags -->
|
||||||
|
* `2`, `2.0`, `2.0.55`, `latest`, `stable`, `stable-2.0.55`
|
||||||
|
* `2.0.54`
|
||||||
|
* `2.0.53`
|
||||||
|
* `2.0.52`
|
||||||
|
* `2.0.51`
|
||||||
|
* `2.0.50`
|
||||||
|
* `2.0.49`
|
||||||
|
* `2.0.48`
|
||||||
|
* `2.0`, `2.0.47`, `stable-2.0.47`
|
||||||
|
* `2.0.46`
|
||||||
|
* `2.0.45`
|
||||||
|
* `2.0.44`
|
||||||
|
* `2.0`, `2.0.43`, `stable-2.0.43`
|
||||||
|
* `2.0`, `2.0.42`, `stable-2.0.42`
|
||||||
|
* `2.0`, `2.0.41`, `stable-2.0.41`
|
||||||
|
* `2.0.40`
|
||||||
|
* `2.0`, `2.0.39`, `stable-2.0.39`
|
||||||
|
* `2.0.38`
|
||||||
|
* `2.0.37`
|
||||||
|
* `2.0.36`
|
||||||
|
* `2.0.35`
|
||||||
|
* `2.0.34`
|
||||||
|
* `2.0.33`
|
||||||
|
* `2.0`, `2.0.32`, `stable-2.0.32`
|
||||||
|
* `2.0.31`
|
||||||
|
* `2.0`, `2.0.30`, `stable-2.0.30`
|
||||||
|
* `2.0.29`
|
||||||
|
* `2.0`, `2.0.28`, `stable-2.0.28`
|
||||||
|
* `2.0.27`
|
||||||
|
* `2.0.26`
|
||||||
|
* `2.0.25`
|
||||||
|
* `2.0.24`
|
||||||
|
* `2.0`, `2.0.23`, `stable-2.0.23`
|
||||||
|
* `2.0.22`
|
||||||
|
* `2.0`, `2.0.21`, `stable-2.0.21`
|
||||||
|
* `2.0`, `2.0.20`, `stable-2.0.20`
|
||||||
|
* `2.0.19`
|
||||||
|
* `2.0.18`
|
||||||
|
* `2.0.17`
|
||||||
|
* `2.0.16`
|
||||||
|
* `2.0`, `2.0.15`, `stable-2.0.15`
|
||||||
|
* `2.0`, `2.0.14`, `stable-2.0.14`
|
||||||
|
* `2.0`, `2.0.13`, `stable-2.0.13`
|
||||||
|
* `1`, `1.1`, `1.1.110`, `stable-1.1.110`
|
||||||
|
* `1.0`, `1.0.0`
|
||||||
|
* `0.17`, `0.17.79`
|
||||||
|
* `0.16`, `0.16.51`
|
||||||
|
* `0.15`, `0.15.40`
|
||||||
|
* `0.14`, `0.14.23`
|
||||||
|
* `0.13`, `0.13.20`
|
||||||
|
* `0.12`, `0.12.35`<!-- end autogeneration tags -->
|
||||||
|
|
||||||
|
## 标签描述
|
||||||
|
|
||||||
|
* `latest` - 最新版本(可能含有实验性功能)。
|
||||||
|
* `stable` - 在 [factorio.com](https://www.factorio.com) 上声明为稳定的版本([FFF-435 自 2.0 版本起,版本首先作为实验版发布,一旦稳定就会被标记为稳定版](https://factorio.com/blog/post/fff-435))。
|
||||||
|
* `0.x` - 某个分支上的最新版本。
|
||||||
|
* `0.x.y` - 具体的版本。
|
||||||
|
* `0.x-z` - 该版本的增量修复。
|
||||||
|
|
||||||
|
## 什么是 Factorio?
|
||||||
|
|
||||||
|
[Factorio](https://www.factorio.com) 是一款建造和维护工厂的游戏。
|
||||||
|
|
||||||
|
在游戏中,你将挖掘资源、研发科技、建设基础设施、自动化生产并与敌人战斗。发挥你的想象力来设计工厂,将简单的元素组合成巧妙的结构,运用管理技能保持其正常运转,最后保护它不受那些不太喜欢你的生物的侵害。
|
||||||
|
|
||||||
|
游戏非常稳定,并为建造大规模工厂进行了优化。你可以创建自己的地图,用 Lua 编写模组,或通过多人游戏与朋友一起游戏。
|
||||||
|
|
||||||
|
**注意**:这仅仅是服务端。完整游戏可在 [Factorio.com](https://www.factorio.com)、[Steam](https://store.steampowered.com/app/427520/)、[GOG.com](https://www.gog.com/game/factorio) 和 [Humble Bundle](https://www.humblebundle.com/store/factorio) 获得。
|
||||||
|
|
||||||
|
## 使用方法
|
||||||
|
|
||||||
|
### 快速开始
|
||||||
|
|
||||||
|
运行服务端以创建必要的文件夹结构和配置文件。在这个例子中,数据存储在 `/opt/factorio`。
|
||||||
|
|
||||||
|
```shell
|
||||||
|
sudo mkdir -p /opt/factorio
|
||||||
|
sudo chown 845:845 /opt/factorio
|
||||||
|
sudo docker run -d \
|
||||||
|
-p 34197:34197/udp \
|
||||||
|
-p 27015:27015/tcp \
|
||||||
|
-v /opt/factorio:/factorio \
|
||||||
|
--name factorio \
|
||||||
|
--restart=unless-stopped \
|
||||||
|
factoriotools/factorio
|
||||||
|
```
|
||||||
|
|
||||||
|
对于 Docker 新手,这里解释一下选项:
|
||||||
|
|
||||||
|
* `-d` - 以守护进程方式运行("分离"模式)。
|
||||||
|
* `-p` - 暴露端口。
|
||||||
|
* `-v` - 将本地文件系统的 `/opt/factorio` 挂载到容器中的 `/factorio`。
|
||||||
|
* `--restart` - 如果服务端崩溃或系统启动时重启服务端。
|
||||||
|
* `--name` - 将容器命名为 "factorio"(否则它会有一个有趣的随机名称)。
|
||||||
|
|
||||||
|
需要 `chown` 命令是因为在 0.16+ 版本中,出于安全原因,我们不再以 root 身份运行游戏服务端,而是以用户 ID 为 845 的 'factorio' 用户身份运行。因此主机必须允许该用户写入这些文件。
|
||||||
|
|
||||||
|
检查日志以查看发生了什么:
|
||||||
|
|
||||||
|
```shell
|
||||||
|
docker logs factorio
|
||||||
|
```
|
||||||
|
|
||||||
|
停止服务端:
|
||||||
|
|
||||||
|
```shell
|
||||||
|
docker stop factorio
|
||||||
|
```
|
||||||
|
|
||||||
|
现在在 `/opt/factorio/config` 文件夹中有一个 `server-settings.json` 文件。根据你的喜好修改它并重启服务端:
|
||||||
|
|
||||||
|
```shell
|
||||||
|
docker start factorio
|
||||||
|
```
|
||||||
|
|
||||||
|
尝试连接到服务端。如果无法正常工作,请检查日志。
|
||||||
|
|
||||||
|
### 控制台
|
||||||
|
|
||||||
|
要向服务端发出控制台命令,请使用 `-it` 以交互模式启动服务端。使用 `docker attach` 打开控制台,然后输入命令。
|
||||||
|
|
||||||
|
```shell
|
||||||
|
docker run -d -it \
|
||||||
|
--name factorio \
|
||||||
|
factoriotools/factorio
|
||||||
|
docker attach factorio
|
||||||
|
```
|
||||||
|
|
||||||
|
### RCON (2.0.18+)
|
||||||
|
|
||||||
|
或者(例如用于脚本),可以使用 RCON 连接向正在运行的 factorio 服务端发送命令。
|
||||||
|
这不需要暴露 RCON 连接。
|
||||||
|
|
||||||
|
```shell
|
||||||
|
docker exec factorio rcon /h
|
||||||
|
```
|
||||||
|
|
||||||
|
### 更新
|
||||||
|
|
||||||
|
在升级服务端之前,请备份存档。在客户端中制作存档很容易。
|
||||||
|
|
||||||
|
确保在运行服务端时使用了 `-v` 参数,这样存档就在 Docker 容器外部。`docker rm` 命令会完全销毁容器,如果存档没有存储在数据卷中,也会包括存档。
|
||||||
|
|
||||||
|
删除容器并刷新镜像:
|
||||||
|
|
||||||
|
```shell
|
||||||
|
docker stop factorio
|
||||||
|
docker rm factorio
|
||||||
|
docker pull factoriotools/factorio
|
||||||
|
```
|
||||||
|
|
||||||
|
现在像之前一样运行服务端。大约一分钟后,新版本的 Factorio 应该就会运行起来,完整保留存档和配置!
|
||||||
|
|
||||||
|
### 存档
|
||||||
|
|
||||||
|
服务端首次启动时会生成一个名为 `_autosave1.zip` 的新地图。使用 `/opt/factorio/config` 中的 `map-gen-settings.json` 和 `map-settings.json` 文件作为地图设置。在后续运行中使用最新的存档。
|
||||||
|
|
||||||
|
要加载旧存档,请停止服务端并运行命令 `touch oldsave.zip`。这会重置日期。然后重启服务端。另一个选择是删除除一个存档外的所有存档。
|
||||||
|
|
||||||
|
要生成新地图,请停止服务端,删除所有存档并重启服务端。
|
||||||
|
|
||||||
|
#### 直接指定存档(0.17.79-2+)
|
||||||
|
|
||||||
|
你可以通过一组环境变量配置服务端来指定要加载的特定存档:
|
||||||
|
|
||||||
|
要加载现有存档,请将 `SAVE_NAME` 设置为位于 `saves` 目录中的现有存档文件名,不包含 `.zip` 扩展名:
|
||||||
|
|
||||||
|
```shell
|
||||||
|
sudo docker run -d \
|
||||||
|
-p 34197:34197/udp \
|
||||||
|
-p 27015:27015/tcp \
|
||||||
|
-v /opt/factorio:/factorio \
|
||||||
|
-e LOAD_LATEST_SAVE=false \
|
||||||
|
-e SAVE_NAME=replaceme \
|
||||||
|
--name factorio \
|
||||||
|
--restart=unless-stopped \
|
||||||
|
factoriotools/factorio
|
||||||
|
```
|
||||||
|
|
||||||
|
要生成新地图,请设置 `GENERATE_NEW_SAVE=true` 并指定 `SAVE_NAME`:
|
||||||
|
|
||||||
|
```shell
|
||||||
|
sudo docker run -d \
|
||||||
|
-p 34197:34197/udp \
|
||||||
|
-p 27015:27015/tcp \
|
||||||
|
-v /opt/factorio:/factorio \
|
||||||
|
-e LOAD_LATEST_SAVE=false \
|
||||||
|
-e GENERATE_NEW_SAVE=true \
|
||||||
|
-e SAVE_NAME=replaceme \
|
||||||
|
--name factorio \
|
||||||
|
--restart=unless-stopped \
|
||||||
|
factoriotools/factorio
|
||||||
|
```
|
||||||
|
|
||||||
|
### Mods-模组
|
||||||
|
|
||||||
|
将模组复制到 mods 文件夹中并重启服务端。
|
||||||
|
|
||||||
|
从 0.17 版本开始,添加了一个新的环境变量 `UPDATE_MODS_ON_START`,如果设置为 `true`,将在服务端启动时更新模组。如果设置了此选项,必须提供有效的 [Factorio 用户名和令牌](https://www.factorio.com/profile),否则服务端将不会启动。它们可以设置为 docker secrets、环境变量,或从 server-settings.json 文件中获取。
|
||||||
|
|
||||||
|
### Scenarios-场景
|
||||||
|
|
||||||
|
如果你想从全新开始启动场景(而不是从保存的地图),你需要从备用入口点启动 docker 镜像。为此,请使用存储在卷中 /factorio/entrypoints 目录中的示例入口点文件,并使用以下语法启动镜像。请注意,这是正常语法,添加了 --entrypoint 设置和末尾的附加参数,这是 Scenarios 文件夹中场景的名称。
|
||||||
|
|
||||||
|
```shell
|
||||||
|
docker run -d \
|
||||||
|
-p 34197:34197/udp \
|
||||||
|
-p 27015:27015/tcp \
|
||||||
|
-v /opt/factorio:/factorio \
|
||||||
|
--name factorio \
|
||||||
|
--restart=unless-stopped \
|
||||||
|
--entrypoint "/scenario.sh" \
|
||||||
|
factoriotools/factorio \
|
||||||
|
MyScenarioName
|
||||||
|
```
|
||||||
|
|
||||||
|
### 将场景转换为常规地图
|
||||||
|
|
||||||
|
如果你想将场景导出为保存的地图,可以使用类似于上述场景用法的示例入口点。Factorio 将运行一次,将场景转换为 saves 目录中的保存地图。然后使用标准选项重启 docker 镜像将加载该地图,就像上述场景示例刚启动的场景一样。
|
||||||
|
|
||||||
|
```shell
|
||||||
|
docker run -d \
|
||||||
|
-p 34197:34197/udp \
|
||||||
|
-p 27015:27015/tcp \
|
||||||
|
-v /opt/factorio:/factorio \
|
||||||
|
--name factorio \
|
||||||
|
--restart=unless-stopped \
|
||||||
|
--entrypoint "/scenario2map.sh" \
|
||||||
|
factoriotools/factorio
|
||||||
|
MyScenarioName
|
||||||
|
```
|
||||||
|
|
||||||
|
### RCON
|
||||||
|
|
||||||
|
在 `rconpw` 文件中设置 RCON 密码。如果 `rconpw` 不存在,将生成随机密码。
|
||||||
|
|
||||||
|
要更改密码,请停止服务端,修改 `rconpw`,然后重启服务端。
|
||||||
|
|
||||||
|
要"禁用" RCON,请不要暴露端口 27015,即不使用 `-p 27015:27015/tcp` 启动服务端。RCON 仍在运行,但没有人可以连接到它。
|
||||||
|
|
||||||
|
|
||||||
|
### 白名单 (0.15.3+)
|
||||||
|
|
||||||
|
创建文件 `config/server-whitelist.json` 并添加白名单用户。
|
||||||
|
|
||||||
|
```json
|
||||||
|
[
|
||||||
|
"you",
|
||||||
|
"friend"
|
||||||
|
]
|
||||||
|
```
|
||||||
|
|
||||||
|
### 黑名单 (0.17.1+)
|
||||||
|
|
||||||
|
创建文件 `config/server-banlist.json` 并添加黑名单用户。
|
||||||
|
|
||||||
|
```json
|
||||||
|
[
|
||||||
|
"bad_person",
|
||||||
|
"other_bad_person"
|
||||||
|
]
|
||||||
|
```
|
||||||
|
|
||||||
|
### 管理员列表 (0.17.1+)
|
||||||
|
|
||||||
|
创建文件 `config/server-adminlist.json` 并添加管理员用户。
|
||||||
|
|
||||||
|
```json
|
||||||
|
[
|
||||||
|
"you",
|
||||||
|
"friend"
|
||||||
|
]
|
||||||
|
```
|
||||||
|
|
||||||
|
### 自定义配置文件 (0.17.x+)
|
||||||
|
|
||||||
|
开箱即用的 factorio 不支持配置文件中的环境变量。一个解决方法是使用 `envsubst`,它在启动期间从 docker-compose 中设置的环境变量动态生成配置文件:
|
||||||
|
|
||||||
|
替换 server-settings.json 的示例:
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
factorio_1:
|
||||||
|
image: factoriotools/factorio
|
||||||
|
ports:
|
||||||
|
- "34197:34197/udp"
|
||||||
|
volumes:
|
||||||
|
- /opt/factorio:/factorio
|
||||||
|
- ./server-settings.json:/server-settings.json
|
||||||
|
environment:
|
||||||
|
- INSTANCE_NAME=Your Instance's Name
|
||||||
|
- INSTANCE_DESC=Your Instance's Description
|
||||||
|
entrypoint: /bin/sh -c "mkdir -p /factorio/config && envsubst < /server-settings.json > /factorio/config/server-settings.json && exec /docker-entrypoint.sh"
|
||||||
|
```
|
||||||
|
|
||||||
|
然后 `server-settings.json` 文件可能包含这样的变量引用:
|
||||||
|
|
||||||
|
```json
|
||||||
|
"name": "${INSTANCE_NAME}",
|
||||||
|
"description": "${INSTANCE_DESC}",
|
||||||
|
```
|
||||||
|
|
||||||
|
### 环境变量
|
||||||
|
|
||||||
|
这些是可以在容器运行时指定的环境变量。
|
||||||
|
|
||||||
|
| 变量名 | 描述 | 默认值 | 可用版本 |
|
||||||
|
|---------------------|----------------------------------------------------------------|----------------|--------------|
|
||||||
|
| GENERATE_NEW_SAVE | 如果在启动服务端之前不存在存档,则生成新存档 | false | 0.17+ |
|
||||||
|
| LOAD_LATEST_SAVE | 为 true 时加载最新存档。否则加载 SAVE_NAME | true | 0.17+ |
|
||||||
|
| PORT | 服务端监听的 UDP 端口 | 34197 | 0.15+ |
|
||||||
|
| BIND | 服务端监听的 IP 地址(v4 或 v6)(IP\[:PORT]) | | 0.15+ |
|
||||||
|
| RCON_PORT | rcon 服务端监听的 TCP 端口 | 27015 | 0.15+ |
|
||||||
|
| SAVE_NAME | 存档文件使用的名称 | _autosave1 | 0.17+ |
|
||||||
|
| TOKEN | factorio.com 令牌 | | 0.17+ |
|
||||||
|
| UPDATE_MODS_ON_START| 是否在启动服务端之前更新模组 | | 0.17+ |
|
||||||
|
| USERNAME | factorio.com 用户名 | | 0.17+ |
|
||||||
|
| CONSOLE_LOG_LOCATION| 将控制台日志保存到指定位置 | | |
|
||||||
|
| DLC_SPACE_AGE | 在 mod-list.json 中启用或禁用 DLC Space Age 的模组[^1] | true | 2.0.8+ |
|
||||||
|
| MODS | 要使用的模组目录 | /factorio/mods | 2.0.8+ |
|
||||||
|
|
||||||
|
**注意**:所有环境变量都作为字符串进行比较
|
||||||
|
|
||||||
|
## 容器细节
|
||||||
|
|
||||||
|
理念是[保持简单](http://wiki.c2.com/?KeepItSimple)。
|
||||||
|
|
||||||
|
* 服务端应该自启动。
|
||||||
|
* 优先使用配置文件而不是环境变量。
|
||||||
|
* 使用一个数据卷。
|
||||||
|
|
||||||
|
### 数据卷
|
||||||
|
|
||||||
|
为了保持简单,容器使用挂载在 `/factorio` 的单个卷。此卷存储配置、模组和存档。
|
||||||
|
|
||||||
|
此卷中的文件应该由 factorio 用户拥有,uid 845。
|
||||||
|
|
||||||
|
```text
|
||||||
|
factorio
|
||||||
|
|-- config
|
||||||
|
| |-- map-gen-settings.json
|
||||||
|
| |-- map-settings.json
|
||||||
|
| |-- rconpw
|
||||||
|
| |-- server-adminlist.json
|
||||||
|
| |-- server-banlist.json
|
||||||
|
| |-- server-settings.json
|
||||||
|
| `-- server-whitelist.json
|
||||||
|
|-- mods
|
||||||
|
| `-- fancymod.zip
|
||||||
|
`-- saves
|
||||||
|
`-- _autosave1.zip
|
||||||
|
```
|
||||||
|
|
||||||
|
## Docker Compose
|
||||||
|
|
||||||
|
[Docker Compose](https://docs.docker.com/compose/install/) 是运行 Docker 容器的简便方法。
|
||||||
|
|
||||||
|
* 需要 docker-engine >= 1.10.0
|
||||||
|
* 需要 docker-compose >=1.6.0
|
||||||
|
|
||||||
|
首先获取一个 [docker-compose.yml](https://github.com/factoriotools/factorio-docker/blob/master/docker/docker-compose.yml) 文件。从此仓库获取:
|
||||||
|
|
||||||
|
```shell
|
||||||
|
git clone https://github.com/factoriotools/factorio-docker.git
|
||||||
|
cd factorio-docker/docker
|
||||||
|
```
|
||||||
|
|
||||||
|
或者创建你自己的:
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
version: '2'
|
||||||
|
services:
|
||||||
|
factorio:
|
||||||
|
image: factoriotools/factorio
|
||||||
|
ports:
|
||||||
|
- "34197:34197/udp"
|
||||||
|
- "27015:27015/tcp"
|
||||||
|
volumes:
|
||||||
|
- /opt/factorio:/factorio
|
||||||
|
```
|
||||||
|
|
||||||
|
现在 cd 到包含 docker-compose.yml 的目录并运行:
|
||||||
|
|
||||||
|
```shell
|
||||||
|
sudo mkdir -p /opt/factorio
|
||||||
|
sudo chown 845:845 /opt/factorio
|
||||||
|
sudo docker-compose up -d
|
||||||
|
```
|
||||||
|
|
||||||
|
### 端口
|
||||||
|
|
||||||
|
* `34197/udp` - 游戏服务端(必需)。可以通过 `PORT` 环境变量更改。
|
||||||
|
* `27015/tcp` - RCON(可选)。
|
||||||
|
|
||||||
|
## 局域网游戏
|
||||||
|
|
||||||
|
确保 server-settings.json 中的 `lan` 设置为 `true`。
|
||||||
|
|
||||||
|
```json
|
||||||
|
"visibility":
|
||||||
|
{
|
||||||
|
"public": false,
|
||||||
|
"lan": true
|
||||||
|
},
|
||||||
|
```
|
||||||
|
|
||||||
|
使用 `--network=host` 选项启动容器,以便客户端可以自动找到局域网游戏。参考快速入门来创建 `/opt/factorio` 目录。
|
||||||
|
|
||||||
|
```shell
|
||||||
|
sudo docker run -d \
|
||||||
|
--network=host \
|
||||||
|
-p 34197:34197/udp \
|
||||||
|
-p 27015:27015/tcp \
|
||||||
|
-v /opt/factorio:/factorio \
|
||||||
|
--name factorio \
|
||||||
|
--restart=unless-stopped \
|
||||||
|
factoriotools/factorio
|
||||||
|
```
|
||||||
|
|
||||||
|
## 部署到其他平台
|
||||||
|
|
||||||
|
### Vagrant
|
||||||
|
|
||||||
|
[Vagrant](https://www.vagrantup.com/) 是设置虚拟机(VM)运行 Docker 的简便方法。[Factorio Vagrant box 仓库](https://github.com/dtandersen/factorio-lan-vagrant)包含一个示例 Vagrantfile。
|
||||||
|
|
||||||
|
对于局域网游戏,VM 需要内部 IP 以便客户端连接。一种方法是使用公共网络。VM 使用 DHCP 获取 IP 地址。VM 还必须转发端口 34197。
|
||||||
|
|
||||||
|
```ruby
|
||||||
|
config.vm.network "public_network"
|
||||||
|
config.vm.network "forwarded_port", guest: 34197, host: 34197
|
||||||
|
```
|
||||||
|
|
||||||
|
### Amazon Web Services (AWS) 部署
|
||||||
|
|
||||||
|
如果你正在寻找一种简单的方法将此部署到 Amazon Web Services 云,请查看 [Factorio Server Deployment (CloudFormation) 仓库](https://github.com/m-chandler/factorio-spot-pricing)。此仓库包含一个 CloudFormation 模板,可以让你在几分钟内在 AWS 上运行起来。它可选择使用 Spot Pricing,因此服务端非常便宜,你可以在不使用时轻松关闭它。
|
||||||
|
|
||||||
|
## 使用反向代理
|
||||||
|
|
||||||
|
如果你需要使用反向代理,可以使用以下 nginx 片段:
|
||||||
|
|
||||||
|
```
|
||||||
|
stream {
|
||||||
|
server {
|
||||||
|
listen 34197 udp reuseport;
|
||||||
|
proxy_pass my.upstream.host:34197;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
如果你的 factorio 主机使用多个 IP 地址(IPv6 非常常见),你可能还需要将 Factorio 绑定到单个 IP(否则 UDP 代理可能会因 IP 不匹配而混乱)。要做到这一点,将 `BIND` 环境变量传递给容器:`docker run --network=host -e BIND=2a02:1234::5678 ...`
|
||||||
|
|
||||||
|
## 疑难解答
|
||||||
|
|
||||||
|
### 我的服务端在服务端浏览器中列出,但没有人可以连接
|
||||||
|
|
||||||
|
检查日志。如果有一行显示 `Own address is RIGHT IP:WRONG PORT`,那么这可能是由 Docker 代理引起的。如果 IP 和端口是正确的,可能是端口转发或防火墙问题。
|
||||||
|
|
||||||
|
默认情况下,Docker 通过代理路由流量。代理更改源 UDP 端口,因此检测到错误的端口。有关详细信息,请参阅论坛帖子 *[docker 托管服务端检测到错误端口](https://forums.factorio.com/viewtopic.php?f=49&t=35255)*。
|
||||||
|
|
||||||
|
为了修复错误端口,使用 `--userland-proxy=false` 开关启动 Docker 服务。Docker 将使用 iptables 规则而不是代理路由流量。将开关添加到 `DOCKER_OPTS` 环境变量或 Docker systemd 服务定义中的 `ExecStart`。具体情况因操作系统而异。
|
||||||
|
|
||||||
|
### 当我在 34197 之外的端口上运行服务端时,没有人可以从服务端浏览器连接
|
||||||
|
|
||||||
|
使用 `PORT` 环境变量在不同端口上启动服务端,例如 `docker run -e "PORT=34198"`。这会更改用于端口检测的数据包的源端口。`-p 34198:34197` 对于私人服务端工作正常,但服务端浏览器检测到错误的端口。
|
||||||
|
|
||||||
|
## 贡献者
|
||||||
|
|
||||||
|
* [dtandersen](https://github.com/dtandersen) - 维护者
|
||||||
|
* [Fank](https://github.com/Fankserver) - Factorio 监视程序的程序员,保持版本更新。
|
||||||
|
* [SuperSandro2000](https://github.com/supersandro2000) - CI 负责人,维护者和 Factorio 监视程序的运行者。贡献版本更新并编写了 Travis 脚本。
|
||||||
|
* [DBendit](https://github.com/DBendit/docker_factorio_server) - 编写了管理员列表、禁止列表支持并贡献版本更新
|
||||||
|
* [Zopanix](https://github.com/zopanix/docker_factorio_server) - 原作者
|
||||||
|
* [Rfvgyhn](https://github.com/Rfvgyhn/docker-factorio) - 编写了随机生成的 RCON 密码
|
||||||
|
* [gnomus](https://github.com/gnomus/docker_factorio_server) - 编写了白名单支持
|
||||||
|
* [bplein](https://github.com/bplein/docker_factorio_server) - 编写了场景支持
|
||||||
|
* [jaredledvina](https://github.com/jaredledvina/docker_factorio_server) - 贡献版本更新
|
||||||
|
* [carlbennett](https://github.com/carlbennett) - 贡献版本更新和错误修复
|
||||||
|
|
||||||
|
[^1]: Space Age 模组也可以通过使用它们的名称(用空格分隔)来单独启用。
|
||||||
|
示例 1:使用 `true` 启用所有
|
||||||
|
示例 2:通过列出模组名称启用所有 `space-age elevated-rails quality`
|
||||||
|
示例 3:仅启用 Elevated rails `elevated-rails`
|
157
build.py
Executable file
157
build.py
Executable file
@@ -0,0 +1,157 @@
|
|||||||
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
import os
|
||||||
|
import json
|
||||||
|
import subprocess
|
||||||
|
import shutil
|
||||||
|
import sys
|
||||||
|
import tempfile
|
||||||
|
import argparse
|
||||||
|
|
||||||
|
|
||||||
|
PLATFORMS = [
|
||||||
|
"linux/arm64",
|
||||||
|
"linux/amd64",
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
def create_builder(build_dir, builder_name, platform):
|
||||||
|
check_exists_command = ["docker", "buildx", "inspect", builder_name]
|
||||||
|
if subprocess.run(check_exists_command, stderr=subprocess.DEVNULL).returncode != 0:
|
||||||
|
create_command = ["docker", "buildx", "create", "--platform", platform, "--name", builder_name]
|
||||||
|
try:
|
||||||
|
subprocess.run(create_command, cwd=build_dir, check=True)
|
||||||
|
except subprocess.CalledProcessError:
|
||||||
|
print("Creating builder failed")
|
||||||
|
exit(1)
|
||||||
|
|
||||||
|
|
||||||
|
def build_and_push_multiarch(build_dir, build_args, push, builder_suffix=""):
|
||||||
|
builder_name = f"factoriotools{builder_suffix}-multiarch"
|
||||||
|
platform = ",".join(PLATFORMS)
|
||||||
|
create_builder(build_dir, builder_name, platform)
|
||||||
|
build_command = ["docker", "buildx", "build", "--platform", platform, "--builder", builder_name] + build_args
|
||||||
|
if push:
|
||||||
|
build_command.append("--push")
|
||||||
|
try:
|
||||||
|
subprocess.run(build_command, cwd=build_dir, check=True)
|
||||||
|
except subprocess.CalledProcessError:
|
||||||
|
print(f"Build and push of {builder_suffix or 'regular'} image failed")
|
||||||
|
exit(1)
|
||||||
|
|
||||||
|
|
||||||
|
def build_singlearch(build_dir, build_args, image_type="regular"):
|
||||||
|
build_command = ["docker", "build"] + build_args
|
||||||
|
try:
|
||||||
|
subprocess.run(build_command, cwd=build_dir, check=True)
|
||||||
|
except subprocess.CalledProcessError:
|
||||||
|
print(f"Build of {image_type} image failed")
|
||||||
|
exit(1)
|
||||||
|
|
||||||
|
|
||||||
|
def push_singlearch(tags):
|
||||||
|
for tag in tags:
|
||||||
|
try:
|
||||||
|
subprocess.run(["docker", "push", f"factoriotools/factorio:{tag}"],
|
||||||
|
check=True)
|
||||||
|
except subprocess.CalledProcessError:
|
||||||
|
print("Docker push failed")
|
||||||
|
exit(1)
|
||||||
|
|
||||||
|
|
||||||
|
def build_and_push(sha256, version, tags, push, multiarch, dockerfile="Dockerfile", builder_suffix=""):
|
||||||
|
build_dir = tempfile.mktemp()
|
||||||
|
shutil.copytree("docker", build_dir)
|
||||||
|
build_args = ["-f", dockerfile, "--build-arg", f"VERSION={version}", "--build-arg", f"SHA256={sha256}", "."]
|
||||||
|
for tag in tags:
|
||||||
|
build_args.extend(["-t", f"factoriotools/factorio:{tag}"])
|
||||||
|
|
||||||
|
image_type = "rootless" if "rootless" in dockerfile.lower() else "regular"
|
||||||
|
|
||||||
|
if multiarch:
|
||||||
|
build_and_push_multiarch(build_dir, build_args, push, builder_suffix)
|
||||||
|
else:
|
||||||
|
build_singlearch(build_dir, build_args, image_type)
|
||||||
|
if push:
|
||||||
|
push_singlearch(tags)
|
||||||
|
|
||||||
|
|
||||||
|
def login():
|
||||||
|
try:
|
||||||
|
username = os.environ["DOCKER_USERNAME"]
|
||||||
|
password = os.environ["DOCKER_PASSWORD"]
|
||||||
|
subprocess.run(["docker", "login", "-u", username, "-p", password], check=True)
|
||||||
|
except KeyError:
|
||||||
|
print("Username and password need to be given")
|
||||||
|
exit(1)
|
||||||
|
except subprocess.CalledProcessError:
|
||||||
|
print("Docker login failed")
|
||||||
|
exit(1)
|
||||||
|
|
||||||
|
|
||||||
|
def generate_rootless_tags(original_tags):
|
||||||
|
"""Generate rootless-specific tags from original tags"""
|
||||||
|
return [f"{tag}-rootless" for tag in original_tags]
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
parser = argparse.ArgumentParser(description='Build Factorio Docker images')
|
||||||
|
parser.add_argument('--push-tags', action='store_true', help='Push images to Docker Hub')
|
||||||
|
parser.add_argument('--multiarch', action='store_true', help='Build multi-architecture images')
|
||||||
|
parser.add_argument('--rootless', action='store_true', help='Build only rootless images')
|
||||||
|
parser.add_argument('--both', action='store_true', help='Build both regular and rootless images')
|
||||||
|
parser.add_argument('--only-stable-latest', action='store_true',
|
||||||
|
help='Build only stable and latest versions (for rootless by default)')
|
||||||
|
|
||||||
|
args = parser.parse_args()
|
||||||
|
|
||||||
|
# Default behavior: build regular images unless specified otherwise
|
||||||
|
build_regular = not args.rootless or args.both
|
||||||
|
build_rootless = args.rootless or args.both
|
||||||
|
|
||||||
|
with open(os.path.join(os.path.dirname(__file__), "buildinfo.json")) as file_handle:
|
||||||
|
builddata = json.load(file_handle)
|
||||||
|
|
||||||
|
if args.push_tags:
|
||||||
|
login()
|
||||||
|
|
||||||
|
# Filter versions if needed
|
||||||
|
versions_to_build = []
|
||||||
|
for version, buildinfo in sorted(builddata.items(), key=lambda item: item[0], reverse=True):
|
||||||
|
if args.only_stable_latest or (build_rootless and not build_regular):
|
||||||
|
# For rootless-only builds, default to stable/latest only
|
||||||
|
if "stable" in buildinfo["tags"] or "latest" in buildinfo["tags"]:
|
||||||
|
versions_to_build.append((version, buildinfo))
|
||||||
|
else:
|
||||||
|
versions_to_build.append((version, buildinfo))
|
||||||
|
|
||||||
|
# Build regular images
|
||||||
|
if build_regular:
|
||||||
|
print("Building regular images...")
|
||||||
|
for version, buildinfo in versions_to_build:
|
||||||
|
sha256 = buildinfo["sha256"]
|
||||||
|
tags = buildinfo["tags"]
|
||||||
|
build_and_push(sha256, version, tags, args.push_tags, args.multiarch)
|
||||||
|
|
||||||
|
# Build rootless images
|
||||||
|
if build_rootless:
|
||||||
|
print("Building rootless images...")
|
||||||
|
# For rootless, only build stable and latest unless building both
|
||||||
|
rootless_versions = []
|
||||||
|
if not build_regular or args.only_stable_latest:
|
||||||
|
for version, buildinfo in builddata.items():
|
||||||
|
if "stable" in buildinfo["tags"] or "latest" in buildinfo["tags"]:
|
||||||
|
rootless_versions.append((version, buildinfo))
|
||||||
|
else:
|
||||||
|
rootless_versions = versions_to_build
|
||||||
|
|
||||||
|
for version, buildinfo in rootless_versions:
|
||||||
|
sha256 = buildinfo["sha256"]
|
||||||
|
original_tags = buildinfo["tags"]
|
||||||
|
rootless_tags = generate_rootless_tags(original_tags)
|
||||||
|
build_and_push(sha256, version, rootless_tags, args.push_tags, args.multiarch,
|
||||||
|
dockerfile="Dockerfile.rootless", builder_suffix="-rootless")
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
main()
|
97
build.sh
97
build.sh
@@ -1,97 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
set -eoux pipefail
|
|
||||||
|
|
||||||
if [[ -z ${1:-} ]] && [[ -n ${CI:-} ]]; then
|
|
||||||
echo 'Usage: ./build.sh VERSION_SHORT'
|
|
||||||
exit 1
|
|
||||||
elif [[ ${CI:-} == true ]] || [[ -n ${1:-} ]]; then
|
|
||||||
VERSION_SHORT="$1"
|
|
||||||
else
|
|
||||||
VERSION_SHORT=$(find . -maxdepth 1 -type d | sort | tail -1 | grep -o "[[0-9]].[[0-9]]*")
|
|
||||||
EXTRA_TAG=latest
|
|
||||||
fi
|
|
||||||
|
|
||||||
cd "$VERSION_SHORT" || exit 1
|
|
||||||
|
|
||||||
VERSION=$(grep -oP '[0-9]+\.[0-9]+\.[0-9]+' Dockerfile | head -1)
|
|
||||||
DOCKER_REPO=factoriotools/factorio
|
|
||||||
|
|
||||||
if [[ ${TRAVIS_PULL_REQUEST:-} == true ]]; then
|
|
||||||
TAGS="$DOCKER_REPO:$TRAVIS_PULL_REQUEST_SLUG"
|
|
||||||
else
|
|
||||||
if [[ -n ${CI:-} ]]; then
|
|
||||||
# we are either on master or on a tag build
|
|
||||||
if [[ $TRAVIS_BRANCH == master ]] || [[ $TRAVIS_BRANCH == "$VERSION" ]]; then
|
|
||||||
TAGS="-t $DOCKER_REPO:$VERSION -t $DOCKER_REPO:$VERSION_SHORT"
|
|
||||||
# we are on an incremental build of a tag
|
|
||||||
elif [[ $VERSION == "${TRAVIS_BRANCH%-*}" ]]; then
|
|
||||||
TAGS="-t $DOCKER_REPO:$TRAVIS_BRANCH -t $DOCKER_REPO:$VERSION -t $DOCKER_REPO:$VERSION_SHORT"
|
|
||||||
# we build a other branch than master
|
|
||||||
elif [[ -n $TRAVIS_BRANCH ]]; then
|
|
||||||
TAGS="-t $DOCKER_REPO:$TRAVIS_BRANCH"
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
# we are not in CI and tag version and version short
|
|
||||||
TAGS="-t $DOCKER_REPO:$VERSION -t $DOCKER_REPO:$VERSION_SHORT"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ -n ${EXTRA_TAG:-} ]]; then
|
|
||||||
IFS=","
|
|
||||||
for TAG in $EXTRA_TAG; do
|
|
||||||
TAGS+=" -t $DOCKER_REPO:$TAG"
|
|
||||||
done
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ ${STABLE:-} == "$VERSION" ]]; then
|
|
||||||
TAGS+=" -t $DOCKER_REPO:stable"
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
# shellcheck disable=SC2068
|
|
||||||
eval docker build . ${TAGS[@]}
|
|
||||||
docker images
|
|
||||||
|
|
||||||
if [[ ${TRAVIS_BRANCH:-} ]]; then
|
|
||||||
TRAVIS_BRANCH_VERSION=${TRAVIS_BRANCH%-*}
|
|
||||||
fi
|
|
||||||
|
|
||||||
# only push when:
|
|
||||||
# latest changes where made in the folder corosponding to the version we build, we are on master and don#t build a PR.
|
|
||||||
if [[ $(dirname "$(git diff --name-only HEAD^)") =~ $VERSION_SHORT ]] && [[ ${TRAVIS_BRANCH:-} == master ]] && [[ $TRAVIS_PULL_REQUEST_BRANCH == "" ]] ||
|
|
||||||
# we build a tag and we are not on master
|
|
||||||
[[ $VERSION == "${TRAVIS_BRANCH_VERSION:-}" ]] && [[ ${TRAVIS_PULL_REQUEST_BRANCH:-} == "" ]] ||
|
|
||||||
# we are not in CI
|
|
||||||
[[ -z ${CI:-} ]]; then
|
|
||||||
|
|
||||||
if [[ ${CI:-} == true ]]; then
|
|
||||||
echo "$DOCKER_PASSWORD" | docker login -u "$DOCKER_USERNAME" --password-stdin
|
|
||||||
fi
|
|
||||||
|
|
||||||
# push a tag on a branch other than master
|
|
||||||
if [[ -n ${TRAVIS_BRANCH:-} ]] && [[ $VERSION != "${TRAVIS_BRANCH_VERSION:-}" ]] && [[ ${TRAVIS_BRANCH:-} != "master" ]]; then
|
|
||||||
docker push "$DOCKER_REPO:$TRAVIS_BRANCH"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# push an incremental tag
|
|
||||||
if [[ $VERSION == "${TRAVIS_BRANCH_VERSION:-}" ]]; then
|
|
||||||
docker push "$DOCKER_REPO:$TRAVIS_BRANCH"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ -n ${TRAVIS_TAG:-} ]] || [[ -z ${CI:-} ]]; then
|
|
||||||
docker push "$DOCKER_REPO:$VERSION"
|
|
||||||
docker push "$DOCKER_REPO:$VERSION_SHORT"
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ -n ${EXTRA_TAG:-} ]]; then
|
|
||||||
IFS=","
|
|
||||||
for TAG in $EXTRA_TAG; do
|
|
||||||
docker push "$DOCKER_REPO:$TAG"
|
|
||||||
done
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ ${STABLE:-} == "$VERSION" ]]; then
|
|
||||||
docker push "$DOCKER_REPO:stable"
|
|
||||||
fi
|
|
||||||
|
|
||||||
curl -X POST https://hooks.microbadger.com/images/factoriotools/factorio/TmmKGNp8jKcFqZvcJhTCIAJVluw=
|
|
||||||
fi
|
|
104
build_legacy.sh
Executable file
104
build_legacy.sh
Executable file
@@ -0,0 +1,104 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
set -eoux pipefail
|
||||||
|
|
||||||
|
if [[ -z ${1:-} && -n ${CI:-} ]]; then
|
||||||
|
echo 'Usage: ./build.sh VERSION_SHORT'
|
||||||
|
exit 1
|
||||||
|
elif [[ ${CI:-} == true || -n ${1:-} ]]; then
|
||||||
|
VERSION_SHORT="$1"
|
||||||
|
else
|
||||||
|
VERSION_SHORT=$(find . -maxdepth 1 -type d | sort | tail -1 | grep -o "[[0-9]].[[0-9]]*")
|
||||||
|
EXTRA_TAG=latest
|
||||||
|
fi
|
||||||
|
|
||||||
|
cd "$VERSION_SHORT" || exit 1
|
||||||
|
|
||||||
|
VERSION=$(grep -oP '[0-9]+\.[0-9]+\.[0-9]+' Dockerfile | head -1)
|
||||||
|
DOCKER_REPO=factoriotools/factorio
|
||||||
|
|
||||||
|
BRANCH=${GITHUB_REF#refs/*/}
|
||||||
|
|
||||||
|
if [[ -n ${GITHUB_BASE_REF:-} ]]; then
|
||||||
|
TAGS="-t $DOCKER_REPO:$GITHUB_BASE_REF"
|
||||||
|
else
|
||||||
|
if [[ -n ${CI:-} ]]; then
|
||||||
|
# we are either on master or on a tag build
|
||||||
|
if [[ ${BRANCH:-} == master || ${BRANCH:-} == "$VERSION" ]]; then
|
||||||
|
TAGS="-t $DOCKER_REPO:$VERSION -t $DOCKER_REPO:$VERSION_SHORT"
|
||||||
|
# we are on an incremental build of a tag
|
||||||
|
elif [[ $VERSION == "${BRANCH%-*}" ]]; then
|
||||||
|
TAGS="-t $DOCKER_REPO:$BRANCH -t $DOCKER_REPO:$VERSION -t $DOCKER_REPO:$VERSION_SHORT"
|
||||||
|
# we build a other branch than master and exclude dependabot branches from tags cause the / is not supported by docker
|
||||||
|
elif [[ -n ${BRANCH:-} && ! $BRANCH =~ "/" ]]; then
|
||||||
|
TAGS="-t $DOCKER_REPO:$BRANCH"
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
# we are not in CI and tag version and version short
|
||||||
|
TAGS="-t $DOCKER_REPO:$VERSION -t $DOCKER_REPO:$VERSION_SHORT"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ -n ${EXTRA_TAG:-} ]]; then
|
||||||
|
IFS=","
|
||||||
|
for TAG in $EXTRA_TAG; do
|
||||||
|
TAGS+=" -t $DOCKER_REPO:$TAG"
|
||||||
|
done
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ ${STABLE:-} == "$VERSION" ]]; then
|
||||||
|
TAGS+=" -t $DOCKER_REPO:stable"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Travis gets rate limited by Docker HUB.
|
||||||
|
if [[ ${CI:-} == true && -n ${DOCKER_PASSWORD:-} && -n ${DOCKER_USERNAME:-} ]]; then
|
||||||
|
echo "$DOCKER_PASSWORD" | docker login -u "$DOCKER_USERNAME" --password-stdin
|
||||||
|
fi
|
||||||
|
|
||||||
|
# shellcheck disable=SC2068
|
||||||
|
eval docker build . ${TAGS[@]:-}
|
||||||
|
docker images
|
||||||
|
|
||||||
|
# remove -1 from incremental tag
|
||||||
|
# eg before: 0.18.24-1, after 0.18.24
|
||||||
|
if [[ ${BRANCH:-} ]]; then
|
||||||
|
BRANCH_VERSION=${BRANCH%-*}
|
||||||
|
fi
|
||||||
|
|
||||||
|
# only push when:
|
||||||
|
# or we build a tag and we don't build a PR
|
||||||
|
if [[ $VERSION == "${BRANCH_VERSION:-}" && ${GITHUB_BASE_REF:-} == "" ]] ||
|
||||||
|
# or we are not in CI
|
||||||
|
[[ -z ${CI:-} ]]; then
|
||||||
|
|
||||||
|
if [[ ${CI:-} == true ]]; then
|
||||||
|
echo "$DOCKER_PASSWORD" | docker login -u "$DOCKER_USERNAME" --password-stdin
|
||||||
|
fi
|
||||||
|
|
||||||
|
# push a tag on a branch other than master except dependabot branches cause docker does not support /
|
||||||
|
if [[ -n ${BRANCH:-} && $VERSION != "${BRANCH_VERSION:-}" && ${BRANCH:-} != "master" && ! ${BRANCH:-} =~ "/" ]]; then
|
||||||
|
docker push "$DOCKER_REPO:$BRANCH"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# push an incremental tag
|
||||||
|
# eg 0.18.24-1
|
||||||
|
if [[ $VERSION == "${BRANCH_VERSION:-}" ]]; then
|
||||||
|
docker push "$DOCKER_REPO:$BRANCH"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# only push on tags or when manually running the script
|
||||||
|
if [[ -n ${BRANCH_VERSION:-} || -z ${CI:-} ]]; then
|
||||||
|
docker push "$DOCKER_REPO:$VERSION"
|
||||||
|
docker push "$DOCKER_REPO:$VERSION_SHORT"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ -n ${EXTRA_TAG:-} ]]; then
|
||||||
|
IFS=","
|
||||||
|
for TAG in $EXTRA_TAG; do
|
||||||
|
docker push "$DOCKER_REPO:$TAG"
|
||||||
|
done
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ ${STABLE:-} == "$VERSION" ]]; then
|
||||||
|
docker push "$DOCKER_REPO:stable"
|
||||||
|
fi
|
||||||
|
fi
|
19
buildinfo.json
Normal file
19
buildinfo.json
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
{
|
||||||
|
"2.0.60": {
|
||||||
|
"sha256": "69b5be1a867fd99524f9914dfee900a1ac386cf4e74c4a63768c05dc4d2b2b0b",
|
||||||
|
"tags": [
|
||||||
|
"stable",
|
||||||
|
"stable-2.0.60",
|
||||||
|
"2",
|
||||||
|
"2.0",
|
||||||
|
"2.0.60"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"2.0.64": {
|
||||||
|
"sha256": "729480a81fc3b3bd105bd0c92e108ee1caaac7840cc168cb32b0f9db8759a28a",
|
||||||
|
"tags": [
|
||||||
|
"latest",
|
||||||
|
"2.0.64"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
38
docker-compose.yml
Normal file
38
docker-compose.yml
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
version: "2"
|
||||||
|
services:
|
||||||
|
factorio:
|
||||||
|
container_name: factorio
|
||||||
|
image: factoriotools/factorio:stable
|
||||||
|
restart: unless-stopped
|
||||||
|
ports:
|
||||||
|
- "34197:34197/udp"
|
||||||
|
- "27015:27015/tcp"
|
||||||
|
volumes:
|
||||||
|
- ./data:/factorio
|
||||||
|
environment:
|
||||||
|
- UPDATE_MODS_ON_START=true
|
||||||
|
|
||||||
|
# Uncomment to enable autoupdate via watchtower
|
||||||
|
#labels:
|
||||||
|
# # Labels to allow watchtower autoupdate only if no players are online
|
||||||
|
# - com.centurylinklabs.watchtower.enable=true
|
||||||
|
# - com.centurylinklabs.watchtower.scope=factorio
|
||||||
|
# - com.centurylinklabs.watchtower.lifecycle.pre-update="/players-online.sh"
|
||||||
|
|
||||||
|
# Uncomment the following files to use watchtower for updating the factorio container
|
||||||
|
# Full documentation of watchtower: https://github.com/containrrr/watchtower
|
||||||
|
#watchtower:
|
||||||
|
# container_name: watchtower_factorio
|
||||||
|
# image: containrrr/watchtower
|
||||||
|
# restart: unless-stopped
|
||||||
|
# volumes:
|
||||||
|
# - /var/run/docker.sock:/var/run/docker.sock
|
||||||
|
# environment:
|
||||||
|
# # Only update containers which have the option 'watchtower.enable=true' set
|
||||||
|
# - WATCHTOWER_TIMEOUT=30s
|
||||||
|
# - WATCHTOWER_LABEL_ENABLE=true
|
||||||
|
# - WATCHTOWER_POLL_INTERVAL=3600
|
||||||
|
# - WATCHTOWER_LIFECYCLE_HOOKS=true
|
||||||
|
# - WATCHTOWER_SCOPE=factorio
|
||||||
|
# labels:
|
||||||
|
# - com.centurylinklabs.watchtower.scope=factorio
|
97
docker/Dockerfile
Executable file
97
docker/Dockerfile
Executable file
@@ -0,0 +1,97 @@
|
|||||||
|
# build rcon client
|
||||||
|
FROM debian:stable-slim AS rcon-builder
|
||||||
|
RUN apt-get -q update \
|
||||||
|
&& DEBIAN_FRONTEND=noninteractive apt-get -qy install build-essential
|
||||||
|
|
||||||
|
WORKDIR /src
|
||||||
|
COPY rcon/ /src
|
||||||
|
RUN make
|
||||||
|
|
||||||
|
# build factorio image
|
||||||
|
FROM debian:stable-slim
|
||||||
|
LABEL maintainer="https://github.com/factoriotools/factorio-docker"
|
||||||
|
|
||||||
|
ARG USER=factorio
|
||||||
|
ARG GROUP=factorio
|
||||||
|
ARG PUID=845
|
||||||
|
ARG PGID=845
|
||||||
|
ARG BOX64_VERSION=v0.2.4
|
||||||
|
|
||||||
|
# optionally utilize a built-in map-gen-preset (see data/base/prototypes/map-gen-presets
|
||||||
|
# if this is used, the preset will be used over any .json files supplied
|
||||||
|
# vanilla factorio provides the following presets:
|
||||||
|
# rich-resources, marathon, death-world, death-world-marathon, rail-world, ribbon-world, island
|
||||||
|
# a modded factorio example for using this:
|
||||||
|
# space-exploration
|
||||||
|
ARG PRESET
|
||||||
|
|
||||||
|
# number of retries that curl will use when pulling the headless server tarball
|
||||||
|
ARG CURL_RETRIES=8
|
||||||
|
|
||||||
|
ENV PORT=34197 \
|
||||||
|
RCON_PORT=27015 \
|
||||||
|
SAVES=/factorio/saves \
|
||||||
|
PRESET="$PRESET" \
|
||||||
|
CONFIG=/factorio/config \
|
||||||
|
MODS=/factorio/mods \
|
||||||
|
SCENARIOS=/factorio/scenarios \
|
||||||
|
SCRIPTOUTPUT=/factorio/script-output \
|
||||||
|
PUID="$PUID" \
|
||||||
|
PGID="$PGID" \
|
||||||
|
DLC_SPACE_AGE="true"
|
||||||
|
|
||||||
|
SHELL ["/bin/bash", "-eo", "pipefail", "-c"]
|
||||||
|
|
||||||
|
RUN apt-get -q update \
|
||||||
|
&& DEBIAN_FRONTEND=noninteractive apt-get -qy install ca-certificates curl jq pwgen xz-utils procps gettext-base file --no-install-recommends \
|
||||||
|
&& if [[ "$(uname -m)" == "aarch64" ]]; then \
|
||||||
|
echo "installing ARM compatability layer" \
|
||||||
|
&& DEBIAN_FRONTEND=noninteractive apt-get -qy install unzip --no-install-recommends \
|
||||||
|
&& curl -LO https://github.com/ptitSeb/box64/releases/download/${BOX64_VERSION}/box64-GENERIC_ARM-RelWithDebInfo.zip \
|
||||||
|
&& unzip box64-GENERIC_ARM-RelWithDebInfo.zip -d /bin \
|
||||||
|
&& rm -f box64-GENERIC_ARM-RelWithDebInfo.zip \
|
||||||
|
&& chmod +x /bin/box64; \
|
||||||
|
fi \
|
||||||
|
&& rm -rf /var/lib/apt/lists/*
|
||||||
|
|
||||||
|
RUN groupadd --system --gid "$PGID" "$GROUP" \
|
||||||
|
&& useradd --system --uid "$PUID" --gid "$PGID" --no-create-home --shell /bin/sh "$USER"
|
||||||
|
|
||||||
|
# version checksum of the archive to download
|
||||||
|
ARG VERSION
|
||||||
|
ARG SHA256
|
||||||
|
|
||||||
|
LABEL factorio.version=${VERSION}
|
||||||
|
|
||||||
|
ENV VERSION=${VERSION} \
|
||||||
|
SHA256=${SHA256}
|
||||||
|
|
||||||
|
RUN set -ox pipefail \
|
||||||
|
&& if [[ "${VERSION}" == "" ]]; then \
|
||||||
|
echo "build-arg VERSION is required" \
|
||||||
|
&& exit 1; \
|
||||||
|
fi \
|
||||||
|
&& if [[ "${SHA256}" == "" ]]; then \
|
||||||
|
echo "build-arg SHA256 is required" \
|
||||||
|
&& exit 1; \
|
||||||
|
fi \
|
||||||
|
&& archive="/tmp/factorio_headless_x64_$VERSION.tar.xz" \
|
||||||
|
&& mkdir -p /opt /factorio \
|
||||||
|
&& curl -sSL "https://www.factorio.com/get-download/$VERSION/headless/linux64" -o "$archive" --retry $CURL_RETRIES \
|
||||||
|
&& echo "$SHA256 $archive" | sha256sum -c \
|
||||||
|
|| (sha256sum "$archive" && file "$archive" && exit 1) \
|
||||||
|
&& tar xf "$archive" --directory /opt \
|
||||||
|
&& chmod ugo=rwx /opt/factorio \
|
||||||
|
&& rm "$archive" \
|
||||||
|
&& ln -s "$SCENARIOS" /opt/factorio/scenarios \
|
||||||
|
&& ln -s "$SAVES" /opt/factorio/saves \
|
||||||
|
&& mkdir -p /opt/factorio/config/ \
|
||||||
|
&& chown -R "$USER":"$GROUP" /opt/factorio /factorio
|
||||||
|
|
||||||
|
COPY files/*.sh /
|
||||||
|
COPY files/config.ini /opt/factorio/config/config.ini
|
||||||
|
COPY --from=rcon-builder /src/rcon /bin/rcon
|
||||||
|
|
||||||
|
VOLUME /factorio
|
||||||
|
EXPOSE $PORT/udp $RCON_PORT/tcp
|
||||||
|
ENTRYPOINT ["/docker-entrypoint.sh"]
|
91
docker/Dockerfile.rootless
Normal file
91
docker/Dockerfile.rootless
Normal file
@@ -0,0 +1,91 @@
|
|||||||
|
# build rcon client
|
||||||
|
FROM debian:stable-slim AS rcon-builder
|
||||||
|
RUN apt-get -q update \
|
||||||
|
&& DEBIAN_FRONTEND=noninteractive apt-get -qy install build-essential --no-install-recommends
|
||||||
|
|
||||||
|
WORKDIR /src
|
||||||
|
COPY rcon/ /src
|
||||||
|
RUN make
|
||||||
|
|
||||||
|
# build factorio image
|
||||||
|
FROM debian:stable-slim
|
||||||
|
LABEL maintainer="https://github.com/factoriotools/factorio-docker"
|
||||||
|
|
||||||
|
ARG BOX64_VERSION=v0.2.4
|
||||||
|
|
||||||
|
# optionally utilize a built-in map-gen-preset (see data/base/prototypes/map-gen-presets
|
||||||
|
ARG PRESET
|
||||||
|
|
||||||
|
# number of retries that curl will use when pulling the headless server tarball
|
||||||
|
ARG CURL_RETRIES=8
|
||||||
|
|
||||||
|
ENV PORT=34197 \
|
||||||
|
RCON_PORT=27015 \
|
||||||
|
SAVES=/factorio/saves \
|
||||||
|
PRESET="$PRESET" \
|
||||||
|
CONFIG=/factorio/config \
|
||||||
|
MODS=/factorio/mods \
|
||||||
|
SCENARIOS=/factorio/scenarios \
|
||||||
|
SCRIPTOUTPUT=/factorio/script-output \
|
||||||
|
DLC_SPACE_AGE="true"
|
||||||
|
|
||||||
|
SHELL ["/bin/bash", "-eo", "pipefail", "-c"]
|
||||||
|
|
||||||
|
RUN apt-get -q update \
|
||||||
|
&& DEBIAN_FRONTEND=noninteractive apt-get -qy install ca-certificates curl jq pwgen xz-utils procps gettext-base --no-install-recommends \
|
||||||
|
&& if [[ "$(uname -m)" == "aarch64" ]]; then \
|
||||||
|
echo "installing ARM compatability layer" \
|
||||||
|
&& DEBIAN_FRONTEND=noninteractive apt-get -qy install unzip --no-install-recommends \
|
||||||
|
&& curl -LO https://github.com/ptitSeb/box64/releases/download/${BOX64_VERSION}/box64-GENERIC_ARM-RelWithDebInfo.zip \
|
||||||
|
&& unzip box64-GENERIC_ARM-RelWithDebInfo.zip -d /bin \
|
||||||
|
&& rm -f box64-GENERIC_ARM-RelWithDebInfo.zip \
|
||||||
|
&& chmod +x /bin/box64; \
|
||||||
|
fi \
|
||||||
|
&& rm -rf /var/lib/apt/lists/*
|
||||||
|
|
||||||
|
# version checksum of the archive to download
|
||||||
|
ARG VERSION
|
||||||
|
ARG SHA256
|
||||||
|
|
||||||
|
LABEL factorio.version=${VERSION}
|
||||||
|
|
||||||
|
ENV VERSION=${VERSION} \
|
||||||
|
SHA256=${SHA256}
|
||||||
|
|
||||||
|
RUN set -ox pipefail \
|
||||||
|
&& if [[ "${VERSION}" == "" ]]; then \
|
||||||
|
echo "build-arg VERSION is required" \
|
||||||
|
&& exit 1; \
|
||||||
|
fi \
|
||||||
|
&& if [[ "${SHA256}" == "" ]]; then \
|
||||||
|
echo "build-arg SHA256 is required" \
|
||||||
|
&& exit 1; \
|
||||||
|
fi \
|
||||||
|
&& archive="/tmp/factorio_headless_x64_$VERSION.tar.xz" \
|
||||||
|
&& mkdir -p /opt /factorio \
|
||||||
|
&& curl -sSL "https://www.factorio.com/get-download/$VERSION/headless/linux64" -o "$archive" --retry $CURL_RETRIES \
|
||||||
|
&& echo "$SHA256 $archive" | sha256sum -c \
|
||||||
|
|| (sha256sum "$archive" && file "$archive" && exit 1) \
|
||||||
|
&& tar xf "$archive" --directory /opt \
|
||||||
|
&& chmod ugo=rwx /opt/factorio \
|
||||||
|
&& rm "$archive" \
|
||||||
|
&& ln -s "$SCENARIOS" /opt/factorio/scenarios \
|
||||||
|
&& ln -s "$SAVES" /opt/factorio/saves \
|
||||||
|
&& mkdir -p /opt/factorio/config/
|
||||||
|
|
||||||
|
COPY files/*.sh /
|
||||||
|
COPY files/docker-entrypoint-rootless.sh /docker-entrypoint.sh
|
||||||
|
COPY files/config.ini /opt/factorio/config/config.ini
|
||||||
|
COPY --from=rcon-builder /src/rcon /bin/rcon
|
||||||
|
|
||||||
|
# Make all scripts executable and set proper permissions for the factorio directory
|
||||||
|
RUN chmod +x /*.sh \
|
||||||
|
&& chmod -R 777 /opt/factorio /factorio
|
||||||
|
|
||||||
|
VOLUME /factorio
|
||||||
|
EXPOSE $PORT/udp $RCON_PORT/tcp
|
||||||
|
|
||||||
|
# Run as non-root user (UID 1000 is common for the first user in rootless containers)
|
||||||
|
USER 1000:1000
|
||||||
|
|
||||||
|
ENTRYPOINT ["/docker-entrypoint.sh"]
|
28
docker/docker-compose.yml
Normal file
28
docker/docker-compose.yml
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
version: '2'
|
||||||
|
services:
|
||||||
|
factorio:
|
||||||
|
build:
|
||||||
|
context: .
|
||||||
|
args:
|
||||||
|
# Check buildinfo.json for supported versions and SHAs
|
||||||
|
# https://github.com/factoriotools/factorio-docker/blob/master/buildinfo.json
|
||||||
|
- VERSION=2.0.60
|
||||||
|
- SHA256=69b5be1a867fd99524f9914dfee900a1ac386cf4e74c4a63768c05dc4d2b2b0b
|
||||||
|
ports:
|
||||||
|
- "34197:34197/udp"
|
||||||
|
- "27015:27015/tcp"
|
||||||
|
volumes:
|
||||||
|
- /opt/factorio:/factorio
|
||||||
|
# environment:
|
||||||
|
# - PUID=1000
|
||||||
|
# - PGID=1000
|
||||||
|
# - UPDATE_MODS_ON_START=true
|
||||||
|
# - USERNAME=FactorioUsername
|
||||||
|
# - TOKEN=FactorioToken
|
||||||
|
# - PORT=34198
|
||||||
|
# - PRESET=deathworld
|
||||||
|
# - ADDR=::1
|
||||||
|
# # Uncomment the following line to enable the use of the host's network stack,
|
||||||
|
# # which may be necessary for some setups like NAS or when using some proxy service like firewall rules.
|
||||||
|
# extra_hosts:
|
||||||
|
# - "host.docker.internal:host-gateway"
|
755
docker/files/config.ini
Normal file
755
docker/files/config.ini
Normal file
@@ -0,0 +1,755 @@
|
|||||||
|
; version=8
|
||||||
|
; This is INI file : https://en.wikipedia.org/wiki/INI_file#Format
|
||||||
|
; Semicolons (;) at the beginning of the line indicate a comment. Comment lines are ignored.
|
||||||
|
[path]
|
||||||
|
read-data=__PATH__executable__/../../data
|
||||||
|
write-data=__PATH__executable__/../..
|
||||||
|
|
||||||
|
[general]
|
||||||
|
locale=
|
||||||
|
|
||||||
|
[other]
|
||||||
|
; Options: true, false
|
||||||
|
; verbose-logging=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; log-saving-statistics=false
|
||||||
|
|
||||||
|
; autosave-interval=5
|
||||||
|
|
||||||
|
; autosave-slots=3
|
||||||
|
|
||||||
|
; In ticks
|
||||||
|
; minimum-latency-in-multiplayer=0
|
||||||
|
|
||||||
|
; In seconds
|
||||||
|
; multiplayer-initial-connection-timeout=10
|
||||||
|
|
||||||
|
; port=34197
|
||||||
|
|
||||||
|
; max-map-preview-chunk-side=64
|
||||||
|
|
||||||
|
; max-map-preview-threads=7
|
||||||
|
|
||||||
|
; In bytes
|
||||||
|
; max-multiplayer-script-reload-size=1048576
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; enable-steam-networking=true
|
||||||
|
|
||||||
|
; proxy=
|
||||||
|
|
||||||
|
; proxy-username=
|
||||||
|
|
||||||
|
; proxy-password=
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; check-updates=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; enable-experimental-updates=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; enable-new-mods=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; use-mod-settings-per-save=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; disable-minimal-mode=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; disable-blueprint-storage=false
|
||||||
|
|
||||||
|
; Disables tracking which mod created/changed what prototype. Mainly for faster startup during development.
|
||||||
|
;
|
||||||
|
; Options: true, false
|
||||||
|
; disable-prototype-history=false
|
||||||
|
|
||||||
|
; Print a warning for all prototype values that were not accessed.
|
||||||
|
;
|
||||||
|
; Options: true, false
|
||||||
|
; check-unused-prototype-data=false
|
||||||
|
|
||||||
|
; Cache data stage prototype data for faster startup. Experimental.
|
||||||
|
;
|
||||||
|
; Options: true, false
|
||||||
|
; cache-prototype-data=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; enable-razer-chroma-support=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; enable-logitech-led-support=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; enable-crash-log-uploading=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; enable-heap-validation=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; enable-threaded-message-pump=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; enable-taskbar-animation=true
|
||||||
|
|
||||||
|
; Does nothing on Windows
|
||||||
|
;
|
||||||
|
; Options: true, false
|
||||||
|
; non-blocking-saving=false
|
||||||
|
|
||||||
|
; Related to MacOS
|
||||||
|
;
|
||||||
|
; Options: true, false
|
||||||
|
; discard-mouse-events-when-accessibility-zoomed=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; enable-blueprint-storage-cloud-sync=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; force-enable-factorio-version-check=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; bring-window-to-top-on-click=true
|
||||||
|
|
||||||
|
; Options: fast, maximum
|
||||||
|
; multiplayer-compression-level=fast
|
||||||
|
|
||||||
|
; Options: none, fast, maximum
|
||||||
|
; autosave-compression-level=fast
|
||||||
|
|
||||||
|
; Socket to host RCON on when lauching MP server from the menu.
|
||||||
|
; local-rcon-socket=0.0.0.0:0
|
||||||
|
|
||||||
|
; Password for RCON when launching MP server from the menu.
|
||||||
|
; local-rcon-password=
|
||||||
|
|
||||||
|
|
||||||
|
[interface]
|
||||||
|
; Options: true, false
|
||||||
|
; automatic-ui-scale=true
|
||||||
|
|
||||||
|
; custom-ui-scale=1.000000
|
||||||
|
|
||||||
|
; tooltip-delay=0.040000
|
||||||
|
|
||||||
|
; entity-tooltip-delay=0.000000
|
||||||
|
|
||||||
|
; tooltip-offset=20
|
||||||
|
|
||||||
|
; output-console-delay=1200
|
||||||
|
|
||||||
|
; train-stop-label-angle=0.085526
|
||||||
|
|
||||||
|
; active-quick-bars=2
|
||||||
|
|
||||||
|
; shortcut-bar-rows=2
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; autosort-inventory=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; research-finished-stops-game=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; use-item-groups=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; use-item-subgroups=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; use-version-filter-in-browse-games-gui=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; use-version-filter-in-install-mods-gui=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; play-sound-for-chat-messages=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; fuzzy-search-enabled=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; pick-ghost-cursor=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-minimap=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-tips-and-tricks=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-tutorial-notifications=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-turret-radius-when-blueprinting=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-item-labels-in-cursor=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-rail-block-visualization=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-missing-logistic-network-icon=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-interaction-indications=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-grid-when-paused=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-inserter-arrows-when-selected=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-inserter-arrows-when-detailed-info-is-on=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-pump-arrows-when-detailed-info-is-on=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-mining-drill-arrows-when-detailed-info-is-on=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-combinator-settings-when-detailed-info-is-on=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; entity-tooltip-on-the-side=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-mod-owners-in-tooltips=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-descriptions-in-tooltips=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-total-raw-in-recipe-tooltips=true
|
||||||
|
|
||||||
|
; debug-font-size=18
|
||||||
|
|
||||||
|
; train-visualization-length=5
|
||||||
|
|
||||||
|
|
||||||
|
[sound]
|
||||||
|
; master-volume=0.800000
|
||||||
|
|
||||||
|
; music-volume=0.500000
|
||||||
|
|
||||||
|
; game-effects-volume=0.700000
|
||||||
|
|
||||||
|
; gui-effects-volume=0.600000
|
||||||
|
|
||||||
|
; walking-sound-volume=0.250000
|
||||||
|
|
||||||
|
; environment-sounds-volume=0.550000
|
||||||
|
|
||||||
|
; alerts-volume=0.500000
|
||||||
|
|
||||||
|
; wind-volume=0.350000
|
||||||
|
|
||||||
|
; audible-distance=40.000000
|
||||||
|
|
||||||
|
; environment-audible-distance=30.000000
|
||||||
|
|
||||||
|
; maximum-environment-sounds=50
|
||||||
|
|
||||||
|
; active-gui-volume-modifier=0.800000
|
||||||
|
|
||||||
|
; active-gui-environment-volume-modifier=0.400000
|
||||||
|
|
||||||
|
; The maximum volume allowed for any sound.
|
||||||
|
; maximum-volume=2.000000
|
||||||
|
|
||||||
|
; ambient-music-pause-mean-seconds=45.000000
|
||||||
|
|
||||||
|
; ambient-music-pause-variance-seconds=30.000000
|
||||||
|
|
||||||
|
; Options: main-tracks-only, interleave-main-tracks-with-interludes, randomize-all
|
||||||
|
; ambient-music-mode=interleave-main-tracks-with-interludes
|
||||||
|
|
||||||
|
; zoom-audible-distance-coefficient=0.500000
|
||||||
|
|
||||||
|
; zoom-volume-coefficient=0.750000
|
||||||
|
|
||||||
|
|
||||||
|
[map-view]
|
||||||
|
; Options: true, false
|
||||||
|
; show-logistic-network=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-electric-network=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-turret-range=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-pollution=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-networkless-logistic-members=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-train-station-names=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-player-names=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-non-standard-map-info=false
|
||||||
|
|
||||||
|
|
||||||
|
[debug]
|
||||||
|
; force=enemy
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; capture-perf-statistics=false
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-fps=debug
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-detailed-info=debug
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-time-usage=debug
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-gpu-time-usage=debug
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-sprite-counts=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-lua-object-statistics=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-heat-buffer-info=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-multiplayer-waiting-icon=debug
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-multiplayer-statistics=debug
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-multiplayer-selection-rectangles=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-debug-info-in-tooltips=debug
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; hide-mod-guis=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-tile-grid=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-collision-rectangles=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-selection-rectangles=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-render-rectangles=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-entity-positions=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-entity-velocities=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-selected-entity-advanced-tiles=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-selected-input-transport-belts=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-paths=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-path-requests=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-next-waypoint-bb=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-target=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-unit-group-info=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-unit-behavior-info=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-pathfinder-fringe=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-path-cache=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-path-cache-paths=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-rail-paths=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-rolling-stock-count=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-rail-connections=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-rail-joints=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-rail-signal-states=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-rail-segment-collision-boxes=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-train-stop-point=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-train-braking-distance=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-train-signals=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-train-repathing=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-network-connected-entities=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-circuit-network-numbers=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-energy-sources-networks=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-active-state=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-wakeup-lists=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-transport-lines=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-transport-line-gaps=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-pollution-values=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-active-entities-on-chunk-counts=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-active-chunks=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-polluted-chunks=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; hide-chart-tags=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-enemy-expansion-candidate-chunks=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-enemy-expansion-candidate-chunk-values=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-bad-attack-chunks=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-tile-variations=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-raw-tile-transitions=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-fluid-box-fluid-info=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-environment-sound-info=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-environment-sound-area=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-selected-entity-audible-range=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-recently-played-sound-info=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-logistic-robot-targets=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-logistic-robots-on-map=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-recipe-icons-on-map=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-player-robots=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-fire-info=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-sticker-info=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-decorative-names=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-decorative-collision-rectangles=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; allow-increased-zoom=never
|
||||||
|
|
||||||
|
; Options: always, debug, never
|
||||||
|
; show-chunk-components=never
|
||||||
|
|
||||||
|
|
||||||
|
[multiplayer-lobby]
|
||||||
|
; name=
|
||||||
|
|
||||||
|
; description=
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; visibility-public=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; visibility-steam=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; visibility-lan=true
|
||||||
|
|
||||||
|
; max-players=0
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; ignore-player-limit-when-returning=false
|
||||||
|
|
||||||
|
; max-upload-in-kilobytes-per-second=0
|
||||||
|
|
||||||
|
; max-upload-slots=5
|
||||||
|
|
||||||
|
; password=
|
||||||
|
|
||||||
|
; tag-list=
|
||||||
|
|
||||||
|
; afk-auto-kick=0
|
||||||
|
|
||||||
|
; Options: true, false, admins-only
|
||||||
|
; allowed-commands=admins-only
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; only-admins-can-pause=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; autosave-only-on-server=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; non-blocking-saving=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; verify-user-identity=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; enable-whitelist=false
|
||||||
|
|
||||||
|
|
||||||
|
[graphics]
|
||||||
|
; lights-render-quality=0.250000
|
||||||
|
|
||||||
|
; Default preferred display index should force finding primary monitor
|
||||||
|
; preferred-display-index=255
|
||||||
|
|
||||||
|
; screenshots-threads-count=8
|
||||||
|
|
||||||
|
; cache-sprite-atlas-count=1
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; cache-sprite-atlas=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; compress-sprite-atlas-cache=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; texture-streaming=true
|
||||||
|
|
||||||
|
; streamed-atlas-physical-vram-size=0
|
||||||
|
|
||||||
|
; sprite-vertex-buffer-size=1048576
|
||||||
|
|
||||||
|
; max-texture-size=0
|
||||||
|
|
||||||
|
; max-threads=8
|
||||||
|
|
||||||
|
; 'low' and 'very-low' options are deprecated and will be migrated to 'normal'
|
||||||
|
;
|
||||||
|
; Options: high, normal, low, very-low
|
||||||
|
; graphics-quality=normal
|
||||||
|
|
||||||
|
; brightness=0
|
||||||
|
|
||||||
|
; contrast=0
|
||||||
|
|
||||||
|
; saturation=100
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; full-screen=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; minimize-on-focus-loss=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-smoke=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-clouds=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-decoratives=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-particles=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-item-shadows=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-inserter-shadows=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-animated-water=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; show-tree-distortion=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; force-opengl=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; v-sync=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; high-quality-animations=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; high-quality-shadows=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; high-quality-terrain=true
|
||||||
|
|
||||||
|
; Minimum number of turrets required to turn on the turret range overdraw optimization
|
||||||
|
; turret-overdraw-minimum-count=4
|
||||||
|
|
||||||
|
; Scale at which the turret range overdraw optimization will start being applied
|
||||||
|
; turret-overdraw-scale-threshold=0.200000
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; skip-vram-detection=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; halt-rendering-when-minimized=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; runtime-sprite-reload=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; full-color-depth=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; render-in-native-resolution=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; use-flip-presentation-model=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; debug-api=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; discard-buffers-on-begin-frame=true
|
||||||
|
|
||||||
|
; Options: all, high, medium, low
|
||||||
|
; video-memory-usage=high
|
||||||
|
|
||||||
|
; Options: none, high-quality, low-quality
|
||||||
|
; texture-compression-level=high-quality
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; compress-virtual-atlas=true
|
||||||
|
|
||||||
|
; Options: copy, copy-sequential, flip, flip-discard
|
||||||
|
; dxgi-presentation-model=copy
|
||||||
|
|
||||||
|
; Options: none, flush, wait-for-vblank, flush-and-wait-for-vblank
|
||||||
|
; dxgi-action-before-present=none
|
||||||
|
|
||||||
|
; relevant only for flip presentation models
|
||||||
|
;
|
||||||
|
; Options: true, false
|
||||||
|
; dxgi-allow-tearing=false
|
||||||
|
|
||||||
|
; Options: false, true, auto
|
||||||
|
; dxgi-flip-do-not-wait=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; dxgi-present-restart=false
|
||||||
|
|
||||||
|
; dxgi-swap-chain-buffer-count=0
|
||||||
|
|
||||||
|
; dxgi-max-frame-latency=0
|
||||||
|
|
||||||
|
; dxgi-adapter-index=-1
|
||||||
|
|
||||||
|
; max-sprite-loading-threads=32
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; gpu-accelerated-compression=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; gpu-accelerated-mipmap-compression=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; wait-until-mipmap-generation-finished=true
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; check-for-unused-pixels=false
|
||||||
|
|
||||||
|
; ogl-depth-buffer-bit-depth=0
|
||||||
|
|
||||||
|
; Options: false, true, auto
|
||||||
|
; ogl-accelerated-renderer=auto
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; ogl-double-buffered=true
|
||||||
|
|
||||||
|
; Set to true if mipmapped sprites render very blurry on your GPU. Limited support.
|
||||||
|
;
|
||||||
|
; Options: true, false
|
||||||
|
; legacy-gpu-no-mipmaps=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; force-linear-magnification=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; custom-mipmap-workaround=false
|
||||||
|
|
||||||
|
; Options: true, false
|
||||||
|
; buffer-rename-workaround=false
|
||||||
|
|
||||||
|
; Comma separated list of OpenGL extensions that should not be used (for example: ARB_copy_image,KHR_debug)
|
||||||
|
; disabled-opengl-extensions=
|
||||||
|
|
||||||
|
|
57
docker/files/docker-dlc.sh
Executable file
57
docker/files/docker-dlc.sh
Executable file
@@ -0,0 +1,57 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
set -eou pipefail
|
||||||
|
|
||||||
|
# Path to the mod-list.json file
|
||||||
|
MOD_LIST_FILE="$MODS/mod-list.json"
|
||||||
|
|
||||||
|
ALL_SPACE_AGE_MODS=("elevated-rails" "quality" "space-age")
|
||||||
|
|
||||||
|
if [[ ! -f "$MOD_LIST_FILE" ]]; then
|
||||||
|
# Create the mod-list.json file if it doesn't exist
|
||||||
|
echo '{"mods":[{"name":"base","enabled":true}]}' > "$MOD_LIST_FILE"
|
||||||
|
fi
|
||||||
|
|
||||||
|
enable_mod()
|
||||||
|
{
|
||||||
|
echo "Enable mod $1 for DLC"
|
||||||
|
jq --arg mod_name "$1" 'if .mods | map(.name) | index($mod_name) then .mods |= map(if .name == $mod_name and .enabled == false then .enabled = true else . end) else . end' "$MOD_LIST_FILE" > "$MOD_LIST_FILE.tmp"
|
||||||
|
mv "$MOD_LIST_FILE.tmp" "$MOD_LIST_FILE"
|
||||||
|
}
|
||||||
|
|
||||||
|
disable_mod()
|
||||||
|
{
|
||||||
|
echo "Disable mod $1 for DLC"
|
||||||
|
jq --arg mod_name "$1" 'if .mods | map(.name) | index($mod_name) then .mods |= map(if .name == $mod_name and .enabled == true then .enabled = false else . end) else .mods += [{"name": $mod_name, "enabled": false}] end' "$MOD_LIST_FILE" > "$MOD_LIST_FILE.tmp"
|
||||||
|
mv "$MOD_LIST_FILE.tmp" "$MOD_LIST_FILE"
|
||||||
|
}
|
||||||
|
|
||||||
|
# Enable or disable DLCs if configured
|
||||||
|
if [[ ${DLC_SPACE_AGE:-} == "true" ]]; then
|
||||||
|
# Define the DLC mods
|
||||||
|
ENABLE_MODS=(${ALL_SPACE_AGE_MODS[@]})
|
||||||
|
elif [[ ${DLC_SPACE_AGE:-} == "false" ]]; then
|
||||||
|
# Define the DLC mods
|
||||||
|
DISABLE_MODS=(${ALL_SPACE_AGE_MODS[@]})
|
||||||
|
else
|
||||||
|
ENABLE_MODS=()
|
||||||
|
DISABLE_MODS=()
|
||||||
|
|
||||||
|
for SPACE_AGE_MOD in "${ALL_SPACE_AGE_MODS[@]}"; do
|
||||||
|
REGEX="(^|\s)$SPACE_AGE_MOD($|\s)"
|
||||||
|
if [[ "$DLC_SPACE_AGE" =~ $REGEX ]]; then
|
||||||
|
ENABLE_MODS+=($SPACE_AGE_MOD)
|
||||||
|
else
|
||||||
|
DISABLE_MODS+=($SPACE_AGE_MOD)
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Iterate over each DLC mod to enable
|
||||||
|
for MOD in "${ENABLE_MODS[@]}"; do
|
||||||
|
enable_mod "$MOD"
|
||||||
|
done
|
||||||
|
|
||||||
|
# Iterate over each DLC mod to disable
|
||||||
|
for MOD in "${DISABLE_MODS[@]}"; do
|
||||||
|
disable_mod "$MOD"
|
||||||
|
done
|
@@ -1,11 +1,16 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
set -eoux pipefail
|
set -eoux pipefail
|
||||||
|
INSTALLED_DIRECTORY=$(dirname "$(readlink -f "${BASH_SOURCE[0]}")")
|
||||||
FACTORIO_VOL=/factorio
|
FACTORIO_VOL=/factorio
|
||||||
LOAD_LATEST_SAVE="${LOAD_LATEST_SAVE:-true}"
|
LOAD_LATEST_SAVE="${LOAD_LATEST_SAVE:-true}"
|
||||||
GENERATE_NEW_SAVE="${GENERATE_NEW_SAVE:-false}"
|
GENERATE_NEW_SAVE="${GENERATE_NEW_SAVE:-false}"
|
||||||
|
PRESET="${PRESET:-""}"
|
||||||
SAVE_NAME="${SAVE_NAME:-""}"
|
SAVE_NAME="${SAVE_NAME:-""}"
|
||||||
|
BIND="${BIND:-""}"
|
||||||
|
CONSOLE_LOG_LOCATION="${CONSOLE_LOG_LOCATION:-""}"
|
||||||
|
|
||||||
|
# Create directories if they don't exist
|
||||||
|
# In rootless mode, these should be writable by the container user
|
||||||
mkdir -p "$FACTORIO_VOL"
|
mkdir -p "$FACTORIO_VOL"
|
||||||
mkdir -p "$SAVES"
|
mkdir -p "$SAVES"
|
||||||
mkdir -p "$CONFIG"
|
mkdir -p "$CONFIG"
|
||||||
@@ -13,13 +18,13 @@ mkdir -p "$MODS"
|
|||||||
mkdir -p "$SCENARIOS"
|
mkdir -p "$SCENARIOS"
|
||||||
mkdir -p "$SCRIPTOUTPUT"
|
mkdir -p "$SCRIPTOUTPUT"
|
||||||
|
|
||||||
|
# Generate RCON password if needed
|
||||||
if [[ ! -f $CONFIG/rconpw ]]; then
|
if [[ ! -f $CONFIG/rconpw ]]; then
|
||||||
# Generate a new RCON password if none exists
|
|
||||||
pwgen 15 1 >"$CONFIG/rconpw"
|
pwgen 15 1 >"$CONFIG/rconpw"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
# Copy default configs if they don't exist
|
||||||
if [[ ! -f $CONFIG/server-settings.json ]]; then
|
if [[ ! -f $CONFIG/server-settings.json ]]; then
|
||||||
# Copy default settings if server-settings.json doesn't exist
|
|
||||||
cp /opt/factorio/data/server-settings.example.json "$CONFIG/server-settings.json"
|
cp /opt/factorio/data/server-settings.example.json "$CONFIG/server-settings.json"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
@@ -31,28 +36,32 @@ if [[ ! -f $CONFIG/map-settings.json ]]; then
|
|||||||
cp /opt/factorio/data/map-settings.example.json "$CONFIG/map-settings.json"
|
cp /opt/factorio/data/map-settings.example.json "$CONFIG/map-settings.json"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
# Clean up incomplete saves
|
||||||
NRTMPSAVES=$( find -L "$SAVES" -iname \*.tmp.zip -mindepth 1 | wc -l )
|
NRTMPSAVES=$( find -L "$SAVES" -iname \*.tmp.zip -mindepth 1 | wc -l )
|
||||||
if [[ $NRTMPSAVES -gt 0 ]]; then
|
if [[ $NRTMPSAVES -gt 0 ]]; then
|
||||||
# Delete incomplete saves (such as after a forced exit)
|
|
||||||
rm -f "$SAVES"/*.tmp.zip
|
rm -f "$SAVES"/*.tmp.zip
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if [[ ${UPDATE_MODS_ON_START:-} ]]; then
|
# Update mods if requested
|
||||||
./docker-update-mods.sh
|
if [[ ${UPDATE_MODS_ON_START:-} == "true" ]]; then
|
||||||
|
"${INSTALLED_DIRECTORY}"/docker-update-mods.sh
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if [[ $(id -u) = 0 ]]; then
|
# Handle DLC
|
||||||
# Update the User and Group ID based on the PUID/PGID variables
|
"${INSTALLED_DIRECTORY}"/docker-dlc.sh
|
||||||
usermod -o -u "$PUID" factorio
|
|
||||||
groupmod -o -g "$PGID" factorio
|
# In rootless mode, we don't need to handle user switching or chown
|
||||||
# Take ownership of factorio data if running as root
|
# The container runs as the specified user from the start
|
||||||
chown -R factorio:factorio "$FACTORIO_VOL"
|
EXEC=""
|
||||||
# Drop to the factorio user
|
if [[ -f /bin/box64 ]]; then
|
||||||
SU_EXEC="su-exec factorio"
|
# Use emulator for ARM hosts
|
||||||
else
|
EXEC="/bin/box64"
|
||||||
SU_EXEC=""
|
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
# Update config path
|
||||||
|
sed -i '/write-data=/c\write-data=\/factorio/' /opt/factorio/config/config.ini
|
||||||
|
|
||||||
|
# Generate new save if needed
|
||||||
NRSAVES=$(find -L "$SAVES" -iname \*.zip -mindepth 1 | wc -l)
|
NRSAVES=$(find -L "$SAVES" -iname \*.zip -mindepth 1 | wc -l)
|
||||||
if [[ $GENERATE_NEW_SAVE != true && $NRSAVES == 0 ]]; then
|
if [[ $GENERATE_NEW_SAVE != true && $NRSAVES == 0 ]]; then
|
||||||
GENERATE_NEW_SAVE=true
|
GENERATE_NEW_SAVE=true
|
||||||
@@ -67,13 +76,22 @@ if [[ $GENERATE_NEW_SAVE == true ]]; then
|
|||||||
if [[ -f "$SAVES/$SAVE_NAME.zip" ]]; then
|
if [[ -f "$SAVES/$SAVE_NAME.zip" ]]; then
|
||||||
echo "Map $SAVES/$SAVE_NAME.zip already exists, skipping map generation"
|
echo "Map $SAVES/$SAVE_NAME.zip already exists, skipping map generation"
|
||||||
else
|
else
|
||||||
$SU_EXEC /opt/factorio/bin/x64/factorio \
|
if [[ -n "$PRESET" ]]; then
|
||||||
--create "$SAVES/$SAVE_NAME.zip" \
|
$EXEC /opt/factorio/bin/x64/factorio \
|
||||||
--map-gen-settings "$CONFIG/map-gen-settings.json" \
|
--create "$SAVES/$SAVE_NAME.zip" \
|
||||||
--map-settings "$CONFIG/map-settings.json"
|
--preset "$PRESET" \
|
||||||
|
--map-gen-settings "$CONFIG/map-gen-settings.json" \
|
||||||
|
--map-settings "$CONFIG/map-settings.json"
|
||||||
|
else
|
||||||
|
$EXEC /opt/factorio/bin/x64/factorio \
|
||||||
|
--create "$SAVES/$SAVE_NAME.zip" \
|
||||||
|
--map-gen-settings "$CONFIG/map-gen-settings.json" \
|
||||||
|
--map-settings "$CONFIG/map-settings.json"
|
||||||
|
fi
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
# Build command flags
|
||||||
FLAGS=(\
|
FLAGS=(\
|
||||||
--port "$PORT" \
|
--port "$PORT" \
|
||||||
--server-settings "$CONFIG/server-settings.json" \
|
--server-settings "$CONFIG/server-settings.json" \
|
||||||
@@ -84,13 +102,23 @@ FLAGS=(\
|
|||||||
--server-adminlist "$CONFIG/server-adminlist.json" \
|
--server-adminlist "$CONFIG/server-adminlist.json" \
|
||||||
--rcon-password "$(cat "$CONFIG/rconpw")" \
|
--rcon-password "$(cat "$CONFIG/rconpw")" \
|
||||||
--server-id /factorio/config/server-id.json \
|
--server-id /factorio/config/server-id.json \
|
||||||
|
--mod-directory "$MODS" \
|
||||||
)
|
)
|
||||||
|
|
||||||
|
if [ -n "$CONSOLE_LOG_LOCATION" ]; then
|
||||||
|
FLAGS+=( --console-log "$CONSOLE_LOG_LOCATION" )
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -n "$BIND" ]; then
|
||||||
|
FLAGS+=( --bind "$BIND" )
|
||||||
|
fi
|
||||||
|
|
||||||
if [[ $LOAD_LATEST_SAVE == true ]]; then
|
if [[ $LOAD_LATEST_SAVE == true ]]; then
|
||||||
FLAGS+=( --start-server-load-latest )
|
FLAGS+=( --start-server-load-latest )
|
||||||
else
|
else
|
||||||
FLAGS+=( --start-server "$SAVE_NAME" )
|
FLAGS+=( --start-server "$SAVE_NAME" )
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# shellcheck disable=SC2086
|
# Execute factorio
|
||||||
exec $SU_EXEC /opt/factorio/bin/x64/factorio "${FLAGS[@]}" "$@"
|
# In rootless mode, we run directly without user switching
|
||||||
|
exec $EXEC /opt/factorio/bin/x64/factorio "${FLAGS[@]}" "$@"
|
@@ -1,10 +1,13 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
set -eoux pipefail
|
set -eoux pipefail
|
||||||
|
INSTALLED_DIRECTORY=$(dirname "$(readlink -f "${BASH_SOURCE[0]}")")
|
||||||
FACTORIO_VOL=/factorio
|
FACTORIO_VOL=/factorio
|
||||||
LOAD_LATEST_SAVE="${LOAD_LATEST_SAVE:-true}"
|
LOAD_LATEST_SAVE="${LOAD_LATEST_SAVE:-true}"
|
||||||
GENERATE_NEW_SAVE="${GENERATE_NEW_SAVE:-false}"
|
GENERATE_NEW_SAVE="${GENERATE_NEW_SAVE:-false}"
|
||||||
|
PRESET="${PRESET:-""}"
|
||||||
SAVE_NAME="${SAVE_NAME:-""}"
|
SAVE_NAME="${SAVE_NAME:-""}"
|
||||||
|
BIND="${BIND:-""}"
|
||||||
|
CONSOLE_LOG_LOCATION="${CONSOLE_LOG_LOCATION:-""}"
|
||||||
|
|
||||||
mkdir -p "$FACTORIO_VOL"
|
mkdir -p "$FACTORIO_VOL"
|
||||||
mkdir -p "$SAVES"
|
mkdir -p "$SAVES"
|
||||||
@@ -37,21 +40,29 @@ if [[ $NRTMPSAVES -gt 0 ]]; then
|
|||||||
rm -f "$SAVES"/*.tmp.zip
|
rm -f "$SAVES"/*.tmp.zip
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if [[ ${UPDATE_MODS_ON_START:-} ]]; then
|
if [[ ${UPDATE_MODS_ON_START:-} == "true" ]]; then
|
||||||
./docker-update-mods.sh
|
${INSTALLED_DIRECTORY}/docker-update-mods.sh
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if [[ $(id -u) = 0 ]]; then
|
${INSTALLED_DIRECTORY}/docker-dlc.sh
|
||||||
|
|
||||||
|
EXEC=""
|
||||||
|
if [[ $(id -u) == 0 ]]; then
|
||||||
# Update the User and Group ID based on the PUID/PGID variables
|
# Update the User and Group ID based on the PUID/PGID variables
|
||||||
usermod -o -u "$PUID" factorio
|
usermod -o -u "$PUID" factorio
|
||||||
groupmod -o -g "$PGID" factorio
|
groupmod -o -g "$PGID" factorio
|
||||||
# Take ownership of factorio data if running as root
|
# Take ownership of factorio data if running as root
|
||||||
chown -R factorio:factorio "$FACTORIO_VOL"
|
chown -R factorio:factorio "$FACTORIO_VOL"
|
||||||
# Drop to the factorio user
|
# Drop to the factorio user
|
||||||
SU_EXEC="su-exec factorio"
|
EXEC="runuser -u factorio -g factorio --"
|
||||||
else
|
|
||||||
SU_EXEC=""
|
|
||||||
fi
|
fi
|
||||||
|
if [[ -f /bin/box64 ]]; then
|
||||||
|
# Use an emulator to run on ARM hosts
|
||||||
|
# this only gets installed when the target docker platform is linux/arm64
|
||||||
|
EXEC="$EXEC /bin/box64"
|
||||||
|
fi
|
||||||
|
|
||||||
|
sed -i '/write-data=/c\write-data=\/factorio/' /opt/factorio/config/config.ini
|
||||||
|
|
||||||
NRSAVES=$(find -L "$SAVES" -iname \*.zip -mindepth 1 | wc -l)
|
NRSAVES=$(find -L "$SAVES" -iname \*.zip -mindepth 1 | wc -l)
|
||||||
if [[ $GENERATE_NEW_SAVE != true && $NRSAVES == 0 ]]; then
|
if [[ $GENERATE_NEW_SAVE != true && $NRSAVES == 0 ]]; then
|
||||||
@@ -67,10 +78,18 @@ if [[ $GENERATE_NEW_SAVE == true ]]; then
|
|||||||
if [[ -f "$SAVES/$SAVE_NAME.zip" ]]; then
|
if [[ -f "$SAVES/$SAVE_NAME.zip" ]]; then
|
||||||
echo "Map $SAVES/$SAVE_NAME.zip already exists, skipping map generation"
|
echo "Map $SAVES/$SAVE_NAME.zip already exists, skipping map generation"
|
||||||
else
|
else
|
||||||
$SU_EXEC /opt/factorio/bin/x64/factorio \
|
if [[ ! -z "$PRESET" ]]; then
|
||||||
--create "$SAVES/$SAVE_NAME.zip" \
|
$EXEC /opt/factorio/bin/x64/factorio \
|
||||||
--map-gen-settings "$CONFIG/map-gen-settings.json" \
|
--create "$SAVES/$SAVE_NAME.zip" \
|
||||||
--map-settings "$CONFIG/map-settings.json"
|
--preset "$PRESET" \
|
||||||
|
--map-gen-settings "$CONFIG/map-gen-settings.json" \
|
||||||
|
--map-settings "$CONFIG/map-settings.json"
|
||||||
|
else
|
||||||
|
$EXEC /opt/factorio/bin/x64/factorio \
|
||||||
|
--create "$SAVES/$SAVE_NAME.zip" \
|
||||||
|
--map-gen-settings "$CONFIG/map-gen-settings.json" \
|
||||||
|
--map-settings "$CONFIG/map-settings.json"
|
||||||
|
fi
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
@@ -84,8 +103,17 @@ FLAGS=(\
|
|||||||
--server-adminlist "$CONFIG/server-adminlist.json" \
|
--server-adminlist "$CONFIG/server-adminlist.json" \
|
||||||
--rcon-password "$(cat "$CONFIG/rconpw")" \
|
--rcon-password "$(cat "$CONFIG/rconpw")" \
|
||||||
--server-id /factorio/config/server-id.json \
|
--server-id /factorio/config/server-id.json \
|
||||||
|
--mod-directory "$MODS" \
|
||||||
)
|
)
|
||||||
|
|
||||||
|
if [ -n "$CONSOLE_LOG_LOCATION" ]; then
|
||||||
|
FLAGS+=( --console-log "$CONSOLE_LOG_LOCATION" )
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -n "$BIND" ]; then
|
||||||
|
FLAGS+=( --bind "$BIND" )
|
||||||
|
fi
|
||||||
|
|
||||||
if [[ $LOAD_LATEST_SAVE == true ]]; then
|
if [[ $LOAD_LATEST_SAVE == true ]]; then
|
||||||
FLAGS+=( --start-server-load-latest )
|
FLAGS+=( --start-server-load-latest )
|
||||||
else
|
else
|
||||||
@@ -93,4 +121,4 @@ else
|
|||||||
fi
|
fi
|
||||||
|
|
||||||
# shellcheck disable=SC2086
|
# shellcheck disable=SC2086
|
||||||
exec $SU_EXEC /opt/factorio/bin/x64/factorio "${FLAGS[@]}" "$@"
|
exec $EXEC /opt/factorio/bin/x64/factorio "${FLAGS[@]}" "$@"
|
@@ -5,7 +5,7 @@ if [[ -f /run/secrets/username ]]; then
|
|||||||
USERNAME=$(cat /run/secrets/username)
|
USERNAME=$(cat /run/secrets/username)
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if [[ -f /run/secrets/username ]]; then
|
if [[ -f /run/secrets/token ]]; then
|
||||||
TOKEN=$(cat /run/secrets/token)
|
TOKEN=$(cat /run/secrets/token)
|
||||||
fi
|
fi
|
||||||
|
|
11
docker/files/players-online.sh
Executable file
11
docker/files/players-online.sh
Executable file
@@ -0,0 +1,11 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
PLAYERS=$(rcon /players)
|
||||||
|
ONLINE_COUNT=$(echo "$PLAYERS" | grep -c " (online)$")
|
||||||
|
|
||||||
|
if [[ "$ONLINE_COUNT" -gt "0" ]]; then
|
||||||
|
echo "$PLAYERS"
|
||||||
|
# exit with 75 (EX_TEMPFAIL) so watchtower skips the update
|
||||||
|
# https://containrrr.dev/watchtower/lifecycle-hooks/
|
||||||
|
exit 75
|
||||||
|
fi
|
@@ -6,6 +6,8 @@ if [[ -z ${1:-} ]]; then
|
|||||||
fi
|
fi
|
||||||
|
|
||||||
SERVER_SCENARIO="$1"
|
SERVER_SCENARIO="$1"
|
||||||
|
PRESET="${PRESET:-""}"
|
||||||
|
|
||||||
mkdir -p "$SAVES"
|
mkdir -p "$SAVES"
|
||||||
mkdir -p "$CONFIG"
|
mkdir -p "$CONFIG"
|
||||||
mkdir -p "$MODS"
|
mkdir -p "$MODS"
|
||||||
@@ -32,6 +34,9 @@ fi
|
|||||||
exec /opt/factorio/bin/x64/factorio \
|
exec /opt/factorio/bin/x64/factorio \
|
||||||
--port "$PORT" \
|
--port "$PORT" \
|
||||||
--start-server-load-scenario "$SERVER_SCENARIO" \
|
--start-server-load-scenario "$SERVER_SCENARIO" \
|
||||||
|
--preset "$PRESET" \
|
||||||
|
--map-gen-settings "$CONFIG/map-gen-settings.json" \
|
||||||
|
--map-settings "$CONFIG/map-settings.json" \
|
||||||
--server-settings "$CONFIG/server-settings.json" \
|
--server-settings "$CONFIG/server-settings.json" \
|
||||||
--server-banlist "$CONFIG/server-banlist.json" \
|
--server-banlist "$CONFIG/server-banlist.json" \
|
||||||
--server-whitelist "$CONFIG/server-whitelist.json" \
|
--server-whitelist "$CONFIG/server-whitelist.json" \
|
239
docker/files/update-mods.sh
Executable file
239
docker/files/update-mods.sh
Executable file
@@ -0,0 +1,239 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
set -eou pipefail
|
||||||
|
|
||||||
|
FACTORIO_VERSION=$1
|
||||||
|
MOD_DIR=$2
|
||||||
|
USERNAME=$3
|
||||||
|
TOKEN=$4
|
||||||
|
|
||||||
|
MOD_BASE_URL="https://mods.factorio.com"
|
||||||
|
|
||||||
|
print_step()
|
||||||
|
{
|
||||||
|
echo "$1"
|
||||||
|
}
|
||||||
|
|
||||||
|
print_success()
|
||||||
|
{
|
||||||
|
echo "$1"
|
||||||
|
}
|
||||||
|
|
||||||
|
print_failure()
|
||||||
|
{
|
||||||
|
echo "$1"
|
||||||
|
}
|
||||||
|
|
||||||
|
# Checks if the current game version satisfies the mod's minimum required version.
|
||||||
|
# Returns 1 if the game version is compatible with the mod, 0 if not
|
||||||
|
check_game_version() {
|
||||||
|
local mod_required_version="$1" # The minimum Factorio version required by the mod
|
||||||
|
local current_game_version="$2" # The current Factorio version
|
||||||
|
|
||||||
|
local mod_major mod_minor game_major game_minor
|
||||||
|
mod_major=$(echo "$mod_required_version" | cut -d '.' -f1)
|
||||||
|
mod_minor=$(echo "$mod_required_version" | cut -d '.' -f2)
|
||||||
|
game_major=$(echo "$current_game_version" | cut -d '.' -f1)
|
||||||
|
game_minor=$(echo "$current_game_version" | cut -d '.' -f2)
|
||||||
|
|
||||||
|
# If game major version is greater than mod's required major version, it's compatible
|
||||||
|
if [[ "$game_major" -gt "$mod_major" ]]; then
|
||||||
|
echo 1
|
||||||
|
return
|
||||||
|
fi
|
||||||
|
|
||||||
|
# If game major version is less than mod's required major version, it's not compatible
|
||||||
|
if [[ "$game_major" -lt "$mod_major" ]]; then
|
||||||
|
echo 0
|
||||||
|
return
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Major versions are equal, check minor versions
|
||||||
|
# Game minor version must be >= mod's required minor version
|
||||||
|
if [[ "$game_minor" -ge "$mod_minor" ]]; then
|
||||||
|
echo 1
|
||||||
|
else
|
||||||
|
echo 0
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
# Checks dependency string with provided version.
|
||||||
|
# Only checks for operator based string, ignoring everything else
|
||||||
|
# Returns 1 if check is ok, 0 if not
|
||||||
|
check_dependency_version()
|
||||||
|
{
|
||||||
|
local dependency="$1"
|
||||||
|
local mod_version="$2"
|
||||||
|
|
||||||
|
if [[ "$dependency" =~ ^(\?|!|~|\(~\)) ]]; then
|
||||||
|
echo 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
local condition
|
||||||
|
condition=$(echo "$dependency" | grep -oE '(>=|<=|>|<|=) [0-9]+(\.[0-9]+)*')
|
||||||
|
|
||||||
|
if [[ -z "$condition" ]]; then
|
||||||
|
echo 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
local operator required_version
|
||||||
|
operator=$(echo "$condition" | awk '{print $1}')
|
||||||
|
required_version=$(echo "$condition" | awk '{print $2}')
|
||||||
|
|
||||||
|
case "$operator" in
|
||||||
|
">=")
|
||||||
|
if [[ "$(printf '%s\n%s\n' "$required_version" "$mod_version" | sort -V | head -n1)" == "$required_version" ]]; then
|
||||||
|
echo 1
|
||||||
|
else
|
||||||
|
echo 0
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
">")
|
||||||
|
if [[ "$(printf '%s\n%s\n' "$required_version" "$mod_version" | sort -V | head -n1)" == "$required_version" && "$required_version" != "$mod_version" ]]; then
|
||||||
|
echo 1
|
||||||
|
else
|
||||||
|
echo 0
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
"<=")
|
||||||
|
if [[ "$(printf '%s\n%s\n' "$required_version" "$mod_version" | sort -V | tail -n1)" == "$required_version" ]]; then
|
||||||
|
echo 1
|
||||||
|
else
|
||||||
|
echo 0
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
"<")
|
||||||
|
if [[ "$(printf '%s\n%s\n' "$required_version" "$mod_version" | sort -V | tail -n1)" == "$required_version" && "$required_version" != "$mod_version" ]]; then
|
||||||
|
echo 1
|
||||||
|
else
|
||||||
|
echo 0
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
"=")
|
||||||
|
if [[ "$mod_version" == "$required_version" ]]; then
|
||||||
|
echo 1
|
||||||
|
else
|
||||||
|
echo 0
|
||||||
|
fi
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
echo 0
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
}
|
||||||
|
|
||||||
|
get_mod_info()
|
||||||
|
{
|
||||||
|
local mod_info_json="$1"
|
||||||
|
|
||||||
|
# Process mod releases from newest to oldest, looking for a compatible version
|
||||||
|
while IFS= read -r mod_release_info; do
|
||||||
|
local mod_version mod_factorio_version
|
||||||
|
mod_version=$(echo "$mod_release_info" | jq -r ".version")
|
||||||
|
mod_factorio_version=$(echo "$mod_release_info" | jq -r ".info_json.factorio_version")
|
||||||
|
|
||||||
|
# Check if this mod version is compatible with our Factorio version
|
||||||
|
# This prevents downloading mods that require a newer Factorio version (fixes #468)
|
||||||
|
# and ensures backward compatibility (e.g., Factorio 2.0 can use 1.x mods) (fixes #517)
|
||||||
|
if [[ $(check_game_version "$mod_factorio_version" "$FACTORIO_VERSION") == 0 ]]; then
|
||||||
|
echo " Skipping mod version $mod_version because of factorio version mismatch" >&2
|
||||||
|
continue
|
||||||
|
fi
|
||||||
|
|
||||||
|
# If we found 'dependencies' element, we also check versions there
|
||||||
|
if [[ $(echo "$mod_release_info" | jq -e '.info_json | has("dependencies") and (.dependencies | length > 0)') == true ]]; then
|
||||||
|
while IFS= read -r dependency; do
|
||||||
|
|
||||||
|
# We only check for 'base' dependency
|
||||||
|
if [[ "$dependency" == base* ]] && [[ $(check_dependency_version "$dependency" "$FACTORIO_VERSION") == 0 ]]; then
|
||||||
|
echo " Skipping mod version $mod_version, unsatisfied base dependency: $dependency" >&2
|
||||||
|
continue 2
|
||||||
|
fi
|
||||||
|
|
||||||
|
done < <(echo "$mod_release_info" | jq -r '.info_json.dependencies[]')
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "$mod_release_info" | jq -j ".file_name, \";\", .download_url, \";\", .sha1"
|
||||||
|
break
|
||||||
|
|
||||||
|
done < <(echo "$mod_info_json" | jq -c ".releases|sort_by(.released_at)|reverse|.[]")
|
||||||
|
}
|
||||||
|
|
||||||
|
update_mod()
|
||||||
|
{
|
||||||
|
MOD_NAME="$1"
|
||||||
|
MOD_NAME_ENCODED="${1// /%20}"
|
||||||
|
|
||||||
|
print_step "Checking for update of mod $MOD_NAME for factorio $FACTORIO_VERSION ..."
|
||||||
|
|
||||||
|
MOD_INFO_URL="$MOD_BASE_URL/api/mods/$MOD_NAME_ENCODED/full"
|
||||||
|
MOD_INFO_JSON=$(curl --silent "$MOD_INFO_URL")
|
||||||
|
|
||||||
|
if ! echo "$MOD_INFO_JSON" | jq -e .name >/dev/null; then
|
||||||
|
print_success " Custom mod not on $MOD_BASE_URL, skipped."
|
||||||
|
return 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
MOD_INFO=$(get_mod_info "$MOD_INFO_JSON")
|
||||||
|
|
||||||
|
if [[ "$MOD_INFO" == "" ]]; then
|
||||||
|
print_failure " Not compatible with version"
|
||||||
|
return 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
MOD_FILENAME=$(echo "$MOD_INFO" | cut -f1 -d";")
|
||||||
|
MOD_URL=$(echo "$MOD_INFO" | cut -f2 -d";")
|
||||||
|
MOD_SHA1=$(echo "$MOD_INFO" | cut -f3 -d";")
|
||||||
|
|
||||||
|
if [[ $MOD_FILENAME == null ]]; then
|
||||||
|
print_failure " Not compatible with version"
|
||||||
|
return 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ -f $MOD_DIR/$MOD_FILENAME ]]; then
|
||||||
|
print_success " Already up-to-date."
|
||||||
|
return 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
print_step " Downloading $MOD_FILENAME"
|
||||||
|
FULL_URL="$MOD_BASE_URL$MOD_URL?username=$USERNAME&token=$TOKEN"
|
||||||
|
HTTP_STATUS=$(curl --silent -L -w "%{http_code}" -o "$MOD_DIR/$MOD_FILENAME" "$FULL_URL")
|
||||||
|
|
||||||
|
if [[ $HTTP_STATUS != 200 ]]; then
|
||||||
|
print_failure " Download failed: Code $HTTP_STATUS."
|
||||||
|
rm -f "$MOD_DIR/$MOD_FILENAME"
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ ! -f $MOD_DIR/$MOD_FILENAME ]]; then
|
||||||
|
print_failure " Downloaded file missing!"
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
if ! [[ $(sha1sum "$MOD_DIR/$MOD_FILENAME") =~ $MOD_SHA1 ]]; then
|
||||||
|
print_failure " SHA1 mismatch!"
|
||||||
|
rm -f "$MOD_DIR/$MOD_FILENAME"
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
print_success " Download complete."
|
||||||
|
|
||||||
|
for file in "$MOD_DIR/${MOD_NAME}_"*".zip"; do # wildcard does usually not work in quotes: https://unix.stackexchange.com/a/67761
|
||||||
|
if [[ $file != $MOD_DIR/$MOD_FILENAME ]]; then
|
||||||
|
print_success " Deleting old version: $file"
|
||||||
|
rm -f "$file"
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
# Process all enabled mods from mod-list.json, but skip built-in mods
|
||||||
|
# The Space Age DLC includes built-in mods (elevated-rails, quality, space-age) that should not be downloaded
|
||||||
|
if [[ -f $MOD_DIR/mod-list.json ]]; then
|
||||||
|
jq -r ".mods|map(select(.enabled))|.[].name" "$MOD_DIR/mod-list.json" | while read -r mod; do
|
||||||
|
# Skip base mod and DLC built-in mods
|
||||||
|
if [[ $mod != base ]] && [[ $mod != elevated-rails ]] && [[ $mod != quality ]] && [[ $mod != space-age ]]; then
|
||||||
|
update_mod "$mod" || true
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
fi
|
13
docker/rcon/Makefile
Normal file
13
docker/rcon/Makefile
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
# Optimization
|
||||||
|
OPT = -O3 -flto
|
||||||
|
TARGET = rcon
|
||||||
|
|
||||||
|
CC = gcc
|
||||||
|
CFLAGS = -std=c17 -Wall -Wextra -pedantic $(OPT)
|
||||||
|
REMOVE = rm -f
|
||||||
|
|
||||||
|
all:
|
||||||
|
$(CC) $(CFLAGS) main.c -o $(TARGET)
|
||||||
|
|
||||||
|
clean:
|
||||||
|
$(REMOVE) $(TARGET)
|
219
docker/rcon/main.c
Normal file
219
docker/rcon/main.c
Normal file
@@ -0,0 +1,219 @@
|
|||||||
|
#define _GNU_SOURCE
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <stdbool.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <netdb.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <time.h>
|
||||||
|
|
||||||
|
#include <arpa/inet.h>
|
||||||
|
|
||||||
|
#define MIN_PACKET 10
|
||||||
|
#define MAX_PACKET 4096
|
||||||
|
#define MAX_BODY (MAX_PACKET - (3 * sizeof(uint32_t)) - 2)
|
||||||
|
|
||||||
|
#define RCON_HOST "127.0.0.1"
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
RCON_TYPE_RESPONSE = 0,
|
||||||
|
RCON_TYPE_EXECCOMMAND = 2,
|
||||||
|
RCON_TYPE_AUTH_RESPONSE = 2,
|
||||||
|
RCON_TYPE_AUTH = 3,
|
||||||
|
} packet_type;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
uint32_t length;
|
||||||
|
uint32_t id;
|
||||||
|
packet_type type;
|
||||||
|
char body[MAX_BODY];
|
||||||
|
} packet;
|
||||||
|
|
||||||
|
int rcon_open(const char *port);
|
||||||
|
void rcon_create(packet* pkt, packet_type type, const char* body);
|
||||||
|
bool rcon_send(int rcon_socket, const packet* pkt);
|
||||||
|
bool rcon_auth(int rcon_socket, const char* password);
|
||||||
|
bool rcon_recv(int rcon_socket, packet* pkt, packet_type expected_type);
|
||||||
|
char* combine_args(int argc, char* argv[]);
|
||||||
|
char* read_password(const char* conf_dir);
|
||||||
|
|
||||||
|
int main(int argc, char* argv[]) {
|
||||||
|
if (argc < 2) {
|
||||||
|
fprintf(stderr, "error: missing command argument\n");
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
srand((unsigned int)time(NULL));
|
||||||
|
|
||||||
|
const char* port = getenv("RCON_PORT");
|
||||||
|
if (port == NULL) {
|
||||||
|
fprintf(stderr, "error: missing $RCON_PORT env\n");
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char* conf_dir = getenv("CONFIG");
|
||||||
|
if (conf_dir == NULL) {
|
||||||
|
fprintf(stderr, "error: missing $CONFIG env");
|
||||||
|
exit(EXIT_FAILURE);
|
||||||
|
}
|
||||||
|
|
||||||
|
int rcon_socket = rcon_open(port);
|
||||||
|
if (rcon_socket == -1) {
|
||||||
|
fprintf(stderr, "error: could not connect\n");
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!rcon_auth(rcon_socket, read_password(conf_dir))) {
|
||||||
|
fprintf(stderr, "error: login failed\n");
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
packet pkt;
|
||||||
|
rcon_create(&pkt, RCON_TYPE_EXECCOMMAND, combine_args(argc, argv));
|
||||||
|
if (!rcon_send(rcon_socket, &pkt)) {
|
||||||
|
fprintf(stderr, "error: send command failed\n");
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rcon_recv(rcon_socket, &pkt, RCON_TYPE_RESPONSE) && pkt.length > 0) {
|
||||||
|
puts(pkt.body);
|
||||||
|
}
|
||||||
|
|
||||||
|
return EXIT_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
char* combine_args(int argc, char* argv[]) {
|
||||||
|
// combine all cli arguments
|
||||||
|
char* command = malloc(MAX_BODY);
|
||||||
|
memset(command, 0, MAX_BODY);
|
||||||
|
strcat(command, argv[1]);
|
||||||
|
|
||||||
|
for (int idx = 2; idx < argc; idx++) {
|
||||||
|
strcat(command, " ");
|
||||||
|
strcat(command, argv[idx]);
|
||||||
|
}
|
||||||
|
|
||||||
|
return command;
|
||||||
|
}
|
||||||
|
|
||||||
|
char* read_password(const char* conf_dir) {
|
||||||
|
char* path = malloc(strlen(conf_dir) + 64);
|
||||||
|
strcpy(path, conf_dir);
|
||||||
|
strcat(path, "/rconpw");
|
||||||
|
|
||||||
|
FILE* fptr = fopen(path, "r");
|
||||||
|
fseek(fptr, 0, SEEK_END);
|
||||||
|
long fsize = ftell(fptr);
|
||||||
|
fseek(fptr, 0, SEEK_SET); /* same as rewind(f); */
|
||||||
|
|
||||||
|
char *password = malloc(fsize + 1);
|
||||||
|
size_t bytes_read = fread(password, fsize, 1, fptr);
|
||||||
|
(void)bytes_read; // Suppress unused warning
|
||||||
|
fclose(fptr);
|
||||||
|
|
||||||
|
password[fsize] = 0;
|
||||||
|
if (password[fsize-1] == '\n') {
|
||||||
|
password[fsize-1] = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return password;
|
||||||
|
}
|
||||||
|
|
||||||
|
int rcon_open(const char *port) {
|
||||||
|
struct sockaddr_in address = {
|
||||||
|
.sin_family = AF_INET,
|
||||||
|
.sin_port = htons(atoi(port))
|
||||||
|
};
|
||||||
|
inet_aton(RCON_HOST, &address.sin_addr);
|
||||||
|
|
||||||
|
int rcon_socket = socket(AF_INET, SOCK_STREAM, 0);
|
||||||
|
if (connect(rcon_socket, (struct sockaddr*) &address, sizeof(address)) < 0) {
|
||||||
|
return -1;
|
||||||
|
} else {
|
||||||
|
return rcon_socket;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void rcon_create(packet* pkt, packet_type type, const char* body) {
|
||||||
|
size_t body_length = strlen(body);
|
||||||
|
if (body_length >= MAX_BODY - 2) {
|
||||||
|
fprintf(stderr, "error: command to long");
|
||||||
|
exit(EXIT_FAILURE);
|
||||||
|
}
|
||||||
|
|
||||||
|
pkt->id = abs(rand());
|
||||||
|
pkt->type = type;
|
||||||
|
pkt->length = (uint32_t)(sizeof(pkt->id) + sizeof(pkt->type) + body_length + 2);
|
||||||
|
|
||||||
|
memset(pkt->body, 0, MAX_BODY);
|
||||||
|
strncpy(pkt->body, body, MAX_BODY);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool rcon_recv(int rcon_socket, packet* pkt, packet_type expected_type) {
|
||||||
|
memset(pkt, 0, sizeof(*pkt));
|
||||||
|
|
||||||
|
// Read response packet length
|
||||||
|
ssize_t expected_length_bytes = sizeof(pkt->length);
|
||||||
|
ssize_t rx_bytes = recv(rcon_socket, &(pkt->length), expected_length_bytes, 0);
|
||||||
|
|
||||||
|
if (rx_bytes == -1) {
|
||||||
|
perror("error: socket error");
|
||||||
|
return false;
|
||||||
|
} else if (rx_bytes == 0) {
|
||||||
|
fprintf(stderr, "error: no data recieved\n");
|
||||||
|
return false;
|
||||||
|
} else if (rx_bytes < expected_length_bytes || pkt->length < MIN_PACKET || pkt->length > MAX_PACKET) {
|
||||||
|
fprintf(stderr, "error: invalid data\n");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
ssize_t received = 0;
|
||||||
|
while (received < pkt->length) {
|
||||||
|
rx_bytes = recv(rcon_socket, (char *)pkt + sizeof(pkt->length) + received, pkt->length - received, 0);
|
||||||
|
if (rx_bytes < 0) {
|
||||||
|
perror("error: socket error");
|
||||||
|
return false;
|
||||||
|
} else if (rx_bytes == 0) {
|
||||||
|
fprintf(stderr, "error: connection lost\n");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
received += rx_bytes;
|
||||||
|
}
|
||||||
|
|
||||||
|
return pkt->type == expected_type;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool rcon_send(int rcon_socket, const packet* pkt) {
|
||||||
|
size_t length = sizeof(pkt->length) + pkt->length;
|
||||||
|
char *ptr = (char*) pkt;
|
||||||
|
|
||||||
|
while (length > 0) {
|
||||||
|
ssize_t ret = send(rcon_socket, ptr, length, 0);
|
||||||
|
|
||||||
|
if (ret == -1) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
ptr += ret;
|
||||||
|
length -= ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool rcon_auth(int rcon_socket, const char* password) {
|
||||||
|
packet pkt;
|
||||||
|
rcon_create(&pkt, RCON_TYPE_AUTH, password);
|
||||||
|
|
||||||
|
if (!rcon_send(rcon_socket, &pkt)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!rcon_recv(rcon_socket, &pkt, RCON_TYPE_AUTH_RESPONSE)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
169
update.sh
Executable file
169
update.sh
Executable file
@@ -0,0 +1,169 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
set -e
|
||||||
|
SEMVER_REGEX="^(0|[1-9][0-9]*)\\.(0|[1-9][0-9]*)\\.(0|[1-9][0-9]*)$"
|
||||||
|
|
||||||
|
stable_online_version=$(curl 'https://factorio.com/api/latest-releases' | jq '.stable.headless' -r)
|
||||||
|
experimental_online_version=$(curl 'https://factorio.com/api/latest-releases' | jq '.experimental.headless' -r)
|
||||||
|
|
||||||
|
stable_sha256=$(curl "https://factorio.com/download/sha256sums/" | grep -E "(factorio_headless_x64_|factorio-headless_linux_)${stable_online_version}.tar.xz" | awk '{print $1}')
|
||||||
|
experimental_sha256=$(curl "https://factorio.com/download/sha256sums/" | grep -E "(factorio_headless_x64_|factorio-headless_linux_)${experimental_online_version}.tar.xz" | awk '{print $1}')
|
||||||
|
|
||||||
|
stable_current_version=$(jq 'with_entries(select(.value.tags | index("stable"))) | keys | .[0]' buildinfo.json -r)
|
||||||
|
latest_current_version=$(jq 'with_entries(select(.value.tags | index("latest"))) | keys | .[0]' buildinfo.json -r)
|
||||||
|
|
||||||
|
echo "stable_online_version=${stable_online_version} experimental_online_version=${experimental_online_version}"
|
||||||
|
echo "stable_current_version=${stable_current_version} latest_current_version=${latest_current_version}"
|
||||||
|
|
||||||
|
if [[ -z "${stable_online_version}" ]] || [[ -z "${experimental_online_version}" ]]; then
|
||||||
|
exit
|
||||||
|
fi
|
||||||
|
if [[ "${stable_current_version}" == "${stable_online_version}" ]] && [[ "${latest_current_version}" == "${experimental_online_version}" ]]; then
|
||||||
|
exit
|
||||||
|
fi
|
||||||
|
|
||||||
|
function get-semver(){
|
||||||
|
local ver=$1
|
||||||
|
local type=$2
|
||||||
|
if [[ "$ver" =~ $SEMVER_REGEX ]]; then
|
||||||
|
local major=${BASH_REMATCH[1]}
|
||||||
|
local minor=${BASH_REMATCH[2]}
|
||||||
|
local patch=${BASH_REMATCH[3]}
|
||||||
|
fi
|
||||||
|
case $type in
|
||||||
|
major)
|
||||||
|
echo "$major"
|
||||||
|
;;
|
||||||
|
minor)
|
||||||
|
echo "$minor"
|
||||||
|
;;
|
||||||
|
patch)
|
||||||
|
echo "$patch"
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
}
|
||||||
|
|
||||||
|
stableOnlineVersionMajor=$(get-semver "${stable_online_version}" major)
|
||||||
|
stableOnlineVersionMinor=$(get-semver "${stable_online_version}" minor)
|
||||||
|
experimentalOnlineVersionMajor=$(get-semver "${experimental_online_version}" major)
|
||||||
|
experimentalOnlineVersionMinor=$(get-semver "${experimental_online_version}" minor)
|
||||||
|
stableCurrentVersionMajor=$(get-semver "${stable_current_version}" major)
|
||||||
|
stableCurrentVersionMinor=$(get-semver "${stable_current_version}" minor)
|
||||||
|
latestCurrentVersionMajor=$(get-semver "${latest_current_version}" major)
|
||||||
|
latestCurrentVersionMinor=$(get-semver "${latest_current_version}" minor)
|
||||||
|
|
||||||
|
stableOnlineVersionShort=$stableOnlineVersionMajor.$stableOnlineVersionMinor
|
||||||
|
experimentalOnlineVersionShort=$experimentalOnlineVersionMajor.$experimentalOnlineVersionMinor
|
||||||
|
stableCurrentVersionShort=$stableCurrentVersionMajor.$stableCurrentVersionMinor
|
||||||
|
latestCurrentVersionShort=$latestCurrentVersionMajor.$latestCurrentVersionMinor
|
||||||
|
|
||||||
|
echo "stableOnlineVersionShort=${stableOnlineVersionShort} experimentalOnlineVersionShort=${experimentalOnlineVersionShort}"
|
||||||
|
echo "stableCurrentVersionShort=${stableCurrentVersionShort} latestCurrentVersionShort=${latestCurrentVersionShort}"
|
||||||
|
|
||||||
|
# Create new buildinfo.json with only current versions
|
||||||
|
tmpfile=$(mktemp)
|
||||||
|
|
||||||
|
# Start with empty JSON object
|
||||||
|
echo '{}' > "$tmpfile"
|
||||||
|
|
||||||
|
# Add stable version
|
||||||
|
if [[ "$stable_online_version" == "$experimental_online_version" ]]; then
|
||||||
|
# Stable and experimental are the same version
|
||||||
|
jq --arg stable_online_version "$stable_online_version" --arg sha256 "$stable_sha256" --arg stableOnlineVersionShort "$stableOnlineVersionShort" --arg stableOnlineVersionMajor "$stableOnlineVersionMajor" \
|
||||||
|
'. + {($stable_online_version): {sha256: $sha256, tags: ["latest", "stable", ("stable-" + $stable_online_version), $stableOnlineVersionMajor, $stableOnlineVersionShort, $stable_online_version]}}' "$tmpfile" > buildinfo.json
|
||||||
|
else
|
||||||
|
# Different stable and experimental versions
|
||||||
|
# First add stable
|
||||||
|
jq --arg stable_online_version "$stable_online_version" --arg sha256 "$stable_sha256" --arg stableOnlineVersionShort "$stableOnlineVersionShort" --arg stableOnlineVersionMajor "$stableOnlineVersionMajor" \
|
||||||
|
'. + {($stable_online_version): {sha256: $sha256, tags: ["stable", ("stable-" + $stable_online_version), $stableOnlineVersionMajor, $stableOnlineVersionShort, $stable_online_version]}}' "$tmpfile" > buildinfo.json.tmp
|
||||||
|
mv buildinfo.json.tmp "$tmpfile"
|
||||||
|
|
||||||
|
# Then add experimental
|
||||||
|
if [[ $stableOnlineVersionShort == "$experimentalOnlineVersionShort" ]]; then
|
||||||
|
jq --arg experimental_online_version "$experimental_online_version" --arg sha256 "$experimental_sha256" \
|
||||||
|
'. + {($experimental_online_version): {sha256: $sha256, tags: ["latest", $experimental_online_version]}}' "$tmpfile" > buildinfo.json
|
||||||
|
else
|
||||||
|
jq --arg experimental_online_version "$experimental_online_version" --arg sha256 "$experimental_sha256" --arg experimentalOnlineVersionShort "$experimentalOnlineVersionShort" --arg experimentalOnlineVersionMajor "$experimentalOnlineVersionMajor" \
|
||||||
|
'. + {($experimental_online_version): {sha256: $sha256, tags: ["latest", $experimentalOnlineVersionMajor, $experimentalOnlineVersionShort, $experimental_online_version]}}' "$tmpfile" > buildinfo.json
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
rm -f -- "$tmpfile"
|
||||||
|
|
||||||
|
# Generate README tags with logical sorting and de-duplication
|
||||||
|
# First, collect all unique tags with their versions
|
||||||
|
# Use regular arrays for bash compatibility
|
||||||
|
declare tag_versions
|
||||||
|
while IFS= read -r version; do
|
||||||
|
while IFS= read -r tag; do
|
||||||
|
# If this tag is already seen, compare versions to keep the latest
|
||||||
|
if [[ -n "${tag_versions[$tag]}" ]]; then
|
||||||
|
# Compare version strings - keep the higher one
|
||||||
|
if [[ "$version" > "${tag_versions[$tag]}" ]]; then
|
||||||
|
tag_versions[$tag]="$version"
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
tag_versions[$tag]="$version"
|
||||||
|
fi
|
||||||
|
done < <(jq -r ".\"$version\".tags[]" buildinfo.json)
|
||||||
|
done < <(jq -r 'keys[]' buildinfo.json | sort -V -r)
|
||||||
|
|
||||||
|
# Build the tags list for README
|
||||||
|
readme_tags=""
|
||||||
|
# First add the current latest and stable tags
|
||||||
|
latest_version=$(jq -r 'to_entries | map(select(.value.tags | contains(["latest"]))) | .[0].key' buildinfo.json)
|
||||||
|
stable_version=$(jq -r 'to_entries | map(select(.value.tags | index("stable"))) | .[0].key' buildinfo.json)
|
||||||
|
|
||||||
|
if [[ -n "$latest_version" ]]; then
|
||||||
|
latest_tags=$(jq -r ".\"$latest_version\".tags | map(select(. == \"latest\" or . == \"$latest_version\")) | join(\", \")" buildinfo.json | sed 's/"/`/g')
|
||||||
|
readme_tags="${readme_tags}\n* \`${latest_tags}\`"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ -n "$stable_version" ]] && [[ "$stable_version" != "$latest_version" ]]; then
|
||||||
|
stable_tags=$(jq -r ".\"$stable_version\".tags | sort | join(\", \")" buildinfo.json | sed 's/"/`/g')
|
||||||
|
readme_tags="${readme_tags}\n* \`${stable_tags}\`"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Add major.minor tags (e.g., 2.0, 1.1) - only the latest version for each
|
||||||
|
declare -A major_minor_seen
|
||||||
|
while IFS= read -r version; do
|
||||||
|
if [[ "$version" =~ ^([0-9]+)\.([0-9]+)\.([0-9]+)$ ]]; then
|
||||||
|
major="${BASH_REMATCH[1]}"
|
||||||
|
minor="${BASH_REMATCH[2]}"
|
||||||
|
major_minor="$major.$minor"
|
||||||
|
|
||||||
|
# Skip if this is the latest or stable version (already added above)
|
||||||
|
if [[ "$version" == "$latest_version" ]] || [[ "$version" == "$stable_version" ]]; then
|
||||||
|
continue
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Only add if we haven't seen this major.minor yet
|
||||||
|
if [[ -z "${major_minor_seen[$major_minor]}" ]]; then
|
||||||
|
major_minor_seen[$major_minor]=1
|
||||||
|
tags=$(jq -r ".\"$version\".tags | join(\", \")" buildinfo.json | sed 's/"/`/g')
|
||||||
|
if [[ -n "$tags" ]]; then
|
||||||
|
readme_tags="${readme_tags}\n* \`${tags}\`"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
done < <(jq -r 'keys[]' buildinfo.json | sort -V -r)
|
||||||
|
|
||||||
|
readme_tags="${readme_tags}\n"
|
||||||
|
|
||||||
|
perl -i -0777 -pe "s/<!-- start autogeneration tags -->.+<!-- end autogeneration tags -->/<!-- start autogeneration tags -->$readme_tags<!-- end autogeneration tags -->/s" README.md
|
||||||
|
|
||||||
|
# Replace VERSION and SHA256 args in docker-compose.yaml with latest stable values.
|
||||||
|
docker_compose_path="docker/docker-compose.yml"
|
||||||
|
sov="VERSION=${stable_online_version}" yq -i '.services.factorio.build.args[0] = env(sov)' "$docker_compose_path"
|
||||||
|
sha="SHA256=${stable_sha256}" yq -i '.services.factorio.build.args[1] = env(sha)' "$docker_compose_path"
|
||||||
|
|
||||||
|
git config user.name github-actions[bot]
|
||||||
|
git config user.email 41898282+github-actions[bot]@users.noreply.github.com
|
||||||
|
|
||||||
|
git add buildinfo.json
|
||||||
|
git add README.md
|
||||||
|
git add docker/docker-compose.yml
|
||||||
|
git commit -a -m "Auto Update Factorio to stable version: ${stable_online_version} experimental version: ${experimental_online_version}"
|
||||||
|
|
||||||
|
git tag -f latest
|
||||||
|
git push
|
||||||
|
git push origin --tags -f
|
Reference in New Issue
Block a user