Golang在Linux上的自动化运维实现指南
在Linux(如Debian、Ubuntu)上使用Golang前,需先完成环境配置:
wget https://golang.org/dl/go1.23.linux-amd64.tar.gz
sudo tar -C /usr/local -xzf go1.23.linux-amd64.tar.gz
~/.bashrc(或~/.zshrc),添加以下内容:export GOROOT=/usr/local/go
export GOPATH=$HOME/go
export PATH=$PATH:$GOROOT/bin:$GOPATH/bin
运行source ~/.bashrc使配置生效。使用time.Ticker实现周期性任务(每5分钟检查一次),通过exec.Command执行df -h命令,解析输出并打印磁盘状态:
package main
import (
"fmt"
"os/exec"
"strings"
"time"
)
func checkDiskUsage() {
cmd := exec.Command("df", "-h")
output, err := cmd.Output()
if err != nil {
fmt.Printf("执行df命令失败: %v\n", err)
return
}
lines := strings.Split(string(output), "\n")
for _, line := range lines {
if strings.Contains(line, "/dev/") {
fmt.Println("磁盘状态:", line)
}
}
}
func main() {
ticker := time.NewTicker(5 * time.Minute)
defer ticker.Stop()
checkDiskUsage() // 立即执行一次
for range ticker.C {
checkDiskUsage()
}
}
通过filepath.Walk遍历日志目录(如/var/log/myapp),删除修改时间超过7天的.log文件,使用time.Ticker每日执行一次:
package main
import (
"fmt"
"os"
"path/filepath"
"time"
)
func cleanupLogs(logDir string, maxAgeDays int) {
now := time.Now()
cutoff := now.AddDate(0, 0, -maxAgeDays)
err := filepath.Walk(logDir, func(path string, info os.FileInfo, err error) error {
if err != nil {
return nil
}
if !info.IsDir() && filepath.Ext(path) == ".log" && info.ModTime().Before(cutoff) {
if err := os.Remove(path); err == nil {
fmt.Printf("已删除过期日志: %s\n", path)
}
}
return nil
})
if err != nil {
fmt.Printf("清理日志时出错: %v\n", err)
}
}
func main() {
for range time.NewTicker(24 * time.Hour).C {
cleanupLogs("/var/log/myapp", 7)
}
}
通过systemctl is-active命令检查服务(如myweb)状态,若未运行则执行systemctl start重启:
package main
import (
"fmt"
"os/exec"
"strings"
"time"
)
func isServiceRunning(serviceName string) bool {
cmd := exec.Command("systemctl", "is-active", serviceName)
output, err := cmd.Output()
return err == nil && strings.TrimSpace(string(output)) == "active"
}
func startService(serviceName string) {
cmd := exec.Command("systemctl", "start", serviceName)
if err := cmd.Run(); err != nil {
fmt.Printf("启动服务 %s 失败: %v\n", serviceName, err)
} else {
fmt.Printf("已启动服务: %s\n", serviceName)
}
}
func monitorService(serviceName string) {
ticker := time.NewTicker(30 * time.Second)
defer ticker.Stop()
for range ticker.C {
if !isServiceRunning(serviceName) {
fmt.Printf("服务 %s 未运行,尝试重启...\n", serviceName)
startService(serviceName)
}
}
}
func main() {
monitorService("myweb")
}
使用golang.org/x/crypto/ssh包实现SSH连接,批量执行命令(如查看远程服务器的磁盘使用率):
package main
import (
"bytes"
"fmt"
"golang.org/x/crypto/ssh"
"log"
)
func executeRemoteCommand(user, password, host, command string) {
config := &ssh.ClientConfig{
User: user,
Auth: []ssh.AuthMethod{
ssh.Password(password),
},
HostKeyCallback: ssh.InsecureIgnoreHostKey(), // 生产环境建议使用HostKeyCallback验证
}
client, err := ssh.Dial("tcp", host+":22", config)
if err != nil {
log.Fatalf("无法连接到服务器 %s: %v", host, err)
}
defer client.Close()
session, err := client.NewSession()
if err != nil {
log.Fatalf("无法创建会话: %v", err)
}
defer session.Close()
var output bytes.Buffer
session.Stdout = &output
if err := session.Run(command); err != nil {
log.Printf("执行命令失败: %v", err)
} else {
fmt.Printf("服务器 %s 输出: %s\n", host, output.String())
}
}
func main() {
servers := []string{"192.168.1.101:22", "192.168.1.102:22"}
user := "root"
password := "your_password"
command := "df -h"
for _, server := range servers {
executeRemoteCommand(user, password, server, command)
}
}
Cobra是Golang主流的CLI框架,支持子命令、全局参数、自动帮助生成。以myops工具为例:
cobra init --pkg-name myops
cobra add deploy # 添加deploy子命令
cobra add rollback # 添加rollback子命令
cmd/deploy.go,定义Run函数实现部署逻辑:package cmd
import (
"fmt"
"github.com/spf13/cobra"
)
var env, app string
var deployCmd = &cobra.Command{
Use: "deploy",
Short: "Deploy application",
Run: func(cmd *cobra.Command, args []string) {
fmt.Printf("Deploying %s to %s environment\n", app, env)
// 实现部署逻辑(如调用Docker、K8s API)
},
}
func init() {
deployCmd.Flags().StringVarP(&env, "env", "e", "prod", "Environment (prod/stage/dev)")
deployCmd.Flags().StringVarP(&app, "app", "a", "web", "Application name")
rootCmd.AddCommand(deployCmd)
}
go build -o myops
./myops deploy --env=prod --app=web
通过交叉编译生成Linux平台无依赖二进制文件,便于分发:
GOOS=linux GOARCH=amd64 go build -o myops-linux
结合CI/CD(如GitHub Actions),可实现自动构建多平台版本并上传至仓库。
exec.Command的返回值检查),避免程序崩溃。zap或logrus库实现分级日志(info/debug/error),便于排查问题。sync.Mutex或channel保证并发安全。