Compare commits
45 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| c35a869afd | |||
| 7e0d6618ad | |||
| 0b35588788 | |||
| 4ca9047229 | |||
| 5cdb679fba | |||
| de0950a0ff | |||
| 6de057bbc8 | |||
| 1c52e8c3e9 | |||
| 9c84514759 | |||
| 319443102e | |||
| 64adca6bcd | |||
| bd93c3a466 | |||
| a40b06c394 | |||
| 8d8579361d | |||
| 370dcf1dfe | |||
| 3c7f6d955e | |||
| 8cda10298a | |||
| 334c1cb790 | |||
| ba8e819123 | |||
| 64e6204d18 | |||
| ff3b74d111 | |||
| 6383e6a6ef | |||
| def1ed9109 | |||
| b2b78ee21f | |||
| 0a9373bb2b | |||
| b68375d6be | |||
| 926b6e58fb | |||
| 4870e4b53f | |||
| 4b52cc8c46 | |||
| b7d97ffd41 | |||
| 3666fc9e6a | |||
| 64bec1b12c | |||
| 6f0c2a9d54 | |||
| 476e6fd9fc | |||
| e6b0ebcab9 | |||
| 1ec7b281e8 | |||
| cee4225919 | |||
| 23c5973a87 | |||
| 81d3d14e0c | |||
| e10730cbe0 | |||
| 84851c8714 | |||
| 2ba13f0dbc | |||
| 8783bdbd8a | |||
| 6cf7022897 | |||
| 039f8fb3de |
@@ -1,12 +0,0 @@
|
|||||||
.git
|
|
||||||
.gitignore
|
|
||||||
|
|
||||||
./alpine
|
|
||||||
./Makefile
|
|
||||||
|
|
||||||
*.yml
|
|
||||||
*.yaml
|
|
||||||
|
|
||||||
./LICENSE
|
|
||||||
./README.md
|
|
||||||
./img
|
|
||||||
-26
@@ -1,26 +0,0 @@
|
|||||||
.DS_Store
|
|
||||||
.AppleDouble
|
|
||||||
.LSOverride
|
|
||||||
|
|
||||||
# Icon must end with two \r
|
|
||||||
|
|
||||||
Icon\r\r
|
|
||||||
|
|
||||||
# Thumbnails
|
|
||||||
._*
|
|
||||||
|
|
||||||
# Files that might appear in the root of a volume
|
|
||||||
.DocumentRevisions-V100
|
|
||||||
.fseventsd
|
|
||||||
.Spotlight-V100
|
|
||||||
.TemporaryItems
|
|
||||||
.Trashes
|
|
||||||
.VolumeIcon.icns
|
|
||||||
.com.apple.timemachine.donotpresent
|
|
||||||
|
|
||||||
# Directories potentially created on remote AFP share
|
|
||||||
.AppleDB
|
|
||||||
.AppleDesktop
|
|
||||||
Network Trash Folder
|
|
||||||
Temporary Items
|
|
||||||
.apdisk
|
|
||||||
@@ -1,61 +0,0 @@
|
|||||||
# CI/CD 的阶段定义,按顺序执行各阶段;默认包含`.pre`(最先执行)/`.post`(最后执行)两个阶段,不用显示定义
|
|
||||||
stages:
|
|
||||||
- build
|
|
||||||
- test
|
|
||||||
- deploy
|
|
||||||
|
|
||||||
# 全局变量定义
|
|
||||||
variables:
|
|
||||||
IMG_URL: "$HARBOR_HOST/$HARBOR_PROJECT/$CI_PROJECT_NAME"
|
|
||||||
IMG_TAG: ":latest"
|
|
||||||
|
|
||||||
# 默认值信息配置
|
|
||||||
default:
|
|
||||||
# 各 stage 使用的默认镜像,如果不定义,则为 gitlab-runner 创建时指定的镜像;各 stage 可以覆盖该值以使用不同的镜像
|
|
||||||
image: docker.colovu.com/library/docker:20.10.16
|
|
||||||
# Gitlab-runner 配置的执行器为 Docker 时,需要 配置对应的 dind 服务
|
|
||||||
services:
|
|
||||||
- name: docker.colovu.com/library/docker:20.10.16-dind
|
|
||||||
alias: docker
|
|
||||||
# 流水线中,各阶段都会执行的脚本命令,包括`before_script`(在各阶段 script 前执行)/`after_script`(在各阶段 script 后执行)
|
|
||||||
before_script:
|
|
||||||
- |
|
|
||||||
if [[ "$CI_COMMIT_BRANCH" == "$CI_DEFAULT_BRANCH" ]]; then
|
|
||||||
IMG_TAG=":latest"
|
|
||||||
else
|
|
||||||
IMG_TAG=":$CI_COMMIT_REF_SLUG"
|
|
||||||
fi
|
|
||||||
- docker login -u "$HARBOR_USERNAME" -p "$HARBOR_PASSWORD" $HARBOR_URL
|
|
||||||
|
|
||||||
# 编译阶段任务
|
|
||||||
build-arm64:
|
|
||||||
stage: build
|
|
||||||
script:
|
|
||||||
- export
|
|
||||||
- env
|
|
||||||
- docker buildx build --platform=linux/arm64 --pull -t "$IMG_URL$IMG_TAG-linux-arm64" . --push
|
|
||||||
|
|
||||||
build-amd64:
|
|
||||||
stage: build
|
|
||||||
script:
|
|
||||||
- docker buildx build --platform=linux/amd64 --pull -t "$IMG_URL$IMG_TAG-linux-amd64" . --push
|
|
||||||
|
|
||||||
build-artifact:
|
|
||||||
stage: build
|
|
||||||
needs: [build-amd64, build-arm64]
|
|
||||||
script:
|
|
||||||
- docker manifest create "$IMG_URL$IMG_TAG" "$IMG_URL$IMG_TAG-linux-arm64" "$IMG_URL$IMG_TAG-linux-amd64"
|
|
||||||
- docker manifest push "$IMG_URL$IMG_TAG"
|
|
||||||
|
|
||||||
# 测试阶段任务
|
|
||||||
test:
|
|
||||||
stage: test
|
|
||||||
script:
|
|
||||||
- docker run --rm --platform=linux/arm64 "$IMG_URL$IMG_TAG" /bin/uname -a
|
|
||||||
- docker run --rm --platform=linux/amd64 "$IMG_URL$IMG_TAG" /bin/uname -a
|
|
||||||
|
|
||||||
# 部署阶段任务
|
|
||||||
deploy:
|
|
||||||
stage: deploy
|
|
||||||
script:
|
|
||||||
- echo "deploy stage"
|
|
||||||
-75
@@ -1,75 +0,0 @@
|
|||||||
# Ver: 1.9 by Endial Fang (endial@126.com)
|
|
||||||
#
|
|
||||||
|
|
||||||
# 默认变量 ========================================================================
|
|
||||||
# 该部分变量为系统根据编译命令默认设置
|
|
||||||
|
|
||||||
# `TARGETPLATFORM`:构建后的目标平台信息。如 `linux/amd64`,`linux/arm/v7`,`windows/amd64`
|
|
||||||
# `TARGETOS`:目标平台信息(TARGETPLATFORM)中的操作系统部分,如:`linux`、`windows`
|
|
||||||
# `TARGETARCH`:目标平台信息(TARGETPLATFORM)中的平台架构部分,如:`amd64`、`arm`
|
|
||||||
# `TARGETVARIANT`:目标平台信息(TARGETPLATFORM)中的版本变体部分,如:`v7`
|
|
||||||
# `BUILDPLATFORM`:用于构建的节点平台信息
|
|
||||||
# `BUILDOS`:用于构建的节点平台信息(BUILDPLATFORM)中的操作系统部分
|
|
||||||
# `BUILDARCH`用于构建的节点平台信息(BUILDPLATFORM)中的平台架构部分
|
|
||||||
# `BUILDVARIANT`用于构建的节点平台信息(BUILDPLATFORM)中的版本变体部分
|
|
||||||
|
|
||||||
# 可变参数 ========================================================================
|
|
||||||
# 该部分变量,在编译命令中通过 `--build-arg` 传入;如果未设置,则使用下面对应的默认值
|
|
||||||
|
|
||||||
# 设置当前应用名称及版本
|
|
||||||
ARG APP_NAME=alpine
|
|
||||||
ARG APP_VER=3.16
|
|
||||||
|
|
||||||
# 设置默认仓库地址,默认为本地仓库;定义时需要包含末尾的`/`
|
|
||||||
ARG REGISTRY_URL="docker.colovu.com/"
|
|
||||||
|
|
||||||
# 设置 apt-get 源:default / ustc / aliyun
|
|
||||||
ARG APT_SOURCE=aliyun
|
|
||||||
|
|
||||||
# 编译镜像时指定用于加速的本地软件包存储服务器地址
|
|
||||||
ARG LOCAL_URL=""
|
|
||||||
|
|
||||||
# 1. 生成镜像 =====================================================================
|
|
||||||
FROM --platform=${TARGETPLATFORM:-linux/amd64} ${REGISTRY_URL}library/alpine:${APP_VER}
|
|
||||||
|
|
||||||
# 声明需要使用的全局可变参数
|
|
||||||
ARG APP_NAME
|
|
||||||
ARG APP_VER
|
|
||||||
ARG REGISTRY_URL
|
|
||||||
ARG APT_SOURCE
|
|
||||||
ARG LOCAL_URL
|
|
||||||
ARG TARGETARCH
|
|
||||||
|
|
||||||
LABEL \
|
|
||||||
"Version"="v${APP_VER}" \
|
|
||||||
"Description"="Docker image for Alpine." \
|
|
||||||
"Github"="https://github.com/colovu/docker-alpine" \
|
|
||||||
"Vendor"="Endial Fang (endial@126.com)"
|
|
||||||
|
|
||||||
# 拷贝默认的通用脚本文件
|
|
||||||
COPY prebuilds /
|
|
||||||
|
|
||||||
# 选择软件包源(Optional),以加速后续软件包安装
|
|
||||||
RUN select_source ${APT_SOURCE}
|
|
||||||
|
|
||||||
# 增加musl版本的locales支持,并设置默认为 UTF-8
|
|
||||||
RUN apk add --no-cache libintl; \
|
|
||||||
apk add --no-cache --virtual .locale_build git cmake make musl-dev gcc gettext-dev; \
|
|
||||||
git clone https://gitlab.com/rilian-la-te/musl-locales; \
|
|
||||||
cd musl-locales && cmake -DLOCALE_PROFILE=OFF -DCMAKE_INSTALL_PREFIX:PATH=/usr . && make && make install; \
|
|
||||||
cd .. && rm -r musl-locales; \
|
|
||||||
apk del .locale_build; \
|
|
||||||
rm -rf /var/cache/apk/*;
|
|
||||||
|
|
||||||
# 配置时区默认为 Shanghai
|
|
||||||
RUN install_pkg bash tzdata curl; \
|
|
||||||
cp /usr/share/zoneinfo/Asia/Shanghai /etc/localtime;
|
|
||||||
|
|
||||||
ENV LANG=en_US.UTF-8 \
|
|
||||||
LANGUAGE=en_US.UTF-8 \
|
|
||||||
LC_ALL=en_US.UTF-8
|
|
||||||
|
|
||||||
WORKDIR /
|
|
||||||
|
|
||||||
# 应用程序的服务命令,必须使用非守护进程方式运行。如果使用变量,则该变量必须在运行环境中存在(ENV可以获取)
|
|
||||||
CMD []
|
|
||||||
@@ -1,21 +0,0 @@
|
|||||||
MIT License
|
|
||||||
|
|
||||||
Copyright (c) 2020 Endial Fang (endial@126.com)
|
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
@@ -1,52 +0,0 @@
|
|||||||
# Ver: 1.9 by Endial Fang (endial@126.com)
|
|
||||||
#
|
|
||||||
# 当前 Docker 镜像的编译脚本
|
|
||||||
|
|
||||||
# 定义镜像名称
|
|
||||||
image_name :=colovu/alpine
|
|
||||||
|
|
||||||
# 定义默认镜像仓库地址
|
|
||||||
REGISTRY_URL :=docker.colovu.com
|
|
||||||
|
|
||||||
# 定义系统默认使用的源服务器,包含:default / ustc / aliyun
|
|
||||||
APT_SOURCE :=aliyun
|
|
||||||
|
|
||||||
# 定义镜像TAG,类似:
|
|
||||||
# <镜像名>:<分支名>-<7位Git ID> # Git 仓库且无文件修改直接编译
|
|
||||||
# <镜像名>:<分支名>-<年月日>-<时分秒> # Git 仓库有文件修改后的编译
|
|
||||||
# <镜像名>:latest-<年月日>-<时分秒> # 非 Git 仓库编译
|
|
||||||
current_subversion:=$(shell if [ ! `git status >/dev/null 2>&1` ]; then git rev-parse --short HEAD; else date +%y%m%d-%H%M%S; fi)
|
|
||||||
image_tag:=$(shell if [ ! `git status >/dev/null 2>&1` ]; then git rev-parse --abbrev-ref HEAD | sed -e 's/master/latest/'; else echo "latest"; fi)-$(current_subversion)
|
|
||||||
|
|
||||||
build-arg:=--build-arg REGISTRY_URL=$(REGISTRY_URL)
|
|
||||||
build-arg+=--build-arg APT_SOURCE=$(APT_SOURCE)
|
|
||||||
|
|
||||||
# 设置本地下载服务器路径,加速调试时的本地编译速度
|
|
||||||
local_ip:=`echo "en0 eth0" | xargs -n1 ip addr show 2>/dev/null | grep inet | grep -v 127.0.0.1 | grep -v inet6 | tr "/" " " | awk '{print $$2}'`
|
|
||||||
build-arg+=--build-arg LOCAL_URL=http://$(local_ip)/dist-files
|
|
||||||
|
|
||||||
.PHONY: build clean clearclean upgrade
|
|
||||||
|
|
||||||
build:
|
|
||||||
@echo "Build $(image_name):$(image_tag)"
|
|
||||||
@docker build --force-rm $(build-arg) -t $(image_name):$(image_tag) .
|
|
||||||
@echo "Add tag: $(image_name):latest"
|
|
||||||
@docker tag $(image_name):$(image_tag) $(image_name):latest
|
|
||||||
@echo "Build complete"
|
|
||||||
|
|
||||||
# 清理悬空的镜像(无TAG)及停止的容器
|
|
||||||
clearclean: clean
|
|
||||||
@echo "Clean untaged images and stoped containers..."
|
|
||||||
@docker ps -a | grep "Exited" | awk '{print $$1}' | sort -u | xargs -L 1 docker rm
|
|
||||||
@docker images | grep '<none>' | awk '{print $$3}' | sort -u | xargs -L 1 docker rmi -f
|
|
||||||
|
|
||||||
# 为了防止删除前缀名相同的镜像,在过滤条件中加入一个空格进行过滤
|
|
||||||
clean:
|
|
||||||
@echo "Clean all images for current application..."
|
|
||||||
@docker images | grep "$(image_name) " | awk '{print $$3}' | sort -u | xargs -L 1 docker rmi -f
|
|
||||||
|
|
||||||
# 更新所有 colovu 仓库的镜像
|
|
||||||
upgrade:
|
|
||||||
@echo "Upgrade all images..."
|
|
||||||
@docker images | grep 'colovu' | grep -v '<none>' | grep -v "latest-" | awk '{print $$1":"$$2}' | sort -u | xargs -L 1 docker pull
|
|
||||||
|
|
||||||
@@ -1,101 +1,3 @@
|
|||||||
# Alpine
|
# Alpine
|
||||||
|
|
||||||
简单、高性能、小体积的基础 [Alpine](http://www.alpinelinux.org) 系统镜像,用于为其他应用提供基础镜像。
|
停止维护 Alpine v3.18 版本.
|
||||||
|
|
||||||
使用说明可参照:[官方说明](https://wiki.alpinelinux.org)
|
|
||||||
|
|
||||||
|
|
||||||
**版本信息:**
|
|
||||||
|
|
||||||
- 3.16、latest
|
|
||||||
|
|
||||||
**镜像信息:**
|
|
||||||
|
|
||||||
* 镜像地址:
|
|
||||||
* 阿里云: registry.cn-shenzhen.aliyuncs.com/colovu/alpine:latest
|
|
||||||
* Docker Hub: colovu/alpine:latest
|
|
||||||
* Colovu Registry: docker.colovu.com/colovu/alpine:latest
|
|
||||||
* 依赖镜像:docker.io/library/alpine:3.16
|
|
||||||
|
|
||||||
> 后续相关命令行默认使用`[Colovu Registry](https://docker.colovu.com)`镜像服务器做说明
|
|
||||||
|
|
||||||
**与官方镜像差异:**
|
|
||||||
|
|
||||||
- 增加 `default、ustc、aliyun` 源配置文件,可在编译时通过 `ARG` 变量`APT_SOURCE`进行选择
|
|
||||||
- 增加常用 Shell 脚本文件
|
|
||||||
- 更新已安装的软件包
|
|
||||||
- 增加`locales`,并设置默认编码格式为`en_US.utf8`
|
|
||||||
- 增加`bash`
|
|
||||||
- 设置默认时区信息为 `Asia/Shanghai`
|
|
||||||
- 默认增加 nss_wrapper 支持
|
|
||||||
- 默认增加 curl 软件,用作镜像健康检查
|
|
||||||
|
|
||||||
## TL;DR
|
|
||||||
|
|
||||||
Docker 快速启动命令:
|
|
||||||
|
|
||||||
```shell
|
|
||||||
# 从 Registry 服务器下载镜像并启动
|
|
||||||
$ docker run -it docker.colovu.com/colovu/alpine:latest /bin/bash
|
|
||||||
```
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## 使用说明
|
|
||||||
|
|
||||||
**下载镜像:**
|
|
||||||
|
|
||||||
```shell
|
|
||||||
$ docker pull docker.colovu.com/colovu/alpine:latest
|
|
||||||
```
|
|
||||||
|
|
||||||
- latest:为镜像的 TAG,可针对性选择不同的 TAG 进行下载;可使用`latest`选择最新的镜像
|
|
||||||
- 不指定 TAG 时,默认下载`latest`镜像
|
|
||||||
|
|
||||||
**查看镜像:**
|
|
||||||
|
|
||||||
```shell
|
|
||||||
$ docker images
|
|
||||||
```
|
|
||||||
|
|
||||||
**命令行方式运行容器:**
|
|
||||||
|
|
||||||
```shell
|
|
||||||
$ docker run -it --rm docker.colovu.com/colovu/alpine:latest /bin/bash
|
|
||||||
```
|
|
||||||
|
|
||||||
- `-it`:使用交互式终端启动容器
|
|
||||||
- `--rm`:退出时删除容器
|
|
||||||
- `docker.colovu.com/colovu/alpine:latest`:镜像名称及版本标签
|
|
||||||
- `/bin/bash`:在容器中执行`/bin/bash`命令;如果不执行命令,容器会在启动后立即结束并退出。
|
|
||||||
|
|
||||||
以该方式启动后,直接进入容器的命令行操作界面。如果需要退出,直接使用命令`exit`退出。
|
|
||||||
|
|
||||||
**后台方式运行容器:**
|
|
||||||
|
|
||||||
```shell
|
|
||||||
$ docker run -d --name test docker.colovu.com/colovu/alpine:latest tail /dev/stderr
|
|
||||||
```
|
|
||||||
|
|
||||||
- `--name test`:命名容器为`test`
|
|
||||||
- `-d`:以后台进程方式启动容器
|
|
||||||
- `docker.colovu.com/colovu/alpine:latest`:镜像名称及版本标签
|
|
||||||
- `tail /dev/stderr`:在容器中执行`tail /dev/stderr`命令,以防止容器直接退出
|
|
||||||
|
|
||||||
以该方式启动后,如果想进入容器,可以使用以下命令:
|
|
||||||
|
|
||||||
```shell
|
|
||||||
$ docker exec -it test /bin/bash
|
|
||||||
```
|
|
||||||
|
|
||||||
- `-it`:使用交互式执行
|
|
||||||
- `test`:之前启动的容器名
|
|
||||||
- `/bin/bash`:执行的命令
|
|
||||||
|
|
||||||
## 更新记录
|
|
||||||
|
|
||||||
- 3.16、latest
|
|
||||||
|
|
||||||
----
|
|
||||||
|
|
||||||
本文原始来源 [Endial Fang](https://github.com/colovu) @ [Github.com](https://github.com)
|
|
||||||
|
|||||||
@@ -1,2 +0,0 @@
|
|||||||
http://mirrors.aliyun.com/alpine/v3.12/main
|
|
||||||
http://mirrors.aliyun.com/alpine/v3.12/community
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
http://dl-cdn.alpinelinux.org/alpine/v3.12/main
|
|
||||||
http://dl-cdn.alpinelinux.org/alpine/v3.12/community
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
http://mirrors.huaweicloud.com/alpine/v3.12/main
|
|
||||||
http://mirrors.huaweicloud.com/alpine/v3.12/community
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
http://mirrors.cloud.tencent.com/alpine/v3.12/main
|
|
||||||
http://mirrors.cloud.tencent.com/alpine/v3.12/community
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
http://mirrors.ustc.edu.cn/alpine/v3.12/main
|
|
||||||
http://mirrors.ustc.edu.cn/alpine/v3.12/community
|
|
||||||
@@ -1,21 +0,0 @@
|
|||||||
MIT License
|
|
||||||
|
|
||||||
Copyright (c) 2020 Endial Fang (endial@126.com)
|
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
@@ -1,119 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
# Ver: 1.3 by Endial Fang (endial@126.com)
|
|
||||||
#
|
|
||||||
# 通用函数库
|
|
||||||
|
|
||||||
# 加载依赖项
|
|
||||||
. /usr/local/scripts/liblog.sh # 日志输出函数库
|
|
||||||
|
|
||||||
# 函数列表
|
|
||||||
|
|
||||||
# 打印包含包含Logo的欢迎信息
|
|
||||||
print_welcome_info() {
|
|
||||||
[[ -n "${APP_NAME}" ]] && github_url="/docker-${APP_NAME}"
|
|
||||||
|
|
||||||
LOG_I ' ____ _ '
|
|
||||||
LOG_I ' / ___|___ | | _____ ___ _ '
|
|
||||||
LOG_I '| | / _ \| |/ _ \ \ / / | | | '"Docker : ${BOLD}${APP_NAME:-undefined}${RESET}"
|
|
||||||
LOG_I '| |__| (_) | | (_) \ V /| |_| | '"Version: ${BOLD}${APP_VERSION:-0.0}${RESET}"
|
|
||||||
LOG_I ' \____\___/|_|\___/ \_/ \__,_| '"PowerBy: ${BOLD}Endial@126.com${RESET}"
|
|
||||||
LOG_D " Project Repo: https://github.com/colovu/${github_url:-}"
|
|
||||||
LOG_I ""
|
|
||||||
}
|
|
||||||
|
|
||||||
# 根据需要打印欢迎信息
|
|
||||||
print_image_welcome() {
|
|
||||||
if [[ "$(id -u)" = "0" ]]; then
|
|
||||||
print_welcome_info
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
# 检测可能导致容器执行后直接退出的命令,如"--help";如果存在,直接返回 0
|
|
||||||
# 参数:
|
|
||||||
# $1 - 待检测的参数表
|
|
||||||
print_command_help() {
|
|
||||||
local arg
|
|
||||||
for arg; do
|
|
||||||
case "$arg" in
|
|
||||||
-'?'|--help|-V|--version|-version)
|
|
||||||
exec "$@"
|
|
||||||
exit
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
done
|
|
||||||
}
|
|
||||||
|
|
||||||
# 检测应用相应的配置文件是否存在,如果不存在,则从默认配置文件目录拷贝一份
|
|
||||||
# 默认配置文件路径:/etc/${APP_NAME}
|
|
||||||
# 目标配置文件路径:/srv/conf/${APP_NAME}
|
|
||||||
# 参数:
|
|
||||||
# $1 - 基础路径
|
|
||||||
# $* - 基础路径下的文件及目录列表,以" "分割
|
|
||||||
# 例子:
|
|
||||||
# ensure_config_file_exist /etc/${APP_NAME} conf.d server.conf
|
|
||||||
ensure_config_file_exist() {
|
|
||||||
local -r base_path="${1:?paths is missing}"
|
|
||||||
local f=""
|
|
||||||
local dist=""
|
|
||||||
|
|
||||||
shift 1
|
|
||||||
LOG_D "List to check: $@"
|
|
||||||
while [ "$#" -gt 0 ]; do
|
|
||||||
f="${1}"
|
|
||||||
LOG_D " Process \"${f}\""
|
|
||||||
if [ -d "${base_path}/${f}" ]; then
|
|
||||||
dist="$(echo ${base_path}/${f} | sed -e 's/\/etc/\/srv\/conf/g')"
|
|
||||||
[[ ! -d "${dist}" ]] && LOG_D " Create directory: ${dist}" && mkdir -p "${dist}"
|
|
||||||
[[ ! -z $(ls -A "${base_path}/${f}") ]] && ensure_config_file_exist "${base_path}/${f}" $(ls -A "${base_path}/${f}")
|
|
||||||
else
|
|
||||||
dist="$(echo ${base_path}/${f} | sed -e 's/\/etc/\/srv\/conf/g')"
|
|
||||||
[[ ! -e "${dist}" ]] && LOG_D " Copy: ${base_path}/${f} ===> ${dist}" && cp "${base_path}/${f}" "${dist}" && rm -rf "/srv/conf/${APP_NAME}/.app_init_flag"
|
|
||||||
fi
|
|
||||||
shift
|
|
||||||
done
|
|
||||||
}
|
|
||||||
|
|
||||||
# 根据脚本扩展名及权限,执行相应的初始化脚本
|
|
||||||
# 参数:
|
|
||||||
# $1 - 文件列表,支持路径通配符
|
|
||||||
# 使用:
|
|
||||||
# process_init_files [file [file [...]]]
|
|
||||||
# 例子:
|
|
||||||
# process_init_files /src/conf/${APP_NAME}/initdb.d/*
|
|
||||||
process_init_files() {
|
|
||||||
echo
|
|
||||||
local f
|
|
||||||
for f; do
|
|
||||||
case "$f" in
|
|
||||||
*.sh)
|
|
||||||
if [ -x "$f" ]; then
|
|
||||||
LOG_I "$0: running $f"
|
|
||||||
"$f"
|
|
||||||
else
|
|
||||||
LOG_I "$0: sourcing $f"
|
|
||||||
. "$f"
|
|
||||||
fi
|
|
||||||
;;
|
|
||||||
*) LOG_W "$0: ignoring $f" ;;
|
|
||||||
esac
|
|
||||||
echo
|
|
||||||
done
|
|
||||||
}
|
|
||||||
|
|
||||||
# 检测当前是否为 root 用户
|
|
||||||
is_root() {
|
|
||||||
if [[ "$(id -u)" = "0" ]]; then
|
|
||||||
LOG_D "Run as root."
|
|
||||||
true
|
|
||||||
else
|
|
||||||
LOG_D "Run as non-root: $(id -u)"
|
|
||||||
false
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
# 检测当前脚本是被直接执行的,还是从其他脚本中使用 "source" 调用的
|
|
||||||
is_sourced() {
|
|
||||||
[ "${#FUNCNAME[@]}" -ge 2 ] \
|
|
||||||
&& [ "${FUNCNAME[0]}" = 'is_sourced' ] \
|
|
||||||
&& [ "${FUNCNAME[1]}" = 'source' ]
|
|
||||||
}
|
|
||||||
@@ -1,78 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
# Ver: 1.0 by Endial Fang (endial@126.com)
|
|
||||||
#
|
|
||||||
# 文件操作函数库
|
|
||||||
|
|
||||||
# 加载依赖项
|
|
||||||
. /usr/local/scripts/liblog.sh # 日志输出函数库
|
|
||||||
|
|
||||||
# 函数列表
|
|
||||||
|
|
||||||
# 检测"*_FILE"文件,并从文件中读取信息作为参数值;环境变量不允许 VAR 与 VAR_FILE 方式并存
|
|
||||||
# 变量:
|
|
||||||
# $1 - 需要设置的环境变量名称
|
|
||||||
# $2 - 该变量对应的默认值(Option)
|
|
||||||
#
|
|
||||||
# 使用: file_env ENV_VAR [DEFAULT]
|
|
||||||
file_env() {
|
|
||||||
local var="$1"
|
|
||||||
local fileVar="${var}_FILE"
|
|
||||||
local def="${2:-}"
|
|
||||||
if [ "${!var:-}" ] && [ "${!fileVar:-}" ]; then
|
|
||||||
LOG_E "Both $var and $fileVar are set (but are exclusive)"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
local val="$def"
|
|
||||||
if [ "${!var:-}" ]; then
|
|
||||||
val="${!var}"
|
|
||||||
elif [ "${!fileVar:-}" ]; then
|
|
||||||
val="$(< "${!fileVar}")"
|
|
||||||
fi
|
|
||||||
|
|
||||||
export "$var"="$val"
|
|
||||||
unset "$fileVar"
|
|
||||||
}
|
|
||||||
|
|
||||||
# 使用规则表达式在文件中替换数据
|
|
||||||
# 参数:
|
|
||||||
# $1 - 文件名
|
|
||||||
# $2 - 正则表达式
|
|
||||||
# $3 - 替代数据表达式
|
|
||||||
# $4 - 是否使用POSIX表达式. Default: true
|
|
||||||
replace_in_file() {
|
|
||||||
local filename="${1:?filename is required}"
|
|
||||||
local match_regex="${2:?match regex is required}"
|
|
||||||
local substitute_regex="${3:?substitute regex is required}"
|
|
||||||
local posix_regex=${4:-true}
|
|
||||||
|
|
||||||
local result
|
|
||||||
|
|
||||||
# 因部分系统兼容性问题,需要防止使用 'sed in-place' 方式操作
|
|
||||||
if [[ $posix_regex = true ]]; then
|
|
||||||
result="$(sed -E "s@$match_regex@$substitute_regex@g" "$filename")"
|
|
||||||
else
|
|
||||||
result="$(sed "s@$match_regex@$substitute_regex@g" "$filename")"
|
|
||||||
fi
|
|
||||||
echo "$result" > "$filename"
|
|
||||||
}
|
|
||||||
|
|
||||||
# 使用规则表达式在文件中删除数据
|
|
||||||
# 参数:
|
|
||||||
# $1 - 文件名
|
|
||||||
# $2 - 正则表达式
|
|
||||||
# $3 - 是否使用POSIX表达式. Default: true
|
|
||||||
remove_in_file() {
|
|
||||||
local filename="${1:?filename is required}"
|
|
||||||
local match_regex="${2:?match regex is required}"
|
|
||||||
local posix_regex=${3:-true}
|
|
||||||
local result
|
|
||||||
|
|
||||||
# 因部分系统兼容性问题,需要防止使用 'sed in-place' 方式操作
|
|
||||||
if [[ $posix_regex = true ]]; then
|
|
||||||
result="$(sed -E "/$match_regex/d" "$filename")"
|
|
||||||
else
|
|
||||||
result="$(sed "/$match_regex/d" "$filename")"
|
|
||||||
fi
|
|
||||||
echo "$result" > "$filename"
|
|
||||||
}
|
|
||||||
@@ -1,107 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
# Ver: 1.1 by Endial Fang (endial@126.com)
|
|
||||||
#
|
|
||||||
# 文件管理函数库
|
|
||||||
|
|
||||||
# 加载依赖项
|
|
||||||
. /usr/local/scripts/liblog.sh # 日志输出函数库
|
|
||||||
|
|
||||||
# 函数列表
|
|
||||||
|
|
||||||
# 检测目录是否存在,如果不存在则创建,同时修改为指定的用户
|
|
||||||
# 参数:
|
|
||||||
# $1 - 目录路径
|
|
||||||
# $2 - 用户
|
|
||||||
ensure_dir_exists() {
|
|
||||||
local dir="${1:?directory is missing}"
|
|
||||||
local owner="${2:-}"
|
|
||||||
|
|
||||||
mkdir -p "${dir}"
|
|
||||||
if [[ -n $owner ]]; then
|
|
||||||
chown "$owner":"$owner" "$dir"
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
# 检测目录是否存在或为空
|
|
||||||
# 参数:
|
|
||||||
# $1 - 目录路径
|
|
||||||
is_dir_empty() {
|
|
||||||
local dir="${1:?missing directory}"
|
|
||||||
|
|
||||||
if [[ ! -e "$dir" ]] || [[ -z "$(ls -A "$dir")" ]]; then
|
|
||||||
true
|
|
||||||
else
|
|
||||||
false
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
# 循环设置目录中子目录及文件权限
|
|
||||||
# 参数:
|
|
||||||
# $1 - paths (as a string).
|
|
||||||
# Flags:
|
|
||||||
# -f|--file-mode - 文件权限模式
|
|
||||||
# -d|--dir-mode - 目录权限模式
|
|
||||||
# -u|--user - 用户
|
|
||||||
# -g|--group - 用户组
|
|
||||||
configure_permissions_ownership() {
|
|
||||||
local -r paths="${1:?paths is missing}"
|
|
||||||
local dir_mode=""
|
|
||||||
local file_mode=""
|
|
||||||
local user=""
|
|
||||||
local group=""
|
|
||||||
|
|
||||||
# Validate arguments
|
|
||||||
shift 1
|
|
||||||
while [ "$#" -gt 0 ]; do
|
|
||||||
case "$1" in
|
|
||||||
-f|--file-mode)
|
|
||||||
shift
|
|
||||||
file_mode="${1:?missing mode for files}"
|
|
||||||
;;
|
|
||||||
-d|--dir-mode)
|
|
||||||
shift
|
|
||||||
dir_mode="${1:?missing mode for directories}"
|
|
||||||
;;
|
|
||||||
-u|--user)
|
|
||||||
shift
|
|
||||||
user="${1:?missing user}"
|
|
||||||
;;
|
|
||||||
-g|--group)
|
|
||||||
shift
|
|
||||||
group="${1:?missing group}"
|
|
||||||
;;
|
|
||||||
*)
|
|
||||||
LOG_E "Invalid command line flag $1" >&2
|
|
||||||
return 1
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
shift
|
|
||||||
done
|
|
||||||
|
|
||||||
read -r -a filepaths <<< "$paths"
|
|
||||||
for p in "${filepaths[@]}"; do
|
|
||||||
if [[ -e "$p" ]]; then
|
|
||||||
LOG_D "Check $p"
|
|
||||||
if [[ -n ${dir_mode} ]]; then
|
|
||||||
LOG_D "Change permissions to ${dir_mode} of directories in $p"
|
|
||||||
find -L "$p" -type d -print | xargs -i chmod "${dir_mode}" '{}'
|
|
||||||
fi
|
|
||||||
if [[ -n ${file_mode} ]]; then
|
|
||||||
LOG_D "Change permissions to ${file_mode} of files in $p"
|
|
||||||
find -L "$p" -type f -print | xargs -i chmod "${file_mode}" '{}'
|
|
||||||
fi
|
|
||||||
if [[ -n $user ]] && [[ -n ${group} ]]; then
|
|
||||||
LOG_D "Change ownership to ${user}:${group} of files and directories in $p"
|
|
||||||
find -L "$p" \( \! -user ${user} -or \! -group ${group} \) -print | xargs -i chown -L "${user}":"${group}" '{}'
|
|
||||||
elif [[ -n $user ]] && [[ -z $group ]]; then
|
|
||||||
LOG_D "Change user to ${user} of files and directories in $p"
|
|
||||||
find -L "$p" \! -user ${user} -print | xargs -i chown -L "${user}" '{}'
|
|
||||||
elif [[ -z $user ]] && [[ -n $group ]]; then
|
|
||||||
LOG_D "Change group to ${group} of files and directories in $p"
|
|
||||||
find -L "$p" \! -group ${group} -print | xargs -i chgrp -L "${group}" '{}'
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
LOG_E "$p does not exist"
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
}
|
|
||||||
@@ -1,88 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
# Ver: 1.1 by Endial Fang (endial@126.com)
|
|
||||||
#
|
|
||||||
# 日志输出函数库
|
|
||||||
|
|
||||||
#[[ ${ENV_DEBUG:-false} = true ]] && set -x
|
|
||||||
MODULE="$(basename "$0")"
|
|
||||||
|
|
||||||
RESET='\033[0m'
|
|
||||||
BOLD='\033[1m'
|
|
||||||
|
|
||||||
# 前景色
|
|
||||||
BLACK='\033[38;5;0m'
|
|
||||||
RED='\033[38;5;1m'
|
|
||||||
GREEN='\033[38;5;2m'
|
|
||||||
YELLOW='\033[38;5;3m'
|
|
||||||
BLUE='\033[38;5;4m'
|
|
||||||
MAGENTA='\033[38;5;5m'
|
|
||||||
CYAN='\033[38;5;6m'
|
|
||||||
WHITE='\033[38;5;7m'
|
|
||||||
|
|
||||||
# 背景色
|
|
||||||
ON_BLACK='\033[48;5;0m'
|
|
||||||
ON_RED='\033[48;5;1m'
|
|
||||||
ON_GREEN='\033[48;5;2m'
|
|
||||||
ON_YELLOW='\033[48;5;3m'
|
|
||||||
ON_BLUE='\033[48;5;4m'
|
|
||||||
ON_MAGENTA='\033[48;5;5m'
|
|
||||||
ON_CYAN='\033[48;5;6m'
|
|
||||||
ON_WHITE='\033[48;5;7m'
|
|
||||||
|
|
||||||
# 函数列表
|
|
||||||
|
|
||||||
# 打印输出到 STDERR 设备
|
|
||||||
stderr_print() {
|
|
||||||
printf "%b\\n" "${*}" >&2
|
|
||||||
}
|
|
||||||
|
|
||||||
# 输出实际日志信息
|
|
||||||
# 参数:
|
|
||||||
# $1 - 日志类型
|
|
||||||
# $2 - 日志信息
|
|
||||||
LOG() {
|
|
||||||
local -r bool="${ENV_DEBUG:-false}"
|
|
||||||
shopt -s nocasematch
|
|
||||||
if [[ "$bool" = 1 || "$bool" =~ ^(yes|true)$ ]]; then
|
|
||||||
debugInfo="${CYAN}${APP_NAME:-}:${MODULE:-}"
|
|
||||||
else
|
|
||||||
debugInfo="${CYAN}${APP_NAME:-}"
|
|
||||||
fi
|
|
||||||
stderr_print "${debugInfo} ${MAGENTA}$(date "+%T")${RESET} ${*}"
|
|
||||||
}
|
|
||||||
|
|
||||||
# 输出调试类日志信息,尽量少使用
|
|
||||||
# 参数:
|
|
||||||
# $1 - 日志类型
|
|
||||||
# $2 - 日志信息
|
|
||||||
LOG_D() {
|
|
||||||
local -r bool="${ENV_DEBUG:-false}"
|
|
||||||
shopt -s nocasematch
|
|
||||||
if [[ "$bool" = 1 || "$bool" =~ ^(yes|true)$ ]]; then
|
|
||||||
LOG "${BLUE}DBG${RESET}: ${*}"
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
# 输出提示信息类日志信息
|
|
||||||
# 参数:
|
|
||||||
# $1 - 日志类型
|
|
||||||
# $2 - 日志信息
|
|
||||||
LOG_I() {
|
|
||||||
LOG "${GREEN}INF${RESET}: ${*}"
|
|
||||||
}
|
|
||||||
|
|
||||||
# 输出警告类日志信息至sterr
|
|
||||||
# 参数:
|
|
||||||
# $1 - 日志类型
|
|
||||||
# $2 - 日志信息
|
|
||||||
LOG_W() {
|
|
||||||
LOG "${YELLOW}WRN${RESET}: ${*}"
|
|
||||||
}
|
|
||||||
|
|
||||||
# 输出错误类日志信息至sterr,并退出脚本
|
|
||||||
# 参数:
|
|
||||||
# $1 - 日志类型
|
|
||||||
# $2 - 日志信息
|
|
||||||
LOG_E() {
|
|
||||||
LOG "${RED}ERR${RESET}: ${*}"
|
|
||||||
}
|
|
||||||
@@ -1,120 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
# Ver: 1.1 by Endial Fang (endial@126.com)
|
|
||||||
#
|
|
||||||
# 文件管理函数库
|
|
||||||
|
|
||||||
# 加载依赖项
|
|
||||||
. /usr/local/scripts/liblog.sh # 日志输出函数库
|
|
||||||
|
|
||||||
# 函数列表
|
|
||||||
|
|
||||||
# 域名解析
|
|
||||||
# 参数:
|
|
||||||
# $1 - 需要解析的主机名
|
|
||||||
dns_lookup() {
|
|
||||||
local host="${1:?host is missing}"
|
|
||||||
getent ahosts "$host" | awk '/STREAM/ {print $1 }'
|
|
||||||
}
|
|
||||||
|
|
||||||
# 尝试解析域名并返回对应的 IP
|
|
||||||
# 参数:
|
|
||||||
# $1 - 主机名
|
|
||||||
# $2 - 尝试次数
|
|
||||||
# $3 - 重试间隔时间(秒)
|
|
||||||
wait_for_dns_lookup() {
|
|
||||||
local hostname="${1:?hostname is missing}"
|
|
||||||
local retries="${2:-5}"
|
|
||||||
local seconds="${3:-1}"
|
|
||||||
check_host() {
|
|
||||||
if [[ $(dns_lookup "$hostname") == "" ]]; then
|
|
||||||
false
|
|
||||||
else
|
|
||||||
true
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
# Wait for the host to be ready
|
|
||||||
retry_while "check_host ${hostname}" "$retries" "$seconds"
|
|
||||||
dns_lookup "$hostname"
|
|
||||||
}
|
|
||||||
|
|
||||||
# 获取当前主机 IP
|
|
||||||
get_machine_ip() {
|
|
||||||
local -a ip_addresses
|
|
||||||
local hostname
|
|
||||||
hostname="$(hostname)"
|
|
||||||
read -r -a ip_addresses <<< "$(dns_lookup "$hostname" | xargs echo)"
|
|
||||||
if [[ "${#ip_addresses[@]}" -gt 1 ]]; then
|
|
||||||
LOG_W "Found more than one IP address associated to hostname ${hostname}: ${ip_addresses[*]}, will use ${ip_addresses[0]}"
|
|
||||||
elif [[ "${#ip_addresses[@]}" -lt 1 ]]; then
|
|
||||||
LOG_E "Could not find any IP address associated to hostname ${hostname}"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
echo "${ip_addresses[0]}"
|
|
||||||
}
|
|
||||||
|
|
||||||
# Check if the provided argument is a resolved hostname
|
|
||||||
# 参数:
|
|
||||||
# $1 - 待检测的主机名
|
|
||||||
# 返回值:
|
|
||||||
# 布尔值
|
|
||||||
is_hostname_resolved() {
|
|
||||||
local -r host="${1:?missing value}"
|
|
||||||
if [[ -n "$(dns_lookup "$host")" ]]; then
|
|
||||||
true
|
|
||||||
else
|
|
||||||
false
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
# 解析 URL
|
|
||||||
# 参数:
|
|
||||||
# $1 - URI 字符串
|
|
||||||
# $2 - 类型字符串. 有效值 (scheme, authority, userinfo, host, port, path, query or fragment)
|
|
||||||
# 返回值:
|
|
||||||
# 字符串
|
|
||||||
parse_uri() {
|
|
||||||
local uri="${1:?uri is missing}"
|
|
||||||
local component="${2:?component is missing}"
|
|
||||||
|
|
||||||
# Solution based on https://tools.ietf.org/html/rfc3986#appendix-B with
|
|
||||||
# additional sub-expressions to split authority into userinfo, host and port
|
|
||||||
# Credits to Patryk Obara (see https://stackoverflow.com/a/45977232/6694969)
|
|
||||||
local -r URI_REGEX='^(([^:/?#]+):)?(//((([^@/?#]+)@)?([^:/?#]+)(:([0-9]+))?))?(/([^?#]*))?(\?([^#]*))?(#(.*))?'
|
|
||||||
# || | ||| | | | | | | | | |
|
|
||||||
# |2 scheme | ||6 userinfo 7 host | 9 port | 11 rpath | 13 query | 15 fragment
|
|
||||||
# 1 scheme: | |5 userinfo@ 8 :... 10 path 12 ?... 14 #...
|
|
||||||
# | 4 authority
|
|
||||||
# 3 //...
|
|
||||||
local index=0
|
|
||||||
case "$component" in
|
|
||||||
scheme)
|
|
||||||
index=2
|
|
||||||
;;
|
|
||||||
authority)
|
|
||||||
index=4
|
|
||||||
;;
|
|
||||||
userinfo)
|
|
||||||
index=6
|
|
||||||
;;
|
|
||||||
host)
|
|
||||||
index=7
|
|
||||||
;;
|
|
||||||
port)
|
|
||||||
index=9
|
|
||||||
;;
|
|
||||||
path)
|
|
||||||
index=10
|
|
||||||
;;
|
|
||||||
query)
|
|
||||||
index=13
|
|
||||||
;;
|
|
||||||
fragment)
|
|
||||||
index=14
|
|
||||||
;;
|
|
||||||
*)
|
|
||||||
stderr_print "unrecognized component $component"
|
|
||||||
return 1
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
[[ "$uri" =~ $URI_REGEX ]] && echo "${BASH_REMATCH[${index}]}"
|
|
||||||
}
|
|
||||||
@@ -1,109 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
# Ver: 1.2 by Endial Fang (endial@126.com)
|
|
||||||
#
|
|
||||||
# 操作系统控制函数库
|
|
||||||
|
|
||||||
# 加载依赖项
|
|
||||||
. /usr/local/scripts/liblog.sh # 日志输出函数库
|
|
||||||
|
|
||||||
# 函数列表
|
|
||||||
|
|
||||||
# 检测指定用户账户是否存在
|
|
||||||
# 参数:
|
|
||||||
# $1 - 用户账户
|
|
||||||
user_exists() {
|
|
||||||
local user="${1:?user is missing}"
|
|
||||||
id "$user" >/dev/null 2>&1
|
|
||||||
}
|
|
||||||
|
|
||||||
# 检测指定用户分组是否存在
|
|
||||||
# 参数:
|
|
||||||
# $1 - 用户组
|
|
||||||
group_exists() {
|
|
||||||
local group="${1:?group is missing}"
|
|
||||||
getent group "$group" >/dev/null 2>&1
|
|
||||||
}
|
|
||||||
|
|
||||||
# 获取系统可用内存大小(MB)信息
|
|
||||||
get_total_memory() {
|
|
||||||
echo $(($(grep MemTotal /proc/meminfo | awk '{print $2}') / 1024))
|
|
||||||
}
|
|
||||||
|
|
||||||
# 获取以定量方式描述的内存大小
|
|
||||||
# 参数:
|
|
||||||
# $1 - 内存大小 (MB,可选)
|
|
||||||
get_machine_size() {
|
|
||||||
local memory="${1:-}"
|
|
||||||
if [[ -z "$memory" ]]; then
|
|
||||||
debug "Memory was not specified, detecting available memory automatically"
|
|
||||||
memory="$(get_total_memory)"
|
|
||||||
fi
|
|
||||||
sanitized_memory=$(convert_to_mb "$memory")
|
|
||||||
if [[ "$sanitized_memory" -gt 26000 ]]; then
|
|
||||||
echo 2xlarge
|
|
||||||
elif [[ "$sanitized_memory" -gt 13000 ]]; then
|
|
||||||
echo xlarge
|
|
||||||
elif [[ "$sanitized_memory" -gt 6000 ]]; then
|
|
||||||
echo large
|
|
||||||
elif [[ "$sanitized_memory" -gt 3000 ]]; then
|
|
||||||
echo medium
|
|
||||||
elif [[ "$sanitized_memory" -gt 1500 ]]; then
|
|
||||||
echo small
|
|
||||||
else
|
|
||||||
echo micro
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
# 获取已定义的所有内存大小描述
|
|
||||||
get_supported_machine_sizes() {
|
|
||||||
echo micro small medium large xlarge 2xlarge
|
|
||||||
}
|
|
||||||
|
|
||||||
# 将以字符串表示的内存大小转换为以MB为单位的内存大小值 (i.e. 2G -> 2048)
|
|
||||||
# 参数:
|
|
||||||
# $1 - 内存大小
|
|
||||||
convert_to_mb() {
|
|
||||||
local amount="${1:-}"
|
|
||||||
if [[ $amount =~ ^([0-9]+)(M|G) ]]; then
|
|
||||||
size="${BASH_REMATCH[1]}"
|
|
||||||
unit="${BASH_REMATCH[2]}"
|
|
||||||
if [[ "$unit" = "G" ]]; then
|
|
||||||
amount="$((size * 1024))"
|
|
||||||
else
|
|
||||||
amount="$size"
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
echo "$amount"
|
|
||||||
}
|
|
||||||
|
|
||||||
# 如果禁用调试模式,将输出信息重定向至 /dev/null
|
|
||||||
# 参数:
|
|
||||||
# $@ - 待执行的命令
|
|
||||||
debug_execute() {
|
|
||||||
local -r bool="${ENV_DEBUG:-false}"
|
|
||||||
shopt -s nocasematch
|
|
||||||
if [[ "$bool" = 1 || "$bool" =~ ^(yes|true)$ ]]; then
|
|
||||||
"$@"
|
|
||||||
else
|
|
||||||
"$@" >/dev/null 2>&1
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
# 重试执行命令
|
|
||||||
# 参数:
|
|
||||||
# $1 - cmd (as a string)
|
|
||||||
# $2 - 最大尝试次数. Default: 12
|
|
||||||
# $3 - 重试前等待时间(秒). Default: 5
|
|
||||||
retry_while() {
|
|
||||||
local -r cmd="${1:?cmd is missing}"
|
|
||||||
local -r retries="${2:-12}"
|
|
||||||
local -r sleep_time="${3:-5}"
|
|
||||||
local return_value=1
|
|
||||||
|
|
||||||
read -r -a command <<< "$cmd"
|
|
||||||
for ((i = 1 ; i <= retries ; i+=1 )); do
|
|
||||||
"${command[@]}" && return_value=0 && break
|
|
||||||
sleep "$sleep_time"
|
|
||||||
done
|
|
||||||
return $return_value
|
|
||||||
}
|
|
||||||
@@ -1,87 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
# Ver: 1.0 by Endial Fang (endial@126.com)
|
|
||||||
#
|
|
||||||
# 服务管理函数库
|
|
||||||
|
|
||||||
# shellcheck disable=SC1091
|
|
||||||
|
|
||||||
# 加载依赖项
|
|
||||||
. /usr/local/scripts/liblog.sh # 日志输出函数库
|
|
||||||
|
|
||||||
# 函数列表
|
|
||||||
|
|
||||||
# 获取并返回服务 PID
|
|
||||||
# 参数:
|
|
||||||
# $1 - PID 文件
|
|
||||||
get_pid_from_file() {
|
|
||||||
local pid_file="${1:?pid file is missing}"
|
|
||||||
|
|
||||||
if [[ -f "$pid_file" ]]; then
|
|
||||||
if [[ -n "$(< "$pid_file")" ]] && [[ "$(< "$pid_file")" -gt 0 ]]; then
|
|
||||||
echo "$(< "$pid_file")"
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
# 检测 PID 对应的服务是否在运行中
|
|
||||||
# 参数:
|
|
||||||
# $1 - PID
|
|
||||||
is_service_running() {
|
|
||||||
local pid="${1:?pid is missing}"
|
|
||||||
|
|
||||||
kill -0 "$pid" 2>/dev/null
|
|
||||||
}
|
|
||||||
|
|
||||||
# 通过发送信号停止一个指定的服务
|
|
||||||
# 参数:
|
|
||||||
# $1 - PID 文件
|
|
||||||
# $2 - 信号 (可选)
|
|
||||||
stop_service_using_pid() {
|
|
||||||
local pid_file="${1:?pid file is missing}"
|
|
||||||
local signal="${2:-}"
|
|
||||||
local pid
|
|
||||||
|
|
||||||
pid="$(get_pid_from_file "$pid_file")"
|
|
||||||
[[ -z "$pid" ]] || ! is_service_running "$pid" && return
|
|
||||||
|
|
||||||
if [[ -n "$signal" ]]; then
|
|
||||||
kill "-${signal}" "$pid"
|
|
||||||
else
|
|
||||||
kill "$pid"
|
|
||||||
fi
|
|
||||||
|
|
||||||
local counter=10
|
|
||||||
while [[ "$counter" -ne 0 ]] && is_service_running "$pid"; do
|
|
||||||
sleep 1
|
|
||||||
counter=$((counter - 1))
|
|
||||||
done
|
|
||||||
}
|
|
||||||
|
|
||||||
# 生成一个 Logrotate 配置文件
|
|
||||||
# 参数:
|
|
||||||
# $1 - 应用名称
|
|
||||||
# $2 - 日志路径及日志文件名
|
|
||||||
# $3 - 周期
|
|
||||||
# $4 - Rotations 存储的数量
|
|
||||||
# $5 - 其他参数 (可选)
|
|
||||||
generate_logrotate_conf() {
|
|
||||||
local service_name="${1:?service name is missing}"
|
|
||||||
local log_path="${2:?log path is missing}"
|
|
||||||
local period="${3:-weekly}"
|
|
||||||
local rotations="${4:-150}"
|
|
||||||
local extra_options="${5:-}"
|
|
||||||
local logrotate_conf_dir="/etc/logrotate.d"
|
|
||||||
|
|
||||||
mkdir -p "$logrotate_conf_dir"
|
|
||||||
cat >"${logrotate_conf_dir}/${service_name}" <<-'EOF'
|
|
||||||
${log_path} {
|
|
||||||
${period}
|
|
||||||
rotate ${rotations}
|
|
||||||
dateext
|
|
||||||
compress
|
|
||||||
copytruncate
|
|
||||||
missingok
|
|
||||||
${extra_options}
|
|
||||||
}
|
|
||||||
EOF
|
|
||||||
}
|
|
||||||
@@ -1,213 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
# Ver: 1.0 by Endial Fang (endial@126.com)
|
|
||||||
#
|
|
||||||
# 数据有效性校验函数库
|
|
||||||
|
|
||||||
# 加载依赖项
|
|
||||||
. /usr/local/scripts/liblog.sh # 日志输出函数库
|
|
||||||
|
|
||||||
# 函数列表
|
|
||||||
|
|
||||||
# 检测数据是否为整数
|
|
||||||
# 参数:
|
|
||||||
# $1 - 待检测的数据
|
|
||||||
is_int() {
|
|
||||||
local -r int="${1:?missing value}"
|
|
||||||
if [[ "$int" =~ ^-?[0-9]+ ]]; then
|
|
||||||
true
|
|
||||||
else
|
|
||||||
false
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
# 检测数据是否为正整数
|
|
||||||
# 参数:
|
|
||||||
# $1 - 待检测的数据
|
|
||||||
is_positive_int() {
|
|
||||||
local -r int="${1:?missing value}"
|
|
||||||
if is_int "$int" && (( "${int}" >= 0 )); then
|
|
||||||
true
|
|
||||||
else
|
|
||||||
false
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
# 检测数据是否为布尔值 '1' 或字符串 'yes/true'
|
|
||||||
# 参数:
|
|
||||||
# $1 - 待检测的数据
|
|
||||||
is_boolean_yes() {
|
|
||||||
local -r bool="${1:-}"
|
|
||||||
# comparison is performed without regard to the case of alphabetic characters
|
|
||||||
shopt -s nocasematch
|
|
||||||
if [[ "$bool" = 1 || "$bool" =~ ^(yes|true)$ ]]; then
|
|
||||||
true
|
|
||||||
else
|
|
||||||
false
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
# 检测数据是否为字符串 'yes/no'
|
|
||||||
# 参数:
|
|
||||||
# $1 - 待检测的数据
|
|
||||||
is_yes_no_value() {
|
|
||||||
local -r bool="${1:-}"
|
|
||||||
if [[ "$bool" =~ ^(yes|no)$ ]]; then
|
|
||||||
true
|
|
||||||
else
|
|
||||||
false
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
# 检测数据是否为字符串 'true/false'
|
|
||||||
# 参数:
|
|
||||||
# $1 - 待检测的数据
|
|
||||||
is_true_false_value() {
|
|
||||||
local -r bool="${1:-}"
|
|
||||||
if [[ "$bool" =~ ^(true|false)$ ]]; then
|
|
||||||
true
|
|
||||||
else
|
|
||||||
false
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
# 检测提供的参数是否为空字符串或未定义
|
|
||||||
# 参数:
|
|
||||||
# $1 - 待检测的数据
|
|
||||||
is_empty_value() {
|
|
||||||
local -r val="${1:-}"
|
|
||||||
if [[ -z "$val" ]]; then
|
|
||||||
true
|
|
||||||
else
|
|
||||||
false
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
# 检测数据是否为有效的端口号
|
|
||||||
# 参数:
|
|
||||||
# $1 - 待检测的数据
|
|
||||||
# 返回值:
|
|
||||||
# 布尔值 或 错误消息
|
|
||||||
validate_port() {
|
|
||||||
local value
|
|
||||||
local unprivileged=0
|
|
||||||
|
|
||||||
# Parse flags
|
|
||||||
while [[ "$#" -gt 0 ]]; do
|
|
||||||
case "$1" in
|
|
||||||
-unprivileged)
|
|
||||||
unprivileged=1
|
|
||||||
;;
|
|
||||||
--)
|
|
||||||
shift
|
|
||||||
break
|
|
||||||
;;
|
|
||||||
-*)
|
|
||||||
LOG_E "unrecognized flag $1"
|
|
||||||
return 1
|
|
||||||
;;
|
|
||||||
*)
|
|
||||||
break
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
shift
|
|
||||||
done
|
|
||||||
|
|
||||||
if [[ "$#" -gt 1 ]]; then
|
|
||||||
LOG_E "too many arguments provided"
|
|
||||||
return 2
|
|
||||||
elif [[ "$#" -eq 0 ]]; then
|
|
||||||
LOG_E "missing port argument"
|
|
||||||
return 1
|
|
||||||
else
|
|
||||||
value=$1
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ -z "$value" ]]; then
|
|
||||||
LOG_E "the value is empty"
|
|
||||||
return 1
|
|
||||||
else
|
|
||||||
if ! is_int "$value"; then
|
|
||||||
LOG_W "value is not an integer"
|
|
||||||
return 2
|
|
||||||
elif [[ "$value" -lt 0 ]]; then
|
|
||||||
LOG_W "negative value provided"
|
|
||||||
return 2
|
|
||||||
elif [[ "$value" -gt 65535 ]]; then
|
|
||||||
LOG_W "requested port is greater than 65535"
|
|
||||||
return 2
|
|
||||||
elif [[ "$unprivileged" = 1 && "$value" -lt 1024 ]]; then
|
|
||||||
LOG_W "privileged port requested"
|
|
||||||
return 3
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
# 检测数据是否为有效的IPv4地址
|
|
||||||
# 参数:
|
|
||||||
# $1 - 待检测的数据
|
|
||||||
validate_ipv4() {
|
|
||||||
local ip="${1:?ip is missing}"
|
|
||||||
local stat=1
|
|
||||||
|
|
||||||
if [[ $ip =~ ^[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}$ ]]; then
|
|
||||||
read -r -a ip_array <<< "$(tr '.' ' ' <<< "$ip")"
|
|
||||||
[[ ${ip_array[0]} -le 255 && ${ip_array[1]} -le 255 \
|
|
||||||
&& ${ip_array[2]} -le 255 && ${ip_array[3]} -le 255 ]]
|
|
||||||
stat=$?
|
|
||||||
fi
|
|
||||||
return $stat
|
|
||||||
}
|
|
||||||
|
|
||||||
# 校验字符串格式
|
|
||||||
# 参数:
|
|
||||||
# $1 - 待检测的数据
|
|
||||||
validate_string() {
|
|
||||||
local string
|
|
||||||
local min_length=-1
|
|
||||||
local max_length=-1
|
|
||||||
|
|
||||||
# Parse flags
|
|
||||||
while [ "$#" -gt 0 ]; do
|
|
||||||
case "$1" in
|
|
||||||
-min-length)
|
|
||||||
shift
|
|
||||||
min_length=${1:-}
|
|
||||||
;;
|
|
||||||
-max-length)
|
|
||||||
shift
|
|
||||||
max_length=${1:-}
|
|
||||||
;;
|
|
||||||
--)
|
|
||||||
shift
|
|
||||||
break
|
|
||||||
;;
|
|
||||||
-*)
|
|
||||||
LOG_E "unrecognized flag $1"
|
|
||||||
return 1
|
|
||||||
;;
|
|
||||||
*)
|
|
||||||
break
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
shift
|
|
||||||
done
|
|
||||||
|
|
||||||
if [ "$#" -gt 1 ]; then
|
|
||||||
LOG_E "too many arguments provided"
|
|
||||||
return 2
|
|
||||||
elif [ "$#" -eq 0 ]; then
|
|
||||||
LOG_W "missing string"
|
|
||||||
return 1
|
|
||||||
else
|
|
||||||
string=$1
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ "$min_length" -ge 0 ]] && [[ "${#string}" -lt "$min_length" ]]; then
|
|
||||||
LOG_I "string length is less than $min_length"
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
if [[ "$max_length" -ge 0 ]] && [[ "${#string}" -gt "$max_length" ]]; then
|
|
||||||
LOG_I "string length is great than $max_length"
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
@@ -1,164 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
# Ver: 1.1 by Endial Fang (endial@126.com)
|
|
||||||
#
|
|
||||||
# shell 执行参数,分别为 -e(命令执行错误则退出脚本) -u(变量未定义则报错) -x(打印实际待执行的命令行)
|
|
||||||
set -eux
|
|
||||||
. /usr/local/scripts/liblog.sh
|
|
||||||
|
|
||||||
print_usage() {
|
|
||||||
LOG "Usage: download_pkg <COMMAND> <PACKAGE-NAME> \"<URLS>\" [OPTIONS]"
|
|
||||||
LOG ""
|
|
||||||
LOG "Download and install Third-Part packages"
|
|
||||||
LOG ""
|
|
||||||
LOG "Commands:"
|
|
||||||
LOG " download Download a package."
|
|
||||||
LOG " install Download and install a package."
|
|
||||||
LOG " unpack Download and unpack a package."
|
|
||||||
LOG ""
|
|
||||||
LOG "Options:"
|
|
||||||
LOG " -g, --checkpgp Package release bucket."
|
|
||||||
LOG " -s, --checksum SHA256 verification checksum."
|
|
||||||
LOG " -h, --help Show this help message and exit."
|
|
||||||
LOG ""
|
|
||||||
LOG "PACKAGE-NAME: Name with extern name"
|
|
||||||
LOG "URLS: String with URL list"
|
|
||||||
LOG ""
|
|
||||||
LOG "Examples:"
|
|
||||||
LOG " - Unpack package"
|
|
||||||
LOG " \$ download_pkg unpack redis-5.0.8.tar.gz \"http://download.redis.io/releases/\""
|
|
||||||
LOG ""
|
|
||||||
LOG " - Verify and Install package"
|
|
||||||
LOG " \$ download_pkg install redis-5.0.8.tar.gz \"http://download.redis.io/releases/\" --checksum 42cf86a114d2a451b898fcda96acd4d01062a7dbaaad2801d9164a36f898f596"
|
|
||||||
LOG ""
|
|
||||||
}
|
|
||||||
|
|
||||||
check_pgp() {
|
|
||||||
local name_asc=${1:?missing asc file name}
|
|
||||||
local name=${2:?missing file name}
|
|
||||||
local keys="${3:?missing key id}"
|
|
||||||
|
|
||||||
GNUPGHOME="$(mktemp -d)"
|
|
||||||
if which gpg >/dev/null 2>&1; then
|
|
||||||
for key in $keys; do
|
|
||||||
gpg --batch --keyserver ha.pool.sks-keyservers.net --recv-keys "${key}" ||
|
|
||||||
gpg --batch --keyserver pgp.mit.edu --recv-keys "${key}" ||
|
|
||||||
gpg --batch --keyserver keys.gnupg.net --recv-keys "${key}" ||
|
|
||||||
gpg --batch --keyserver keyserver.pgp.com --recv-keys "${key}";
|
|
||||||
done
|
|
||||||
gpg --batch --verify "$name_asc" "$name"
|
|
||||||
command -v gpgconf > /dev/null && gpgconf --kill all
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
# 获取并解析参数
|
|
||||||
ARGS=$(getopt -o g:s:h -l "checkpgp:,checksum:,help" -n "download-pkg" -- "$@")
|
|
||||||
if [ $? -ne 0 ];
|
|
||||||
then
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
eval set -- "$ARGS";
|
|
||||||
while true; do
|
|
||||||
case "$1" in
|
|
||||||
-g|--checkpgp)
|
|
||||||
shift
|
|
||||||
if [ -n "$1" ]; then
|
|
||||||
PACKAGE_KEYS=$1
|
|
||||||
shift
|
|
||||||
fi
|
|
||||||
;;
|
|
||||||
-s|--checksum)
|
|
||||||
shift
|
|
||||||
if [ -n "$1" ]; then
|
|
||||||
PACKAGE_SHA256=$1
|
|
||||||
shift
|
|
||||||
fi
|
|
||||||
;;
|
|
||||||
-h|--help)
|
|
||||||
print_usage
|
|
||||||
exit 0
|
|
||||||
;;
|
|
||||||
--)
|
|
||||||
shift
|
|
||||||
break
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
done
|
|
||||||
|
|
||||||
# 检测输入的命令是否合法
|
|
||||||
case "$1" in
|
|
||||||
download|install|unpack) ;;
|
|
||||||
*)
|
|
||||||
error "Unrecognized command: $1"
|
|
||||||
print_usage
|
|
||||||
exit 1
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
|
|
||||||
# 检测输入参数是否足够,需要至少提供软件包名称 及 下载路径
|
|
||||||
if [ $# -lt 3 ]; then
|
|
||||||
print_usage
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
INSTALL_ROOT=/usr/local
|
|
||||||
CACHE_ROOT=/tmp
|
|
||||||
|
|
||||||
PACKAGE="$2"
|
|
||||||
PACKAGE_URLS=$3
|
|
||||||
|
|
||||||
cd $INSTALL_ROOT
|
|
||||||
|
|
||||||
LOG_I "Downloading $PACKAGE package"
|
|
||||||
for url in $PACKAGE_URLS; do
|
|
||||||
LOG_D "Try $url/$PACKAGE"
|
|
||||||
if wget -O "$CACHE_ROOT/$PACKAGE" "$url/$PACKAGE" && [ -s "$CACHE_ROOT/$PACKAGE" ]; then
|
|
||||||
if [ -n "${PACKAGE_KEYS:-}" ]; then
|
|
||||||
wget -O "$CACHE_ROOT/$PACKAGE.asc" "$url/$PACKAGE.asc" || wget -O "$CACHE_ROOT/$PACKAGE.asc" "$url/$PACKAGE.sign" || :
|
|
||||||
if [ ! -e "$CACHE_ROOT/$PACKAGE.asc" ]; then
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
break
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
|
|
||||||
if [ -n "${PACKAGE_SHA256:-}" ]; then
|
|
||||||
LOG_I "Verifying package integrity"
|
|
||||||
echo "$PACKAGE_SHA256 *$CACHE_ROOT/$PACKAGE" | sha256sum -c -
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ -e "$CACHE_ROOT/$PACKAGE.asc" ]; then
|
|
||||||
LOG_I "Verifying package with PGP"
|
|
||||||
check_pgp "$CACHE_ROOT/$PACKAGE.asc" "$CACHE_ROOT/$PACKAGE" "$PACKAGE_KEYS"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# If the tarball has too many files, it can trigger a bug
|
|
||||||
# in overlayfs when using tar. Install bsdtar in the container image
|
|
||||||
# to workaround it. As the overhead is too big (~40 MB), it is not added by
|
|
||||||
# default. Source: https://github.com/coreos/bugs/issues/1095
|
|
||||||
|
|
||||||
|
|
||||||
# 安装或解压软件
|
|
||||||
case "$1" in
|
|
||||||
download)
|
|
||||||
LOG_I "Download success: $CACHE_ROOT/$PACKAGE"
|
|
||||||
;;
|
|
||||||
install)
|
|
||||||
LOG_I "Installing $PACKAGE"
|
|
||||||
cp $CACHE_ROOT/$PACKAGE /usr/local/bin/
|
|
||||||
;;
|
|
||||||
unpack)
|
|
||||||
if ! tar -taf $CACHE_ROOT/$PACKAGE >/dev/null 2>&1; then
|
|
||||||
LOG_E "Invalid or corrupt '$PACKAGE' package."
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
LOG_I "Unpacking $PACKAGE"
|
|
||||||
if which bsdtar >/dev/null 2>&1; then
|
|
||||||
bsdtar -xf $CACHE_ROOT/$PACKAGE
|
|
||||||
else
|
|
||||||
tar --no-same-owner -xaf $CACHE_ROOT/$PACKAGE
|
|
||||||
fi
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
|
|
||||||
@@ -1,54 +0,0 @@
|
|||||||
#!/bin/sh
|
|
||||||
# Ver: 1.0 by Endial Fang (endial@126.com)
|
|
||||||
#
|
|
||||||
# shell 执行参数,分别为 -e(命令执行错误则退出脚本) -u(变量未定义则报错) -x(打印实际待执行的命令行)
|
|
||||||
set -eux
|
|
||||||
|
|
||||||
print_usage() {
|
|
||||||
echo "Usage: install_pkg <PACKAGE-NAME>"
|
|
||||||
echo ""
|
|
||||||
echo "Download and install packages"
|
|
||||||
echo ""
|
|
||||||
echo "Options:"
|
|
||||||
echo " -h, --help Show this help message and exit."
|
|
||||||
echo ""
|
|
||||||
echo "Examples:"
|
|
||||||
echo " - Unpack package"
|
|
||||||
echo " \$ install_pkg bash curl"
|
|
||||||
echo ""
|
|
||||||
}
|
|
||||||
|
|
||||||
if [ $# -lt 1 ]; then
|
|
||||||
print_usage
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
case "$1" in
|
|
||||||
-h|--help)
|
|
||||||
print_usage
|
|
||||||
exit 0
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
|
|
||||||
retry=0
|
|
||||||
max=2
|
|
||||||
until [ $retry -gt $max ]; do
|
|
||||||
set +e
|
|
||||||
(
|
|
||||||
apk update --no-cache &&
|
|
||||||
apk upgrade --no-cache &&
|
|
||||||
apk add --no-cache "$@"
|
|
||||||
)
|
|
||||||
CODE=$?
|
|
||||||
set -e
|
|
||||||
if [ $CODE -eq 0 ]; then
|
|
||||||
break
|
|
||||||
fi
|
|
||||||
if [ $retry -eq $max ]; then
|
|
||||||
exit $CODE
|
|
||||||
fi
|
|
||||||
echo "APK failed, retrying"
|
|
||||||
retry=$(($retry + 1))
|
|
||||||
done
|
|
||||||
rm -r /var/cache/apk/* /root/.cache /tmp/* || :
|
|
||||||
|
|
||||||
@@ -1,7 +0,0 @@
|
|||||||
#!/bin/sh
|
|
||||||
# Ver: 1.0 by Endial Fang (endial@126.com)
|
|
||||||
#
|
|
||||||
# shell 执行参数,分别为 -e(命令执行错误则退出脚本) -u(变量未定义则报错) -x(打印实际待执行的命令行)
|
|
||||||
set -eux
|
|
||||||
cp /etc/apk/repositories.${1:-default} /etc/apk/repositories
|
|
||||||
|
|
||||||
Reference in New Issue
Block a user