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保證并發安全。