镜像自地址
https://github.com/tuna/tunasync.git
已同步 2025-12-09 07:56:47 +00:00
比较提交
28 次代码提交
| 作者 | SHA1 | 提交日期 | |
|---|---|---|---|
|
|
88b7827e66 | ||
|
|
49b74ae552 | ||
|
|
37255cc827 | ||
|
|
136e01f1cd | ||
|
|
cd73602988 | ||
|
|
2a8fa5636e | ||
|
|
94b9b20626 | ||
|
|
5a9c6b9020 | ||
|
|
75ee481cfa | ||
|
|
2f9e96a75a | ||
|
|
aa36b96828 | ||
|
|
e9ce7fc87a | ||
|
|
3fd71d777b | ||
|
|
984f8a1eb5 | ||
|
|
a4d94cae07 | ||
|
|
8ebace4d9a | ||
|
|
b578237df8 | ||
|
|
9f7f18c2c4 | ||
|
|
fd274cc976 | ||
|
|
b4b81ef7e9 | ||
|
|
c8600d094e | ||
|
|
2ba3a27fa3 | ||
|
|
b34238c097 | ||
|
|
16e458f354 | ||
|
|
16b4df1ec2 | ||
|
|
e3c8cded6c | ||
|
|
3809df6cfb | ||
|
|
600874ae54 |
26
.github/workflows/release.yml
vendored
26
.github/workflows/release.yml
vendored
@@ -21,16 +21,12 @@ jobs:
|
|||||||
- name: Check out code into the Go module directory
|
- name: Check out code into the Go module directory
|
||||||
uses: actions/checkout@v2
|
uses: actions/checkout@v2
|
||||||
|
|
||||||
- name: Get dependencies
|
|
||||||
run: |
|
|
||||||
go get -v -t -d ./cmd/tunasync
|
|
||||||
go get -v -t -d ./cmd/tunasynctl
|
|
||||||
|
|
||||||
- name: Build
|
- name: Build
|
||||||
run: |
|
run: |
|
||||||
make tunasync
|
for i in linux-amd64 linux-arm64; do
|
||||||
make tunasynctl
|
make ARCH=$i all
|
||||||
tar -jcf build/tunasync-linux-bin.tar.bz2 -C build tunasync tunasynctl
|
tar -cz --numeric-owner --owner root --group root -f tunasync-$i-bin.tar.gz -C build-$i tunasync tunasynctl
|
||||||
|
done
|
||||||
|
|
||||||
- name: Create Release
|
- name: Create Release
|
||||||
id: create_release
|
id: create_release
|
||||||
@@ -42,13 +38,9 @@ jobs:
|
|||||||
release_name: Release ${{ github.ref }}
|
release_name: Release ${{ github.ref }}
|
||||||
draft: false
|
draft: false
|
||||||
prerelease: false
|
prerelease: false
|
||||||
- name: Upload Release Asset
|
- name: Upload Release Assets
|
||||||
id: upload-release-asset
|
|
||||||
uses: actions/upload-release-asset@v1
|
|
||||||
env:
|
env:
|
||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
with:
|
TAG_NAME: ${{ github.ref }}
|
||||||
upload_url: ${{ steps.create_release.outputs.upload_url }} # This pulls from the CREATE RELEASE step above, referencing it's ID to get its outputs object, which include a `upload_url`. See this blog post for more info: https://jasonet.co/posts/new-features-of-github-actions/#passing-data-to-future-steps
|
run: |
|
||||||
asset_path: ./build/tunasync-linux-bin.tar.bz2
|
hub release edit $(find . -type f -name "tunasync-*.tar.gz" -printf "-a %p ") -m "" "${TAG_NAME##*/}"
|
||||||
asset_name: tunasync-linux-bin.tar.bz2
|
|
||||||
asset_content_type: application/x-bzip2
|
|
||||||
|
|||||||
2
.github/workflows/tunasync.yml
vendored
2
.github/workflows/tunasync.yml
vendored
@@ -32,7 +32,7 @@ jobs:
|
|||||||
uses: actions/upload-artifact@v1
|
uses: actions/upload-artifact@v1
|
||||||
with:
|
with:
|
||||||
name: tunasync-bin
|
name: tunasync-bin
|
||||||
path: build/
|
path: build-linux-amd64/
|
||||||
|
|
||||||
test:
|
test:
|
||||||
name: Test
|
name: Test
|
||||||
|
|||||||
1
.gitignore
vendored
1
.gitignore
vendored
@@ -1 +1,2 @@
|
|||||||
/build
|
/build
|
||||||
|
/build-*
|
||||||
|
|||||||
25
Makefile
25
Makefile
@@ -1,19 +1,22 @@
|
|||||||
LDFLAGS="-X main.buildstamp=`date -u '+%s'` -X main.githash=`git rev-parse HEAD`"
|
LDFLAGS="-X main.buildstamp=`date -u '+%s'` -X main.githash=`git rev-parse HEAD`"
|
||||||
|
ARCH ?= linux-amd64
|
||||||
|
ARCH_LIST = $(subst -, ,$(ARCH))
|
||||||
|
GOOS = $(word 1, $(ARCH_LIST))
|
||||||
|
GOARCH = $(word 2, $(ARCH_LIST))
|
||||||
|
BUILDBIN = tunasync tunasynctl
|
||||||
|
|
||||||
all: get tunasync tunasynctl
|
all: $(BUILDBIN)
|
||||||
|
|
||||||
get:
|
build-$(ARCH):
|
||||||
go get ./cmd/tunasync
|
mkdir -p $@
|
||||||
go get ./cmd/tunasynctl
|
|
||||||
|
|
||||||
build:
|
$(BUILDBIN): % : build-$(ARCH) build-$(ARCH)/%
|
||||||
mkdir -p build
|
|
||||||
|
|
||||||
tunasync: build
|
$(BUILDBIN:%=build-$(ARCH)/%) : build-$(ARCH)/% : cmd/%
|
||||||
go build -o build/tunasync -ldflags ${LDFLAGS} github.com/tuna/tunasync/cmd/tunasync
|
GOOS=$(GOOS) GOARCH=$(GOARCH) go get ./$<
|
||||||
|
GOOS=$(GOOS) GOARCH=$(GOARCH) go build -o $@ -ldflags ${LDFLAGS} github.com/tuna/tunasync/$<
|
||||||
tunasynctl: build
|
|
||||||
go build -o build/tunasynctl -ldflags ${LDFLAGS} github.com/tuna/tunasync/cmd/tunasynctl
|
|
||||||
|
|
||||||
test:
|
test:
|
||||||
go test -v -covermode=count -coverprofile=profile.cov ./...
|
go test -v -covermode=count -coverprofile=profile.cov ./...
|
||||||
|
|
||||||
|
.PHONY: all test $(BUILDBIN)
|
||||||
|
|||||||
14
README.md
14
README.md
@@ -1,5 +1,4 @@
|
|||||||
tunasync
|
# tunasync
|
||||||
========
|
|
||||||
|
|
||||||

