127 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			127 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
package core
 | 
						|
 | 
						|
import (
 | 
						|
	"fmt"
 | 
						|
	"os"
 | 
						|
	"os/signal"
 | 
						|
	"syscall"
 | 
						|
 | 
						|
	"git.huangwc.com/pig/pig-farm-controller/internal/app/api"
 | 
						|
	"git.huangwc.com/pig/pig-farm-controller/internal/infra/config"
 | 
						|
	"git.huangwc.com/pig/pig-farm-controller/internal/infra/logs"
 | 
						|
)
 | 
						|
 | 
						|
// Application 是整个应用的核心,封装了所有组件和生命周期。
 | 
						|
type Application struct {
 | 
						|
	Config *config.Config
 | 
						|
	Logger *logs.Logger
 | 
						|
	API    *api.API
 | 
						|
 | 
						|
	Infra  *Infrastructure
 | 
						|
	Domain *DomainServices
 | 
						|
	App    *AppServices
 | 
						|
}
 | 
						|
 | 
						|
// NewApplication 创建并初始化一个新的 Application 实例。
 | 
						|
// 这是应用的“组合根”,所有依赖都在这里被创建和注入。
 | 
						|
func NewApplication(configPath string) (*Application, error) {
 | 
						|
	// 1. 初始化基本组件: 配置和日志
 | 
						|
	cfg := config.NewConfig()
 | 
						|
	if err := cfg.Load(configPath); err != nil {
 | 
						|
		return nil, fmt.Errorf("无法加载配置: %w", err)
 | 
						|
	}
 | 
						|
	logger := logs.NewLogger(cfg.Log)
 | 
						|
 | 
						|
	// 2. 初始化所有分层服务
 | 
						|
	infra, err := initInfrastructure(cfg, logger)
 | 
						|
	if err != nil {
 | 
						|
		return nil, fmt.Errorf("初始化基础设施失败: %w", err)
 | 
						|
	}
 | 
						|
	domain := initDomainServices(cfg, infra, logger)
 | 
						|
	appServices := initAppServices(infra, domain, logger)
 | 
						|
 | 
						|
	// 3. 初始化 API 入口点
 | 
						|
	apiServer := api.NewAPI(
 | 
						|
		cfg.Server,
 | 
						|
		logger,
 | 
						|
		infra.Repos.UserRepo,
 | 
						|
		appServices.PigFarmService,
 | 
						|
		appServices.PigBatchService,
 | 
						|
		appServices.MonitorService,
 | 
						|
		appServices.DeviceService,
 | 
						|
		appServices.PlanService,
 | 
						|
		appServices.UserService,
 | 
						|
		infra.TokenService,
 | 
						|
		appServices.AuditService,
 | 
						|
		infra.Lora.ListenHandler,
 | 
						|
	)
 | 
						|
 | 
						|
	// 4. 组装 Application 对象
 | 
						|
	app := &Application{
 | 
						|
		Config: cfg,
 | 
						|
		Logger: logger,
 | 
						|
		API:    apiServer,
 | 
						|
		Infra:  infra,
 | 
						|
		Domain: domain,
 | 
						|
		App:    appServices,
 | 
						|
	}
 | 
						|
 | 
						|
	return app, nil
 | 
						|
}
 | 
						|
 | 
						|
// Start 启动应用的所有组件并阻塞,直到接收到关闭信号。
 | 
						|
func (app *Application) Start() error {
 | 
						|
	app.Logger.Info("应用启动中...")
 | 
						|
 | 
						|
	// 1. 启动底层监听器
 | 
						|
	if err := app.Infra.Lora.LoraListener.Listen(); err != nil {
 | 
						|
		return fmt.Errorf("启动 LoRa Mesh 监听器失败: %w", err)
 | 
						|
	}
 | 
						|
 | 
						|
	// 2. 初始化应用状态 (清理、刷新任务等)
 | 
						|
	if err := app.initializeState(); err != nil {
 | 
						|
		return fmt.Errorf("初始化应用状态失败: %w", err)
 | 
						|
	}
 | 
						|
 | 
						|
	// 3. 启动后台工作协程
 | 
						|
	app.Domain.PlanExecutionManager.Start()
 | 
						|
 | 
						|
	// 4. 启动 API 服务器
 | 
						|
	app.API.Start()
 | 
						|
 | 
						|
	// 5. 等待关闭信号
 | 
						|
	quit := make(chan os.Signal, 1)
 | 
						|
	signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM)
 | 
						|
	<-quit
 | 
						|
 | 
						|
	// 接收到信号后,执行优雅关闭
 | 
						|
	return app.Stop()
 | 
						|
}
 | 
						|
 | 
						|
// Stop 优雅地关闭应用的所有组件。
 | 
						|
func (app *Application) Stop() error {
 | 
						|
	app.Logger.Info("应用关闭中...")
 | 
						|
 | 
						|
	// 关闭 API 服务器
 | 
						|
	app.API.Stop()
 | 
						|
 | 
						|
	// 关闭任务执行器
 | 
						|
	app.Domain.PlanExecutionManager.Stop()
 | 
						|
 | 
						|
	// 断开数据库连接
 | 
						|
	if err := app.Infra.Storage.Disconnect(); err != nil {
 | 
						|
		app.Logger.Errorw("数据库连接断开失败", "error", err)
 | 
						|
	}
 | 
						|
 | 
						|
	// 关闭 LoRa Mesh 监听器
 | 
						|
	if err := app.Infra.Lora.LoraListener.Stop(); err != nil {
 | 
						|
		app.Logger.Errorw("LoRa Mesh 监听器关闭失败", "error", err)
 | 
						|
	}
 | 
						|
 | 
						|
	// 刷新日志缓冲区
 | 
						|
	_ = app.Logger.Sync()
 | 
						|
 | 
						|
	app.Logger.Info("应用已成功关闭")
 | 
						|
	return nil
 | 
						|
}
 |