release: v0.2.0
Some checks failed
Release / Build & Release (push) Has been cancelled

Comprehensive release containing structural, UX, and behavioral upgrades since v0.1.0:

1. Namespace Transition: Renamed core executable and project namespaces from 'smart-monitor' to 'smart-shutdown'.
2. Objective Vocabulary Refactoring: Normalized output strings and logging descriptors system-wide to a strict, professional tone.
3. Advanced Status Query: 'status' subcommand now retrieves the parsed configuration, log locations/sizes, and tails the last 10 lines of the local log file.
4. Runtime Configuration Setter: Introduced 'config set' subcommand to modify the configuration file with strict type validations.
5. Auto-System Deployment: Remapped 'install' to clone the executable into system domains and register global PATH variables.
6. Cleaner Removal: 'uninstall' purges binary clones and clears environmental variables, leaving zero traces.
7. Documentation Rewrite: Generated an objective README file featuring copy-paste ready Markdown blocks.
This commit is contained in:
2026-03-24 16:07:22 +08:00
parent 3525a59976
commit 3be3e19e49
10 changed files with 243 additions and 237 deletions

View File

@@ -1,70 +1,89 @@
package logger
import (
"io"
"log"
"fmt"
"os"
"path/filepath"
"time"
"github.com/fatih/color"
"gopkg.in/natefinch/lumberjack.v2"
"smart-shutdown/pkg/config"
)
var (
infoLogger *log.Logger
succLogger *log.Logger
warnLogger *log.Logger
failLogger *log.Logger
critLogger *log.Logger
)
var fileLogger *lumberjack.Logger
// InitLogger 初始化全局日志系统,将日志同时输出到终端色彩日志和滚动文件
func InitLogger() error {
logDir := config.GetLogDir()
if err := os.MkdirAll(logDir, 0755); err != nil {
// 降级为当前目录,适用于未具有系统写权限的 Local 执行
logDir = "logs"
os.MkdirAll(logDir, 0755)
}
logFile := filepath.Join(logDir, "network_monitor.log")
// 使用 lumberjack 实现每天轮转和最大保留 30 天功能
ljLogger := &lumberjack.Logger{
fileLogger = &lumberjack.Logger{
Filename: logFile,
MaxSize: 10, // 每个切片最大兆字节 (MB)
MaxBackups: 30, // 保留最近 30 个切片
MaxAge: 30, // 保留最近 30 天的文件
Compress: false, // 是否压缩
MaxSize: 10,
MaxBackups: 30,
MaxAge: 30,
Compress: false,
}
// 各级别终端色彩
cSucc := color.New(color.FgGreen).SprintFunc()
cFail := color.New(color.FgRed).SprintFunc()
cWarn := color.New(color.FgYellow).SprintFunc()
cCrit := color.New(color.FgHiRed, color.Bold).SprintFunc()
// 构建复合输出写入器:终端标准输出 + 日志文件
outGeneral := io.MultiWriter(os.Stdout, ljLogger)
outError := io.MultiWriter(os.Stderr, ljLogger)
// 时间戳格式前缀
flags := log.Ldate | log.Ltime
// 实例化各个级别 Logger
infoLogger = log.New(outGeneral, "[INFO] ", flags)
succLogger = log.New(outGeneral, cSucc("[SUCCESS] "), flags)
warnLogger = log.New(outGeneral, cWarn("[WARN] "), flags)
failLogger = log.New(outError, cFail("[FAIL] "), flags)
critLogger = log.New(outError, cCrit("[CRITICAL] "), flags)
return nil
}
// 封装导出供业务调用的函数
func Info(format string, v ...interface{}) { infoLogger.Printf(format, v...) }
func Succ(format string, v ...interface{}) { succLogger.Printf(format, v...) }
func Warn(format string, v ...interface{}) { warnLogger.Printf(format, v...) }
func Fail(format string, v ...interface{}) { failLogger.Printf(format, v...) }
func Crit(format string, v ...interface{}) { critLogger.Printf(format, v...) }
func writeLog(level, plainPrefix, format string, v ...interface{}) {
msg := fmt.Sprintf(format, v...)
timestamp := time.Now().Format("2006/01/02 15:04:05")
if fileLogger != nil {
plainLine := fmt.Sprintf("%s %s %s\n", plainPrefix, timestamp, msg)
fileLogger.Write([]byte(plainLine))
}
prefixColorFunc := getPrefixColor(level)
coloredPrefix := prefixColorFunc(plainPrefix)
coloredLine := fmt.Sprintf("%s %s %s\n", coloredPrefix, timestamp, msg)
if level == "FAIL" || level == "CRITICAL" {
fmt.Fprint(os.Stderr, coloredLine)
} else {
fmt.Fprint(os.Stdout, coloredLine)
}
}
func getPrefixColor(level string) func(a ...interface{}) string {
switch level {
case "SUCCESS":
return color.New(color.FgGreen).SprintFunc()
case "WARN":
return color.New(color.FgYellow).SprintFunc()
case "FAIL":
return color.New(color.FgRed).SprintFunc()
case "CRITICAL":
return color.New(color.FgHiRed).SprintFunc()
default:
return color.New(color.Reset).SprintFunc()
}
}
func Info(format string, v ...interface{}) {
writeLog("INFO", "[INFO]", format, v...)
}
func Succ(format string, v ...interface{}) {
writeLog("SUCCESS", "[SUCCESS]", format, v...)
}
func Warn(format string, v ...interface{}) {
writeLog("WARN", "[WARN]", format, v...)
}
func Fail(format string, v ...interface{}) {
writeLog("FAIL", "[FAIL]", format, v...)
}
func Crit(format string, v ...interface{}) {
writeLog("CRITICAL", "[CRITICAL]", format, v...)
}