|

|
||||||
[](https://coveralls.io/github/tuna/tunasync?branch=master)
|
[](https://coveralls.io/github/tuna/tunasync?branch=master)
|
||||||
@@ -12,11 +11,11 @@ tunasync
|
|||||||
|
|
||||||
## Download
|
## Download
|
||||||
|
|
||||||
Pre-built binary for Linux x86_64 is available at [Github releases](https://github.com/tuna/tunasync/releases/latest).
|
Pre-built binary for Linux x86_64 and ARM64 is available at [Github releases](https://github.com/tuna/tunasync/releases/latest).
|
||||||
|
|
||||||
## Design
|
## Design
|
||||||
|
|
||||||
```
|
```text
|
||||||
# Architecture
|
# Architecture
|
||||||
|
|
||||||
- Manager: Central instance for status and job management
|
- Manager: Central instance for status and job management
|
||||||
@@ -50,13 +49,12 @@ PreSyncing Syncing Succe
|
|||||||
+-----------------+
|
+-----------------+
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|
||||||
## Building
|
## Building
|
||||||
|
|
||||||
Go version: 1.13
|
Go version: 1.13
|
||||||
|
|
||||||
```
|
```shell
|
||||||
make all
|
> make all
|
||||||
```
|
```
|
||||||
|
|
||||||
Binaries in the `build/`.
|
Binaries in the `build-linux-amd64/`.
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ import (
|
|||||||
"os"
|
"os"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
|
"text/template"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/BurntSushi/toml"
|
"github.com/BurntSushi/toml"
|
||||||
@@ -160,8 +161,31 @@ func listJobs(c *cli.Context) error {
|
|||||||
"of all jobs from manager server: %s", err.Error()),
|
"of all jobs from manager server: %s", err.Error()),
|
||||||
1)
|
1)
|
||||||
}
|
}
|
||||||
genericJobs = jobs
|
if statusStr := c.String("status"); statusStr != "" {
|
||||||
|
filteredJobs := make([]tunasync.WebMirrorStatus, 0, len(jobs))
|
||||||
|
var statuses []tunasync.SyncStatus
|
||||||
|
for _, s := range strings.Split(statusStr, ",") {
|
||||||
|
var status tunasync.SyncStatus
|
||||||
|
err = status.UnmarshalJSON([]byte("\"" + strings.TrimSpace(s) + "\""))
|
||||||
|
if err != nil {
|
||||||
|
return cli.NewExitError(
|
||||||
|
fmt.Sprintf("Error parsing status: %s", err.Error()),
|
||||||
|
1)
|
||||||
|
}
|
||||||
|
statuses = append(statuses, status)
|
||||||
|
}
|
||||||
|
for _, job := range jobs {
|
||||||
|
for _, s := range statuses {
|
||||||
|
if job.Status == s {
|
||||||
|
filteredJobs = append(filteredJobs, job)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
genericJobs = filteredJobs
|
||||||
|
} else {
|
||||||
|
genericJobs = jobs
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
var jobs []tunasync.MirrorStatus
|
var jobs []tunasync.MirrorStatus
|
||||||
args := c.Args()
|
args := c.Args()
|
||||||
@@ -196,13 +220,46 @@ func listJobs(c *cli.Context) error {
|
|||||||
genericJobs = jobs
|
genericJobs = jobs
|
||||||
}
|
}
|
||||||
|
|
||||||
b, err := json.MarshalIndent(genericJobs, "", " ")
|
if format := c.String("format"); format != "" {
|
||||||
if err != nil {
|
tpl := template.New("")
|
||||||
return cli.NewExitError(
|
_, err := tpl.Parse(format)
|
||||||
fmt.Sprintf("Error printing out information: %s", err.Error()),
|
if err != nil {
|
||||||
1)
|
return cli.NewExitError(
|
||||||
|
fmt.Sprintf("Error parsing format template: %s", err.Error()),
|
||||||
|
1)
|
||||||
|
}
|
||||||
|
switch jobs := genericJobs.(type) {
|
||||||
|
case []tunasync.WebMirrorStatus:
|
||||||
|
for _, job := range jobs {
|
||||||
|
err = tpl.Execute(os.Stdout, job)
|
||||||
|
if err != nil {
|
||||||
|
return cli.NewExitError(
|
||||||
|
fmt.Sprintf("Error printing out information: %s", err.Error()),
|
||||||
|
1)
|
||||||
|
}
|
||||||
|
fmt.Println()
|
||||||
|
}
|
||||||
|
case []tunasync.MirrorStatus:
|
||||||
|
for _, job := range jobs {
|
||||||
|
err = tpl.Execute(os.Stdout, job)
|
||||||
|
if err != nil {
|
||||||
|
return cli.NewExitError(
|
||||||
|
fmt.Sprintf("Error printing out information: %s", err.Error()),
|
||||||
|
1)
|
||||||
|
}
|
||||||
|
fmt.Println()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
b, err := json.MarshalIndent(genericJobs, "", " ")
|
||||||
|
if err != nil {
|
||||||
|
return cli.NewExitError(
|
||||||
|
fmt.Sprintf("Error printing out information: %s", err.Error()),
|
||||||
|
1)
|
||||||
|
}
|
||||||
|
fmt.Println(string(b))
|
||||||
}
|
}
|
||||||
fmt.Println(string(b))
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -506,6 +563,14 @@ func main() {
|
|||||||
Name: "all, a",
|
Name: "all, a",
|
||||||
Usage: "List all jobs of all workers",
|
Usage: "List all jobs of all workers",
|
||||||
},
|
},
|
||||||
|
cli.StringFlag{
|
||||||
|
Name: "status, s",
|
||||||
|
Usage: "Filter output based on status provided",
|
||||||
|
},
|
||||||
|
cli.StringFlag{
|
||||||
|
Name: "format, f",
|
||||||
|
Usage: "Pretty-print containers using a Go template",
|
||||||
|
},
|
||||||
}...),
|
}...),
|
||||||
Action: initializeWrapper(listJobs),
|
Action: initializeWrapper(listJobs),
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
# tunasync 上手指南
|
# tunasync 上手指南
|
||||||
|
|
||||||
date: 2016-10-31 00:50:00
|
date: 2016-10-31 00:50:00
|
||||||
|
|
||||||
[tunasync](https://github.com/tuna/tunasync) 是[清华大学 TUNA 镜像源](https://mirrors.tuna.tsinghua.edu.cn)目前使用的镜像方案。
|
[tunasync](https://github.com/tuna/tunasync) 是[清华大学 TUNA 镜像源](https://mirrors.tuna.tsinghua.edu.cn)目前使用的镜像方案。
|
||||||
@@ -7,32 +8,32 @@ date: 2016-10-31 00:50:00
|
|||||||
|
|
||||||
本例中:
|
本例中:
|
||||||
|
|
||||||
- 只镜像[elvish](https://elvish.io)项目
|
- 只镜像[elvish](https://elvish.io)项目
|
||||||
- 禁用了https
|
- 禁用了https
|
||||||
- 禁用了cgroup支持
|
- 禁用了cgroup支持
|
||||||
|
|
||||||
## 获得tunasync
|
## 获得tunasync
|
||||||
|
|
||||||
### 二进制包
|
### 二进制包
|
||||||
|
|
||||||
到 [Github Releases](https://github.com/tuna/tunasync/releases/latest) 下载 `tunasync-linux-bin.tar.gz` 即可。
|
到 [Github Releases](https://github.com/tuna/tunasync/releases/latest) 下载 `tunasync-linux-amd64-bin.tar.gz` 即可。
|
||||||
|
|
||||||
### 自行编译
|
### 自行编译
|
||||||
|
|
||||||
```
|
```shell
|
||||||
$ make
|
> make
|
||||||
```
|
```
|
||||||
|
|
||||||
## 配置
|
## 配置
|
||||||
|
|
||||||
```
|
```shell
|
||||||
$ mkdir ~/tunasync_demo
|
> mkdir ~/tunasync_demo
|
||||||
$ mkdir /tmp/tunasync
|
> mkdir /tmp/tunasync
|
||||||
```
|
```
|
||||||
|
|
||||||
`~/tunasync_demo/worker.conf`:
|
编辑 `~/tunasync_demo/worker.conf`:
|
||||||
|
|
||||||
```
|
```conf
|
||||||
[global]
|
[global]
|
||||||
name = "test_worker"
|
name = "test_worker"
|
||||||
log_dir = "/tmp/tunasync/log/tunasync/{{.Name}}"
|
log_dir = "/tmp/tunasync/log/tunasync/{{.Name}}"
|
||||||
@@ -64,9 +65,9 @@ upstream = "rsync://rsync.elvish.io/elvish/"
|
|||||||
use_ipv6 = false
|
use_ipv6 = false
|
||||||
```
|
```
|
||||||
|
|
||||||
`~/tunasync_demo/manager.conf`:
|
编辑 `~/tunasync_demo/manager.conf`:
|
||||||
|
|
||||||
```
|
```conf
|
||||||
debug = false
|
debug = false
|
||||||
|
|
||||||
[server]
|
[server]
|
||||||
@@ -83,26 +84,26 @@ ca_cert = ""
|
|||||||
|
|
||||||
### 运行
|
### 运行
|
||||||
|
|
||||||
```
|
```shell
|
||||||
$ tunasync manager --config ~/tunasync_demo/manager.conf
|
> tunasync manager --config ~/tunasync_demo/manager.conf
|
||||||
$ tunasync worker --config ~/tunasync_demo/worker.conf
|
> tunasync worker --config ~/tunasync_demo/worker.conf
|
||||||
```
|
```
|
||||||
|
|
||||||
本例中,镜像的数据在`/tmp/tunasync/`
|
本例中,镜像的数据在 `/tmp/tunasync/`。
|
||||||
|
|
||||||
### 控制
|
### 控制
|
||||||
|
|
||||||
查看同步状态
|
查看同步状态
|
||||||
|
|
||||||
```
|
```shell
|
||||||
$ tunasynctl list -p 12345 --all
|
> tunasynctl list -p 12345 --all
|
||||||
```
|
```
|
||||||
|
|
||||||
tunasynctl 也支持配置文件。配置文件可以放在 `/etc/tunasync/ctl.conf` 或者 `~/.config/tunasync/ctl.conf` 两个位置,后者可以覆盖前者的配置值。
|
tunasynctl 也支持配置文件。配置文件可以放在 `/etc/tunasync/ctl.conf` 或者 `~/.config/tunasync/ctl.conf` 两个位置,后者可以覆盖前者的配置值。
|
||||||
|
|
||||||
配置文件内容为:
|
配置文件内容为:
|
||||||
|
|
||||||
```
|
```conf
|
||||||
manager_addr = "127.0.0.1"
|
manager_addr = "127.0.0.1"
|
||||||
manager_port = 12345
|
manager_port = 12345
|
||||||
ca_cert = ""
|
ca_cert = ""
|
||||||
@@ -118,13 +119,13 @@ worker 和 manager 之间用 http(s) 通信,如果你 worker 和 manager 都
|
|||||||
|
|
||||||
可以参看
|
可以参看
|
||||||
|
|
||||||
```
|
```shell
|
||||||
$ tunasync manager --help
|
> tunasync manager --help
|
||||||
$ tunasync worker --help
|
> tunasync worker --help
|
||||||
```
|
```
|
||||||
|
|
||||||
可以看一下 log 目录
|
可以看一下 log 目录
|
||||||
|
|
||||||
一些 worker 配置文件示例 [workers.conf](workers.conf)
|
一些 worker 配置文件示例 [workers.conf](workers.conf)。
|
||||||
|
|
||||||
你可能会用到的操作 [tips.md](tips.md)
|
你可能会用到的操作 [tips.md](tips.md)。
|
||||||
|
|||||||
@@ -7,6 +7,11 @@ mirror_dir = "/srv/tunasync"
|
|||||||
concurrent = 10
|
concurrent = 10
|
||||||
interval = 1
|
interval = 1
|
||||||
|
|
||||||
|
# ensure the exec user be add into `docker` group
|
||||||
|
[docker]
|
||||||
|
# in `command provider` can use docker_image and docker_volumes
|
||||||
|
enable = true
|
||||||
|
|
||||||
[manager]
|
[manager]
|
||||||
api_base = "http://localhost:12345"
|
api_base = "http://localhost:12345"
|
||||||
token = "some_token"
|
token = "some_token"
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
package internal
|
package internal
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
@@ -24,10 +26,11 @@ type MirrorStatus struct {
|
|||||||
// A WorkerStatus is the information struct that describe
|
// A WorkerStatus is the information struct that describe
|
||||||
// a worker, and sent from the manager to clients.
|
// a worker, and sent from the manager to clients.
|
||||||
type WorkerStatus struct {
|
type WorkerStatus struct {
|
||||||
ID string `json:"id"`
|
ID string `json:"id"`
|
||||||
URL string `json:"url"` // worker url
|
URL string `json:"url"` // worker url
|
||||||
Token string `json:"token"` // session token
|
Token string `json:"token"` // session token
|
||||||
LastOnline time.Time `json:"last_online"` // last seen
|
LastOnline time.Time `json:"last_online"` // last seen
|
||||||
|
LastRegister time.Time `json:"last_register"` // last register time
|
||||||
}
|
}
|
||||||
|
|
||||||
type MirrorSchedules struct {
|
type MirrorSchedules struct {
|
||||||
@@ -59,21 +62,45 @@ const (
|
|||||||
)
|
)
|
||||||
|
|
||||||
func (c CmdVerb) String() string {
|
func (c CmdVerb) String() string {
|
||||||
switch c {
|
mapping := map[CmdVerb]string{
|
||||||
case CmdStart:
|
CmdStart: "start",
|
||||||
return "start"
|
CmdStop: "stop",
|
||||||
case CmdStop:
|
CmdDisable: "disable",
|
||||||
return "stop"
|
CmdRestart: "restart",
|
||||||
case CmdDisable:
|
CmdPing: "ping",
|
||||||
return "disable"
|
CmdReload: "reload",
|
||||||
case CmdRestart:
|
|
||||||
return "restart"
|
|
||||||
case CmdPing:
|
|
||||||
return "ping"
|
|
||||||
case CmdReload:
|
|
||||||
return "reload"
|
|
||||||
}
|
}
|
||||||
return "unknown"
|
return mapping[c]
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewCmdVerbFromString(s string) CmdVerb {
|
||||||
|
mapping := map[string]CmdVerb{
|
||||||
|
"start": CmdStart,
|
||||||
|
"stop": CmdStop,
|
||||||
|
"disable": CmdDisable,
|
||||||
|
"restart": CmdRestart,
|
||||||
|
"ping": CmdPing,
|
||||||
|
"reload": CmdReload,
|
||||||
|
}
|
||||||
|
return mapping[s]
|
||||||
|
}
|
||||||
|
|
||||||
|
// Marshal and Unmarshal for CmdVerb
|
||||||
|
func (s CmdVerb) MarshalJSON() ([]byte, error) {
|
||||||
|
buffer := bytes.NewBufferString(`"`)
|
||||||
|
buffer.WriteString(s.String())
|
||||||
|
buffer.WriteString(`"`)
|
||||||
|
return buffer.Bytes(), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *CmdVerb) UnmarshalJSON(b []byte) error {
|
||||||
|
var j string
|
||||||
|
err := json.Unmarshal(b, &j)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
*s = NewCmdVerbFromString(j)
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// A WorkerCmd is the command message send from the
|
// A WorkerCmd is the command message send from the
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
package internal
|
package internal
|
||||||
|
|
||||||
// Version of the program
|
// Version of the program
|
||||||
const Version string = "0.6.6"
|
const Version string = "0.6.9"
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ import (
|
|||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"strings"
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/boltdb/bolt"
|
"github.com/boltdb/bolt"
|
||||||
|
|
||||||
@@ -16,6 +17,7 @@ type dbAdapter interface {
|
|||||||
GetWorker(workerID string) (WorkerStatus, error)
|
GetWorker(workerID string) (WorkerStatus, error)
|
||||||
DeleteWorker(workerID string) error
|
DeleteWorker(workerID string) error
|
||||||
CreateWorker(w WorkerStatus) (WorkerStatus, error)
|
CreateWorker(w WorkerStatus) (WorkerStatus, error)
|
||||||
|
RefreshWorker(workerID string) (WorkerStatus, error)
|
||||||
UpdateMirrorStatus(workerID, mirrorID string, status MirrorStatus) (MirrorStatus, error)
|
UpdateMirrorStatus(workerID, mirrorID string, status MirrorStatus) (MirrorStatus, error)
|
||||||
GetMirrorStatus(workerID, mirrorID string) (MirrorStatus, error)
|
GetMirrorStatus(workerID, mirrorID string) (MirrorStatus, error)
|
||||||
ListMirrorStatus(workerID string) ([]MirrorStatus, error)
|
ListMirrorStatus(workerID string) ([]MirrorStatus, error)
|
||||||
@@ -26,7 +28,9 @@ type dbAdapter interface {
|
|||||||
|
|
||||||
func makeDBAdapter(dbType string, dbFile string) (dbAdapter, error) {
|
func makeDBAdapter(dbType string, dbFile string) (dbAdapter, error) {
|
||||||
if dbType == "bolt" {
|
if dbType == "bolt" {
|
||||||
innerDB, err := bolt.Open(dbFile, 0600, nil)
|
innerDB, err := bolt.Open(dbFile, 0600, &bolt.Options{
|
||||||
|
Timeout: 5 * time.Second,
|
||||||
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@@ -122,6 +126,15 @@ func (b *boltAdapter) CreateWorker(w WorkerStatus) (WorkerStatus, error) {
|
|||||||
return w, err
|
return w, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (b *boltAdapter) RefreshWorker(workerID string) (w WorkerStatus, err error) {
|
||||||
|
w, err = b.GetWorker(workerID)
|
||||||
|
if err == nil {
|
||||||
|
w.LastOnline = time.Now()
|
||||||
|
w, err = b.CreateWorker(w)
|
||||||
|
}
|
||||||
|
return w, err
|
||||||
|
}
|
||||||
|
|
||||||
func (b *boltAdapter) UpdateMirrorStatus(workerID, mirrorID string, status MirrorStatus) (MirrorStatus, error) {
|
func (b *boltAdapter) UpdateMirrorStatus(workerID, mirrorID string, status MirrorStatus) (MirrorStatus, error) {
|
||||||
id := mirrorID + "/" + workerID
|
id := mirrorID + "/" + workerID
|
||||||
err := b.db.Update(func(tx *bolt.Tx) error {
|
err := b.db.Update(func(tx *bolt.Tx) error {
|
||||||
|
|||||||
@@ -35,6 +35,7 @@ func TestBoltAdapter(t *testing.T) {
|
|||||||
ID: id,
|
ID: id,
|
||||||
Token: "token_" + id,
|
Token: "token_" + id,
|
||||||
LastOnline: time.Now(),
|
LastOnline: time.Now(),
|
||||||
|
LastRegister: time.Now(),
|
||||||
}
|
}
|
||||||
w, err = boltDB.CreateWorker(w)
|
w, err = boltDB.CreateWorker(w)
|
||||||
So(err, ShouldBeNil)
|
So(err, ShouldBeNil)
|
||||||
|
|||||||
@@ -203,8 +203,11 @@ func (s *Manager) listWorkers(c *gin.Context) {
|
|||||||
for _, w := range workers {
|
for _, w := range workers {
|
||||||
workerInfos = append(workerInfos,
|
workerInfos = append(workerInfos,
|
||||||
WorkerStatus{
|
WorkerStatus{
|
||||||
ID: w.ID,
|
ID: w.ID,
|
||||||
LastOnline: w.LastOnline,
|
URL: w.URL,
|
||||||
|
Token: "REDACTED",
|
||||||
|
LastOnline: w.LastOnline,
|
||||||
|
LastRegister: w.LastRegister,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
c.JSON(http.StatusOK, workerInfos)
|
c.JSON(http.StatusOK, workerInfos)
|
||||||
@@ -215,6 +218,7 @@ func (s *Manager) registerWorker(c *gin.Context) {
|
|||||||
var _worker WorkerStatus
|
var _worker WorkerStatus
|
||||||
c.BindJSON(&_worker)
|
c.BindJSON(&_worker)
|
||||||
_worker.LastOnline = time.Now()
|
_worker.LastOnline = time.Now()
|
||||||
|
_worker.LastRegister = time.Now()
|
||||||
newWorker, err := s.adapter.CreateWorker(_worker)
|
newWorker, err := s.adapter.CreateWorker(_worker)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
err := fmt.Errorf("failed to register worker: %s",
|
err := fmt.Errorf("failed to register worker: %s",
|
||||||
@@ -268,6 +272,7 @@ func (s *Manager) updateSchedulesOfWorker(c *gin.Context) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
s.rwmu.RLock()
|
s.rwmu.RLock()
|
||||||
|
s.adapter.RefreshWorker(workerID)
|
||||||
curStatus, err := s.adapter.GetMirrorStatus(workerID, mirrorName)
|
curStatus, err := s.adapter.GetMirrorStatus(workerID, mirrorName)
|
||||||
s.rwmu.RUnlock()
|
s.rwmu.RUnlock()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -312,6 +317,7 @@ func (s *Manager) updateJobOfWorker(c *gin.Context) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
s.rwmu.RLock()
|
s.rwmu.RLock()
|
||||||
|
s.adapter.RefreshWorker(workerID)
|
||||||
curStatus, _ := s.adapter.GetMirrorStatus(workerID, mirrorName)
|
curStatus, _ := s.adapter.GetMirrorStatus(workerID, mirrorName)
|
||||||
s.rwmu.RUnlock()
|
s.rwmu.RUnlock()
|
||||||
|
|
||||||
@@ -374,6 +380,7 @@ func (s *Manager) updateMirrorSize(c *gin.Context) {
|
|||||||
|
|
||||||
mirrorName := msg.Name
|
mirrorName := msg.Name
|
||||||
s.rwmu.RLock()
|
s.rwmu.RLock()
|
||||||
|
s.adapter.RefreshWorker(workerID)
|
||||||
status, err := s.adapter.GetMirrorStatus(workerID, mirrorName)
|
status, err := s.adapter.GetMirrorStatus(workerID, mirrorName)
|
||||||
s.rwmu.RUnlock()
|
s.rwmu.RUnlock()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|||||||
@@ -462,6 +462,15 @@ func (b *mockDBAdapter) CreateWorker(w WorkerStatus) (WorkerStatus, error) {
|
|||||||
return w, nil
|
return w, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (b *mockDBAdapter) RefreshWorker(workerID string) (w WorkerStatus, err error) {
|
||||||
|
w, err = b.GetWorker(workerID)
|
||||||
|
if err == nil {
|
||||||
|
w.LastOnline = time.Now()
|
||||||
|
w, err = b.CreateWorker(w)
|
||||||
|
}
|
||||||
|
return w, err
|
||||||
|
}
|
||||||
|
|
||||||
func (b *mockDBAdapter) GetMirrorStatus(workerID, mirrorID string) (MirrorStatus, error) {
|
func (b *mockDBAdapter) GetMirrorStatus(workerID, mirrorID string) (MirrorStatus, error) {
|
||||||
id := mirrorID + "/" + workerID
|
id := mirrorID + "/" + workerID
|
||||||
status, ok := b.statusStore[id]
|
status, ok := b.statusStore[id]
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
// +build linux
|
||||||
|
|
||||||
package worker
|
package worker
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
|||||||
@@ -0,0 +1,30 @@
|
|||||||
|
// +build !linux
|
||||||
|
|
||||||
|
package worker
|
||||||
|
|
||||||
|
type btrfsSnapshotHook struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
func newBtrfsSnapshotHook(provider mirrorProvider, snapshotPath string, mirror mirrorConfig) *btrfsSnapshotHook {
|
||||||
|
return &btrfsSnapshotHook{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *btrfsSnapshotHook) postExec() error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *btrfsSnapshotHook) postFail() error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *btrfsSnapshotHook) postSuccess() error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *btrfsSnapshotHook) preExec() error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *btrfsSnapshotHook) preJob() error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
@@ -91,7 +91,7 @@ exit 0
|
|||||||
"Done\n",
|
"Done\n",
|
||||||
targetDir,
|
targetDir,
|
||||||
fmt.Sprintf(
|
fmt.Sprintf(
|
||||||
"-aHvh --no-o --no-g --stats --exclude .~tmp~/ "+
|
"-aHvh --no-o --no-g --stats --filter risk .~tmp~/ --exclude .~tmp~/ "+
|
||||||
"--delete --delete-after --delay-updates --safe-links "+
|
"--delete --delete-after --delay-updates --safe-links "+
|
||||||
"--timeout=120 -6 %s %s",
|
"--timeout=120 -6 %s %s",
|
||||||
provider.upstreamURL, provider.WorkingDir(),
|
provider.upstreamURL, provider.WorkingDir(),
|
||||||
@@ -190,7 +190,7 @@ exit 0
|
|||||||
"Done\n",
|
"Done\n",
|
||||||
targetDir,
|
targetDir,
|
||||||
fmt.Sprintf(
|
fmt.Sprintf(
|
||||||
"%s %s %s -aHvh --no-o --no-g --stats --exclude .~tmp~/ "+
|
"%s %s %s -aHvh --no-o --no-g --stats --filter risk .~tmp~/ --exclude .~tmp~/ "+
|
||||||
"--delete --delete-after --delay-updates --safe-links "+
|
"--delete --delete-after --delay-updates --safe-links "+
|
||||||
"--timeout=30 -4 --delete-excluded %s %s",
|
"--timeout=30 -4 --delete-excluded %s %s",
|
||||||
provider.username, provider.password, proxyAddr,
|
provider.username, provider.password, proxyAddr,
|
||||||
@@ -613,14 +613,14 @@ exit 0
|
|||||||
"Done\n",
|
"Done\n",
|
||||||
targetDir,
|
targetDir,
|
||||||
fmt.Sprintf(
|
fmt.Sprintf(
|
||||||
"-aHvh --no-o --no-g --stats --exclude .~tmp~/ --safe-links "+
|
"-aHvh --no-o --no-g --stats --filter risk .~tmp~/ --exclude .~tmp~/ --safe-links "+
|
||||||
"--exclude dists/ --timeout=30 -6 "+
|
"--include=*.diff/ --exclude=*.diff/Index --exclude=Packages* --exclude=Sources* --exclude=Release* --exclude=InRelease --include=i18n/by-hash --exclude=i18n/* --exclude=ls-lR* --timeout=30 -6 "+
|
||||||
"--exclude-from %s %s %s",
|
"--exclude-from %s %s %s",
|
||||||
provider.excludeFile, provider.upstreamURL, provider.WorkingDir(),
|
provider.excludeFile, provider.upstreamURL, provider.WorkingDir(),
|
||||||
),
|
),
|
||||||
targetDir,
|
targetDir,
|
||||||
fmt.Sprintf(
|
fmt.Sprintf(
|
||||||
"-aHvh --no-o --no-g --stats --exclude .~tmp~/ "+
|
"-aHvh --no-o --no-g --stats --filter risk .~tmp~/ --exclude .~tmp~/ "+
|
||||||
"--delete --delete-after --delay-updates --safe-links "+
|
"--delete --delete-after --delay-updates --safe-links "+
|
||||||
"--delete-excluded --cache --timeout=30 -6 --exclude-from %s %s %s",
|
"--delete-excluded --cache --timeout=30 -6 --exclude-from %s %s %s",
|
||||||
provider.excludeFile, provider.upstreamURL, provider.WorkingDir(),
|
provider.excludeFile, provider.upstreamURL, provider.WorkingDir(),
|
||||||
@@ -655,8 +655,8 @@ exit 0
|
|||||||
So(err, ShouldBeNil)
|
So(err, ShouldBeNil)
|
||||||
|
|
||||||
expectedOutput := fmt.Sprintf(
|
expectedOutput := fmt.Sprintf(
|
||||||
"-aHvh --no-o --no-g --stats --exclude .~tmp~/ --safe-links "+
|
"-aHvh --no-o --no-g --stats --filter risk .~tmp~/ --exclude .~tmp~/ --safe-links "+
|
||||||
"--exclude dists/ --timeout=30 -6 "+
|
"--include=*.diff/ --exclude=*.diff/Index --exclude=Packages* --exclude=Sources* --exclude=Release* --exclude=InRelease --include=i18n/by-hash --exclude=i18n/* --exclude=ls-lR* --timeout=30 -6 "+
|
||||||
"--exclude-from %s %s %s\n",
|
"--exclude-from %s %s %s\n",
|
||||||
provider.excludeFile, provider.upstreamURL, provider.WorkingDir(),
|
provider.excludeFile, provider.upstreamURL, provider.WorkingDir(),
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -67,7 +67,7 @@ func newRsyncProvider(c rsyncConfig) (*rsyncProvider, error) {
|
|||||||
|
|
||||||
options := []string{
|
options := []string{
|
||||||
"-aHvh", "--no-o", "--no-g", "--stats",
|
"-aHvh", "--no-o", "--no-g", "--stats",
|
||||||
"--exclude", ".~tmp~/",
|
"--filter" , "risk .~tmp~/", "--exclude", ".~tmp~/",
|
||||||
"--delete", "--delete-after", "--delay-updates",
|
"--delete", "--delete-after", "--delay-updates",
|
||||||
"--safe-links",
|
"--safe-links",
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -149,10 +149,10 @@ func (c *cmdJob) Terminate() error {
|
|||||||
select {
|
select {
|
||||||
case <-time.After(2 * time.Second):
|
case <-time.After(2 * time.Second):
|
||||||
unix.Kill(c.cmd.Process.Pid, syscall.SIGKILL)
|
unix.Kill(c.cmd.Process.Pid, syscall.SIGKILL)
|
||||||
return errors.New("SIGTERM failed to kill the job")
|
logger.Warningf("SIGTERM failed to kill the job in 2s. SIGKILL sent")
|
||||||
case <-c.finished:
|
case <-c.finished:
|
||||||
return nil
|
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Copied from go-sh
|
// Copied from go-sh
|
||||||
|
|||||||
@@ -34,11 +34,12 @@ type twoStageRsyncProvider struct {
|
|||||||
dataSize string
|
dataSize string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ref: https://salsa.debian.org/mirror-team/archvsync/-/blob/master/bin/ftpsync#L431
|
||||||
var rsyncStage1Profiles = map[string]([]string){
|
var rsyncStage1Profiles = map[string]([]string){
|
||||||
"debian": []string{"dists/"},
|
"debian": []string{"--include=*.diff/", "--exclude=*.diff/Index", "--exclude=Packages*", "--exclude=Sources*", "--exclude=Release*", "--exclude=InRelease", "--include=i18n/by-hash", "--exclude=i18n/*", "--exclude=ls-lR*"},
|
||||||
"debian-oldstyle": []string{
|
"debian-oldstyle": []string{
|
||||||
"Packages*", "Sources*", "Release*",
|
"--exclude=Packages*", "--exclude=Sources*", "--exclude=Release*",
|
||||||
"InRelease", "i18n/*", "ls-lR*", "dep11/*",
|
"--exclude=InRelease", "--exclude=i18n/*", "--exclude=ls-lR*", "--exclude=dep11/*",
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -62,12 +63,12 @@ func newTwoStageRsyncProvider(c twoStageRsyncConfig) (*twoStageRsyncProvider, er
|
|||||||
twoStageRsyncConfig: c,
|
twoStageRsyncConfig: c,
|
||||||
stage1Options: []string{
|
stage1Options: []string{
|
||||||
"-aHvh", "--no-o", "--no-g", "--stats",
|
"-aHvh", "--no-o", "--no-g", "--stats",
|
||||||
"--exclude", ".~tmp~/",
|
"--filter", "risk .~tmp~/", "--exclude", ".~tmp~/",
|
||||||
"--safe-links",
|
"--safe-links",
|
||||||
},
|
},
|
||||||
stage2Options: []string{
|
stage2Options: []string{
|
||||||
"-aHvh", "--no-o", "--no-g", "--stats",
|
"-aHvh", "--no-o", "--no-g", "--stats",
|
||||||
"--exclude", ".~tmp~/",
|
"--filter", "risk .~tmp~/", "--exclude", ".~tmp~/",
|
||||||
"--delete", "--delete-after", "--delay-updates",
|
"--delete", "--delete-after", "--delay-updates",
|
||||||
"--safe-links",
|
"--safe-links",
|
||||||
},
|
},
|
||||||
@@ -109,12 +110,12 @@ func (p *twoStageRsyncProvider) Options(stage int) ([]string, error) {
|
|||||||
var options []string
|
var options []string
|
||||||
if stage == 1 {
|
if stage == 1 {
|
||||||
options = append(options, p.stage1Options...)
|
options = append(options, p.stage1Options...)
|
||||||
stage1Excludes, ok := rsyncStage1Profiles[p.stage1Profile]
|
stage1Profile, ok := rsyncStage1Profiles[p.stage1Profile]
|
||||||
if !ok {
|
if !ok {
|
||||||
return nil, errors.New("Invalid Stage 1 Profile")
|
return nil, errors.New("Invalid Stage 1 Profile")
|
||||||
}
|
}
|
||||||
for _, exc := range stage1Excludes {
|
for _, exc := range stage1Profile {
|
||||||
options = append(options, "--exclude", exc)
|
options = append(options, exc)
|
||||||
}
|
}
|
||||||
|
|
||||||
} else if stage == 2 {
|
} else if stage == 2 {
|
||||||
|
|||||||
@@ -61,7 +61,7 @@ func NewTUNASyncWorker(cfg *Config) *Worker {
|
|||||||
|
|
||||||
// Run runs worker forever
|
// Run runs worker forever
|
||||||
func (w *Worker) Run() {
|
func (w *Worker) Run() {
|
||||||
w.registorWorker()
|
w.registerWorker()
|
||||||
go w.runHTTPServer()
|
go w.runHTTPServer()
|
||||||
w.runSchedule()
|
w.runSchedule()
|
||||||
}
|
}
|
||||||
@@ -393,7 +393,7 @@ func (w *Worker) URL() string {
|
|||||||
return fmt.Sprintf("%s://%s:%d/", proto, w.cfg.Server.Hostname, w.cfg.Server.Port)
|
return fmt.Sprintf("%s://%s:%d/", proto, w.cfg.Server.Hostname, w.cfg.Server.Port)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (w *Worker) registorWorker() {
|
func (w *Worker) registerWorker() {
|
||||||
msg := WorkerStatus{
|
msg := WorkerStatus{
|
||||||
ID: w.Name(),
|
ID: w.Name(),
|
||||||
URL: w.URL(),
|
URL: w.URL(),
|
||||||
|
|||||||
@@ -25,6 +25,7 @@ func makeMockManagerServer(recvData chan interface{}) *gin.Engine {
|
|||||||
var _worker WorkerStatus
|
var _worker WorkerStatus
|
||||||
c.BindJSON(&_worker)
|
c.BindJSON(&_worker)
|
||||||
_worker.LastOnline = time.Now()
|
_worker.LastOnline = time.Now()
|
||||||
|
_worker.LastRegister = time.Now()
|
||||||
recvData <- _worker
|
recvData <- _worker
|
||||||
c.JSON(http.StatusOK, _worker)
|
c.JSON(http.StatusOK, _worker)
|
||||||
})
|
})
|
||||||
|
|||||||
在新工单中引用
屏蔽一个用户