181 lines
4.2 KiB
Go
181 lines
4.2 KiB
Go
|
|
package main
|
|||
|
|
|
|||
|
|
import (
|
|||
|
|
"context"
|
|||
|
|
"coredump-handler/config"
|
|||
|
|
"coredump-handler/logger"
|
|||
|
|
"errors"
|
|||
|
|
"fmt"
|
|||
|
|
"io"
|
|||
|
|
"io/ioutil"
|
|||
|
|
"os"
|
|||
|
|
"regexp"
|
|||
|
|
"strconv"
|
|||
|
|
"syscall"
|
|||
|
|
|
|||
|
|
"github.com/containerd/containerd"
|
|||
|
|
"github.com/containerd/containerd/namespaces"
|
|||
|
|
)
|
|||
|
|
|
|||
|
|
func judgeMemory(pipe_config config.Pipeconfig) (bool, error) {
|
|||
|
|
percent, err := strconv.ParseFloat(pipe_config.Total_file_mem_limit[:len(pipe_config.Total_file_mem_limit)-1], 64)
|
|||
|
|
if err != nil {
|
|||
|
|
return false, err
|
|||
|
|
}
|
|||
|
|
percent = percent / 100.0
|
|||
|
|
var stat syscall.Statfs_t
|
|||
|
|
wd, err := syscall.Getwd()
|
|||
|
|
if err != nil {
|
|||
|
|
fmt.Println(err)
|
|||
|
|
return false, err
|
|||
|
|
}
|
|||
|
|
syscall.Statfs(wd, &stat)
|
|||
|
|
// 剩余空间的大小为块的数量 * 每个块的大小
|
|||
|
|
// stat.Bfree表示可用的块的数量,stat.Bsize表示每个块的大小
|
|||
|
|
usedSpacePer := float64(int64(stat.Blocks)-int64(stat.Bfree)) / int64(stat.Blocks)
|
|||
|
|
if usedSpacePer >= percent {
|
|||
|
|
return false, nil
|
|||
|
|
}
|
|||
|
|
return true, nil
|
|||
|
|
}
|
|||
|
|
func createCoreDumpDir(pipe_config *config.Pipeconfig, args []string) error {
|
|||
|
|
pipe_config.File_base_path = fmt.Sprintf("%s/%s_%s_%s", pipe_config.File_base_path, args[1], args[2], args[3])
|
|||
|
|
dirName := pipe_config.File_base_path
|
|||
|
|
if _, err := os.Stat(dirName); os.IsNotExist(err) {
|
|||
|
|
// 目录不存在,创建目录
|
|||
|
|
if err := os.MkdirAll(dirName, os.ModePerm); err != nil {
|
|||
|
|
return err
|
|||
|
|
}
|
|||
|
|
} else {
|
|||
|
|
return errors.New("directory already exists")
|
|||
|
|
}
|
|||
|
|
return nil
|
|||
|
|
}
|
|||
|
|
func Chdir(dir string) error {
|
|||
|
|
if err := os.Chdir(dir); err != nil {
|
|||
|
|
return err
|
|||
|
|
}
|
|||
|
|
return nil
|
|||
|
|
}
|
|||
|
|
func getContainerId(pid string) (string, error) {
|
|||
|
|
cgroup_path := fmt.Sprintf("/proc/%s/cgroup", pid)
|
|||
|
|
content, err := ioutil.ReadFile(cgroup_path)
|
|||
|
|
if err != nil {
|
|||
|
|
return "", err
|
|||
|
|
}
|
|||
|
|
re := regexp.MustCompile(`([a-f\d]{64})`)
|
|||
|
|
match := re.FindStringSubmatch(string(content))
|
|||
|
|
if len(match) < 2 {
|
|||
|
|
return "", errors.New("failed to extract container ID from cgroup file")
|
|||
|
|
}
|
|||
|
|
containerID := match[1]
|
|||
|
|
return containerID, nil
|
|||
|
|
}
|
|||
|
|
func getImageId(container_id string, sock_path string) (string, error) {
|
|||
|
|
// 连接 containerd daemon
|
|||
|
|
client, err := containerd.New(sock_path)
|
|||
|
|
if err != nil {
|
|||
|
|
logger.Logger.Println(err)
|
|||
|
|
return "", err
|
|||
|
|
}
|
|||
|
|
defer client.Close()
|
|||
|
|
// 根据容器 ID 获取容器信息
|
|||
|
|
ctx := namespaces.WithNamespace(context.Background(), "k8s.io")
|
|||
|
|
container, err := client.LoadContainer(ctx, container_id)
|
|||
|
|
if err != nil {
|
|||
|
|
logger.Logger.Println(err)
|
|||
|
|
return "", err
|
|||
|
|
}
|
|||
|
|
// 获取容器关联的镜像信息
|
|||
|
|
imageRef, err := container.Image(ctx)
|
|||
|
|
if err != nil {
|
|||
|
|
logger.Logger.Println(err)
|
|||
|
|
return "", err
|
|||
|
|
}
|
|||
|
|
return imageRef.Name(), nil
|
|||
|
|
}
|
|||
|
|
func writeCoreConfig(data string) error {
|
|||
|
|
err := ioutil.WriteFile("coredump.config", []byte(data), 0666)
|
|||
|
|
if err != nil {
|
|||
|
|
return err
|
|||
|
|
}
|
|||
|
|
fmt.Printf("Wrote %d bytes to file\n", len(data))
|
|||
|
|
return nil
|
|||
|
|
}
|
|||
|
|
func writeCoreDump() error {
|
|||
|
|
file, err := os.Create("coredump.info")
|
|||
|
|
if err != nil {
|
|||
|
|
return err
|
|||
|
|
}
|
|||
|
|
defer file.Close()
|
|||
|
|
|
|||
|
|
// 从标准输入流中读取数据,并将其写入文件中
|
|||
|
|
buf := make([]byte, 1024)
|
|||
|
|
for {
|
|||
|
|
n, err := io.ReadAtLeast(os.Stdin, buf, 1)
|
|||
|
|
if err != nil && err != io.EOF {
|
|||
|
|
return err
|
|||
|
|
}
|
|||
|
|
if n == 0 {
|
|||
|
|
break
|
|||
|
|
}
|
|||
|
|
if _, err := file.Write(buf[:n]); err != nil {
|
|||
|
|
return err
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
return nil
|
|||
|
|
}
|
|||
|
|
func main() {
|
|||
|
|
err := Chdir("/root/pipe/corepipe")
|
|||
|
|
if err != nil {
|
|||
|
|
fmt.Println(err)
|
|||
|
|
return
|
|||
|
|
}
|
|||
|
|
pipe_config, err := config.PipeInit()
|
|||
|
|
if err != nil {
|
|||
|
|
fmt.Println(err)
|
|||
|
|
return
|
|||
|
|
}
|
|||
|
|
flag, err := judgeMemory(pipe_config)
|
|||
|
|
if err != nil && !flag {
|
|||
|
|
fmt.Println(err)
|
|||
|
|
return
|
|||
|
|
}
|
|||
|
|
if len(os.Args) < 4 {
|
|||
|
|
fmt.Println("parameter passing exception")
|
|||
|
|
return
|
|||
|
|
}
|
|||
|
|
err = createCoreDumpDir(&pipe_config, os.Args)
|
|||
|
|
if err != nil {
|
|||
|
|
fmt.Println(err)
|
|||
|
|
return
|
|||
|
|
}
|
|||
|
|
err = Chdir(pipe_config.File_base_path)
|
|||
|
|
if err != nil {
|
|||
|
|
fmt.Println(err)
|
|||
|
|
return
|
|||
|
|
}
|
|||
|
|
err = logger.Init()
|
|||
|
|
if err != nil {
|
|||
|
|
fmt.Println(err)
|
|||
|
|
return
|
|||
|
|
}
|
|||
|
|
logger.Logger.Println("logger init ok!")
|
|||
|
|
container_id, err := getContainerId(os.Args[1])
|
|||
|
|
var image_id string
|
|||
|
|
if err == nil && len(container_id) != 0 {
|
|||
|
|
image_id, err = getImageId(container_id, pipe_config.Containered_sock_path)
|
|||
|
|
if err != nil {
|
|||
|
|
logger.Logger.Println(err)
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
err = writeCoreConfig(image_id)
|
|||
|
|
if err != nil {
|
|||
|
|
logger.Logger.Println(err)
|
|||
|
|
}
|
|||
|
|
err = writeCoreDump()
|
|||
|
|
if err != nil {
|
|||
|
|
logger.Logger.Println(err)
|
|||
|
|
}
|
|||
|
|
}
|