package main

import (
	"cicv-data-closedloop/common/util"
	"cicv-data-closedloop/kinglong/common/cfg"
	"cicv-data-closedloop/kinglong/common/log"
	commonService "cicv-data-closedloop/kinglong/common/svc"
	"net/rpc"
	"os"
	"runtime"
	"time"
)

func init() {
	runtime.GOMAXPROCS(1)
	// 初始化日志配置
	log.InitLogConfig("kinglong-control")
	// 初始化本地配置文件(第1处配置,在本地文件)
	cfg.InitLocalConfig()
	// 初始化Oss连接信息
	cfg.InitOssConfig()
	// 初始化业务逻辑配置信息,配置文件在oss上(第2处配置,在oss文件)
	cfg.InitCloudConfig()
}

func main() {
	lastStatus := "NONE"
	//  轮询任务接口判断是否有更新
	for {
		time.Sleep(time.Duration(2) * time.Second)
		// 1 获取当前设备的任务的 status
		status, err := cfg.GetStatus(cfg.PlatformConfig.TaskConfigId)
		if err != nil {
			log.GlobalLogger.Error("获取配置status失败:", err)
			continue
		}
		// 2 判断 status
		// UN_CHANGE 没有新的任务,无需更改
		// CHANGE 有新的任务,需要杀死旧的任务并重启
		// NONE 设备没有配置任务,需要杀死旧的任务
		if status == "UN_CHANGE" {
			lastStatus = "UN_CHANGE"
			continue
		} else if status == "CHANGE" || status == "NONE" {
			if lastStatus == "NONE" && status == "NONE" {
				continue
			}
			// 3 发送rpc信号杀死两个服务,并重启程序
			if lastStatus == "NONE" && status == "CHANGE" {
				if _, err := util.ExecuteWithPath(cfg.LocalConfig.RestartCmd.Dir, cfg.LocalConfig.RestartCmd.Name, cfg.LocalConfig.RestartCmd.Args...); err != nil {
					log.GlobalLogger.Info("启动新程序失败,【path】=", cfg.LocalConfig.RestartCmd.Dir, "【cmd】=", cfg.LocalConfig.RestartCmd.Name, cfg.LocalConfig.RestartCmd.Args, ":", err)
					os.Exit(-1)
				}
				log.GlobalLogger.Info("启动任务,本地执行启动命令:【path】=", cfg.LocalConfig.RestartCmd.Dir, "【cmd】=", cfg.LocalConfig.RestartCmd.Name, cfg.LocalConfig.RestartCmd.Args)
				lastStatus = status
				log.GlobalLogger.Info("获取数据闭环平台最新配置。")
				cfg.InitPlatformConfig()
				continue
			}
			var killArgs *commonService.KillSignal
			if lastStatus == "UN_CHANGE" && status == "CHANGE" {
				killArgs = &commonService.KillSignal{NodeName: cfg.LocalConfig.Node.Name, DropUploadData: cfg.PlatformConfig.DropUploadData, Restart: true}
				log.GlobalLogger.Info("更新任务,发送rpc重启信号到本地"+cfg.LocalConfig.Node.Name+":", killArgs)
			}
			if lastStatus == "UN_CHANGE" && status == "NONE" {
				killArgs = &commonService.KillSignal{NodeName: cfg.LocalConfig.Node.Name, DropUploadData: cfg.PlatformConfig.DropUploadData, Restart: false}
				log.GlobalLogger.Info("杀死任务,发送rpc结束信号到本地"+cfg.LocalConfig.Node.Name+":", killArgs)
			}

			KillRpcClient, err := rpc.Dial("tcp", cfg.LocalConfig.Node.Ip+":"+cfg.CloudConfig.RpcPort)
			if err != nil {
				log.GlobalLogger.Error("创建rpc客户端连接master失败:", err)
				// 此处关闭client会报错
				continue
			}

			reply := 0
			if err = KillRpcClient.Call("KillService.Kill", killArgs, &reply); err != nil {
				log.GlobalLogger.Error("发送rpc请求到master失败:", err)
				//TODO 这里可能会报错 unexpected EOF 但是不影响,先注释 close 和 continue
				//KillRpcClient.Close()
				//continue
			}
			lastStatus = status
			log.GlobalLogger.Info("获取数据闭环平台最新配置。")
			cfg.InitPlatformConfig()
			KillRpcClient.Close()
		} else {
			log.GlobalLogger.Error("未知的采集任务状态。【status】=", status)
		}
	}
}