You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
131 lines
3.8 KiB
131 lines
3.8 KiB
5 years ago
|
package lich
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"crypto/md5"
|
||
|
"encoding/json"
|
||
|
"flag"
|
||
|
"fmt"
|
||
|
"log"
|
||
|
"os"
|
||
|
"os/exec"
|
||
|
"path/filepath"
|
||
|
"time"
|
||
|
)
|
||
|
|
||
|
var (
|
||
|
retry int
|
||
|
yamlPath string
|
||
|
pathHash string
|
||
|
services map[string]*Container
|
||
|
)
|
||
|
|
||
|
func init() {
|
||
|
flag.StringVar(&yamlPath, "f", "docker-compose.yaml", "composer yaml path.")
|
||
|
}
|
||
|
|
||
|
// Setup setup UT related environment dependence for everything.
|
||
|
func Setup() (err error) {
|
||
|
if _, err = os.Stat(yamlPath); os.IsNotExist(err) {
|
||
|
log.Println("composer yaml is not exist!", yamlPath)
|
||
|
return
|
||
|
}
|
||
|
if yamlPath, err = filepath.Abs(yamlPath); err != nil {
|
||
|
log.Printf("filepath.Abs(%s) error(%v)", yamlPath, err)
|
||
|
return
|
||
|
}
|
||
|
pathHash = fmt.Sprintf("%x", md5.Sum([]byte(yamlPath)))[:9]
|
||
|
var args = []string{"-f", yamlPath, "-p", pathHash, "up", "-d"}
|
||
|
if err = exec.Command("docker-compose", args...).Run(); err != nil {
|
||
|
log.Printf("exec.Command(docker-composer) args(%v) error(%v)", args, err)
|
||
|
Teardown()
|
||
|
return
|
||
|
}
|
||
|
// 拿到yaml文件中的服务名,同时通过服务名获取到启动的容器ID
|
||
|
if _, err = getServices(); err != nil {
|
||
|
Teardown()
|
||
|
return
|
||
|
}
|
||
|
// 通过容器ID检测容器的状态,包括容器服务的状态
|
||
|
if _, err = checkServices(); err != nil {
|
||
|
Teardown()
|
||
|
return
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// Teardown unsetup all environment dependence.
|
||
|
func Teardown() (err error) {
|
||
|
if _, err = os.Stat(yamlPath); os.IsNotExist(err) {
|
||
|
log.Println("composer yaml is not exist!")
|
||
|
return
|
||
|
}
|
||
|
if yamlPath, err = filepath.Abs(yamlPath); err != nil {
|
||
|
log.Printf("filepath.Abs(%s) error(%v)", yamlPath, err)
|
||
|
return
|
||
|
}
|
||
|
pathHash = fmt.Sprintf("%x", md5.Sum([]byte(yamlPath)))[:9]
|
||
|
args := []string{"-f", yamlPath, "-p", pathHash, "down"}
|
||
|
if output, err := exec.Command("docker-compose", args...).CombinedOutput(); err != nil {
|
||
|
log.Fatalf("exec.Command(docker-composer) args(%v) stdout(%s) error(%v)", args, string(output), err)
|
||
|
return err
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
func getServices() (output []byte, err error) {
|
||
|
var args = []string{"-f", yamlPath, "-p", pathHash, "config", "--services"}
|
||
|
if output, err = exec.Command("docker-compose", args...).CombinedOutput(); err != nil {
|
||
|
log.Printf("exec.Command(docker-composer) args(%v) stdout(%s) error(%v)", args, string(output), err)
|
||
|
return
|
||
|
}
|
||
|
services = make(map[string]*Container)
|
||
|
output = bytes.TrimSpace(output)
|
||
|
for _, svr := range bytes.Split(output, []byte("\n")) {
|
||
|
args = []string{"-f", yamlPath, "-p", pathHash, "ps", "-a", "-q", string(svr)}
|
||
|
if output, err = exec.Command("docker-compose", args...).CombinedOutput(); err != nil {
|
||
|
log.Printf("exec.Command(docker-composer) args(%v) stdout(%s) error(%v)", args, string(output), err)
|
||
|
return
|
||
|
}
|
||
|
var id = string(bytes.TrimSpace(output))
|
||
|
args = []string{"inspect", id, "--format", "'{{json .}}'"}
|
||
|
if output, err = exec.Command("docker", args...).CombinedOutput(); err != nil {
|
||
|
log.Printf("exec.Command(docker) args(%v) stdout(%s) error(%v)", args, string(output), err)
|
||
|
return
|
||
|
}
|
||
|
if output = bytes.TrimSpace(output); bytes.Equal(output, []byte("")) {
|
||
|
err = fmt.Errorf("service: %s | container: %s fails to launch", svr, id)
|
||
|
log.Printf("exec.Command(docker) args(%v) error(%v)", args, err)
|
||
|
return
|
||
|
}
|
||
|
var c = &Container{}
|
||
|
if err = json.Unmarshal(bytes.Trim(output, "'"), c); err != nil {
|
||
|
log.Printf("json.Unmarshal(%s) error(%v)", string(output), err)
|
||
|
return
|
||
|
}
|
||
|
services[string(svr)] = c
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
func checkServices() (output []byte, err error) {
|
||
|
defer func() {
|
||
|
if err != nil && retry < 4 {
|
||
|
retry++
|
||
|
getServices()
|
||
|
time.Sleep(time.Second * 5)
|
||
|
output, err = checkServices()
|
||
|
return
|
||
|
}
|
||
|
retry = 0
|
||
|
}()
|
||
|
for svr, c := range services {
|
||
|
if err = c.Healthcheck(); err != nil {
|
||
|
log.Printf("healthcheck(%s) error(%v) retrying %d times...", svr, err, 5-retry)
|
||
|
return
|
||
|
}
|
||
|
// TODO About container check and more...
|
||
|
}
|
||
|
return
|
||
|
}
|