animeic 1 年之前
父節點
當前提交
4283940d67
共有 4 個文件被更改,包括 376 次插入59 次删除
  1. 1 1
      src/conf/app.go
  2. 3 3
      src/go.mod
  3. 263 0
      src/lancher1/main.go
  4. 109 55
      src/main.go

+ 1 - 1
src/conf/app.go

@@ -3,7 +3,7 @@ package conf
 import (
 	"fmt"
 	"os"
-	"uploader/log"
+	"updater/log"
 
 	"github.com/spf13/viper"
 )

+ 3 - 3
src/go.mod

@@ -1,4 +1,4 @@
-module uploader
+module updater
 
 go 1.20
 
@@ -9,9 +9,10 @@ require (
 	github.com/huaweicloud/huaweicloud-sdk-go-obs v3.23.9+incompatible
 	github.com/jessevdk/go-flags v1.5.0
 	github.com/natefinch/lumberjack v2.0.0+incompatible
+	github.com/nats-io/nats.go v1.22.1
 	github.com/spf13/viper v1.7.0
 	go.uber.org/zap v1.17.0
-	infish.cn/comm v0.0.0-incompatible
+	infish.cn/comm v0.0.0-00010101000000-000000000000
 )
 
 require (
@@ -34,7 +35,6 @@ require (
 	github.com/mitchellh/mapstructure v1.1.2 // indirect
 	github.com/montanaflynn/stats v0.0.0-20171201202039-1bf9dbcd8cbe // indirect
 	github.com/nats-io/jwt/v2 v2.5.2 // indirect
-	github.com/nats-io/nats.go v1.22.1 // indirect
 	github.com/nats-io/nkeys v0.4.4 // indirect
 	github.com/nats-io/nuid v1.0.1 // indirect
 	github.com/pelletier/go-toml v1.9.3 // indirect

+ 263 - 0
src/lancher1/main.go

@@ -0,0 +1,263 @@
+package main
+
+import (
+	"fmt"
+	"log"
+	"os"
+	"os/exec"
+	"path/filepath"
+	"time"
+	"updater/download"
+
+	"gioui.org/app"
+	"gioui.org/io/system"
+	"gioui.org/layout"
+	"gioui.org/op"
+	"gioui.org/text"
+	"gioui.org/unit"
+	"gioui.org/widget"
+	"gioui.org/widget/material"
+	"github.com/jessevdk/go-flags"
+	"github.com/nats-io/nats.go"
+	"infish.cn/comm"
+)
+
+// 启动程序
+// 下载文件->成功->失败
+// 下载成功->通知nats关闭原程序
+// 安装->成功->失败
+// 成功启动程序
+// 失败还原程序,关闭更新应用
+
+// Define the progress variables, a channel and a variable
+
+var progress float32
+var progressIncrementer chan float32
+
+var GAppOption = &AppOption{}
+
+type AppOption struct {
+	NatsPort int    `short:"p" long:"np" description:"nats port"`
+	Url      string `short:"u" long:"url" description:"url"`
+}
+
+func (o *AppOption) Parse() {
+	flags.NewParser(o, flags.Default|flags.IgnoreUnknown).Parse()
+}
+
+type ProcesCallback = func(int, string)
+
+func upgradeLancherExe(url string, cb ProcesCallback) {
+	exepath, _ := os.Executable()
+	appExeDir, _ := filepath.Split(exepath)
+
+	//1 下载lancher.exe
+	downLancherExe := filepath.Join(appExeDir, "lancher")
+	err := download.DownloadExe(GAppOption.Url, downLancherExe)
+	if err != nil {
+		fmt.Println(err)
+		cb(-1, "下载安装包失败!")
+		return
+	}
+
+	//2 通知lancher.exe退出
+	remoteCnn, err := comm.NewNatsBus(fmt.Sprintf("nats://localhost:%d", GAppOption.NatsPort), 1, 1, []*comm.NatsStreamWather{})
+	if err != nil {
+		fmt.Println(err)
+		cb(-1, "安装失败!")
+		return
+	}
+
+	err = remoteCnn.Publish("lancher.quit", []byte{})
+	if err != nil {
+		fmt.Println(err)
+		cb(-1, "安装失败!")
+		return
+	}
+	remoteCnn.GetNatsConn().Flush()
+
+	time.Sleep(time.Second * 2)
+
+	//3 安装新的lancher.exe
+	currLancherExe := filepath.Join(appExeDir, "lancher.exe")
+	currLancherExeback := filepath.Join(appExeDir, "lancher.exe.back")
+	err = os.Rename(currLancherExe, currLancherExeback)
+	if err != nil {
+		fmt.Println(err)
+		cb(-1, "安装失败!")
+		return
+	}
+
+	err = os.Rename(downLancherExe, currLancherExe)
+
+	if err != nil {
+		fmt.Println(err)
+		cb(-1, "安装失败!")
+		os.Rename(currLancherExeback, currLancherExe)
+		return
+	}
+
+	//4 启动lancher.exe
+	// 重启
+	cmd := exec.Command(currLancherExe)
+	cmd.Stdout = os.Stdout
+	cmd.Stderr = os.Stderr
+	err = cmd.Start()
+	if err != nil {
+		fmt.Printf("failed to call cmd.Start(): %v", err)
+		os.Rename(currLancherExeback, currLancherExe)
+		cb(-1, "新包启动失败!")
+		return
+	}
+	cb(1, "")
+}
+
+var installError = ""
+var installSucc = false
+
+func main() {
+	GAppOption.Parse()
+
+	fmt.Println("firt main xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx")
+
+	//2 通知lancher.exe退出
+	remoteCnn, _ := comm.NewNatsBus(fmt.Sprintf("nats://localhost:%d", GAppOption.NatsPort), 1, 1, []*comm.NatsStreamWather{})
+
+	remoteCnn.AddReplyers(&comm.NatsMsgReplyer{
+		Subject: "lancher.quit",
+		Cb: func(msg *nats.Msg, entity interface{}) interface{} {
+			go func() {
+				time.Sleep(time.Second)
+				os.Exit(0)
+			}()
+			return nil
+		},
+	})
+
+	go func() {
+		remoteCnn.Run(func() {})
+	}()
+
+	// // Setup a separate channel to provide ticks to increment progress
+	progressIncrementer = make(chan float32)
+	go func() {
+		for {
+			time.Sleep(time.Second / 25)
+			progressIncrementer <- 0.004
+		}
+	}()
+
+	w := app.NewWindow(
+		app.Title("uploader"),
+		app.Size(unit.Dp(600), unit.Dp(400)),
+	)
+
+	go func() {
+		// create new window
+
+		if err := draw(w); err != nil {
+			log.Fatal(err)
+		}
+		os.Exit(0)
+	}()
+
+	go func() {
+		time.Sleep(3 * time.Second)
+
+		cmd := exec.Command("./updater", "--np=14220", "--url=https://lancher.obs.cn-east-3.myhuaweicloud.com/test/main")
+		cmd.Stdout = os.Stdout
+		cmd.Stderr = os.Stderr
+
+		fmt.Println("started = >", cmd.Start())
+	}()
+
+	app.Main()
+
+}
+
+type C = layout.Context
+type D = layout.Dimensions
+
+func draw(w *app.Window) error {
+	// ops are the operations from the UI
+	var ops op.Ops
+
+	// startButton is a clickable widget
+	var startButton widget.Clickable
+
+	// is the egg boiling?
+	var boiling bool
+
+	// th defines the material design style
+	th := material.NewTheme([]text.FontFace{})
+
+	for {
+		select {
+		// listen for events in the window.
+		case e := <-w.Events():
+
+			// detect what type of event
+			switch e := e.(type) {
+
+			// this is sent when the application should re-render.
+			case system.FrameEvent:
+				gtx := layout.NewContext(&ops, e)
+				// Let's try out the flexbox layout concept
+				if startButton.Clicked() {
+					boiling = !boiling
+				}
+
+				layout.Flex{
+					// Vertical alignment, from top to bottom
+					Axis: layout.Vertical,
+					// Empty space is left at the start, i.e. at the top
+					Spacing: layout.SpaceStart,
+				}.Layout(gtx,
+					layout.Rigid(
+						func(gtx C) D {
+							bar := material.ProgressBar(th, progress)
+							return bar.Layout(gtx)
+						},
+					),
+					layout.Rigid(
+						func(gtx C) D {
+							// We start by defining a set of margins
+							margins := layout.Inset{
+								Top:    unit.Dp(25),
+								Bottom: unit.Dp(25),
+								Right:  unit.Dp(25),
+								Left:   unit.Dp(25),
+							}
+							// Then we lay out within those margins ...
+							return margins.Layout(gtx,
+								// ...the same function we earlier used to create a button
+								func(gtx C) D {
+									var text string
+									if !boiling {
+										text = "Start"
+									} else {
+										text = "Stop"
+									}
+									btn := material.Button(th, &startButton, text)
+									return btn.Layout(gtx)
+								},
+							)
+						},
+					),
+				)
+				e.Frame(gtx.Ops)
+
+			// this is sent when the application is closed.
+			case system.DestroyEvent:
+				return e.Err
+			}
+
+		//listen for events in the incrementor channel
+		case p := <-progressIncrementer:
+			if boiling && progress < 1 {
+				progress += p
+				w.Invalidate()
+			}
+		}
+	}
+}

+ 109 - 55
src/main.go

@@ -1,15 +1,13 @@
 package main
 
 import (
-	"encoding/json"
 	"fmt"
-	"io/fs"
 	"log"
 	"os"
 	"os/exec"
-	"path"
 	"path/filepath"
-	"uploader/download"
+	"time"
+	"updater/download"
 
 	"gioui.org/app"
 	"gioui.org/io/system"
@@ -38,78 +36,134 @@ var progressIncrementer chan float32
 var GAppOption = &AppOption{}
 
 type AppOption struct {
-	NatsPort   int    `short:"np" long:"np" description:"nats port"`
-	AppExePath string `short:"ap" long:"ap" description:"app exe path"`
-	Url        string `short:"url" long:"url" description:"url"`
+	NatsPort int    `short:"p" long:"np" description:"nats port"`
+	Url      string `short:"u" long:"url" description:"url"`
 }
 
 func (o *AppOption) Parse() {
 	flags.NewParser(o, flags.Default|flags.IgnoreUnknown).Parse()
 }
 
-func main() {
-	GAppOption.Parse()
-	err := download.DownloadExe(GAppOption.Url, GAppOption.AppExePath)
+type ProcesCallback = func(int, string)
+
+func upgradeLancherExe(url string, cb ProcesCallback) {
+	exepath, _ := os.Executable()
+	appExeDir, _ := filepath.Split(exepath)
+
+	//1 下载lancher.exe
+	downLancherExe := filepath.Join(appExeDir, "lancher")
+	err := download.DownloadExe(downLancherExe, GAppOption.Url)
+	if err != nil {
+		fmt.Println(err)
+		cb(-1, "下载安装包失败!")
+		return
+	}
+
+	//2 通知lancher.exe退出
+	remoteCnn, err := comm.NewNatsBus(fmt.Sprintf("nats://localhost:%d", GAppOption.NatsPort), 1, 1, []*comm.NatsStreamWather{})
+	if err != nil {
+		fmt.Println(err)
+		cb(-1, "安装失败!")
+		return
+	}
+
+	err = remoteCnn.Publish("lancher.quit", []byte{})
 	if err != nil {
 		fmt.Println(err)
+		cb(-1, "安装失败!")
 		return
 	}
-	remoteCnn, _ := comm.NewNatsBus(fmt.Sprintf("nats://localhost:%d", GAppOption.NatsPort), 1, 1, []*comm.NatsStreamWather{})
-	if remoteCnn != nil {
-		d, _ := json.Marshal(map[string]string{"shortcut": ""})
-		remoteCnn.Publish("lancher.resume", d)
-		remoteCnn.GetNatsConn().Flush()
+	remoteCnn.GetNatsConn().Flush()
+
+	time.Sleep(time.Second * 3)
+
+	//3 安装新的lancher.exe
+	currLancherExe := filepath.Join(appExeDir, "main")
+	currLancherExeback := filepath.Join(appExeDir, "lancher.exe.back")
+	err = os.Rename(currLancherExe, currLancherExeback)
+	if err != nil {
+		fmt.Println(err)
+		cb(-1, "安装失败!")
+		return
 	}
-	appExeDir, _ := filepath.Split(GAppOption.AppExePath)
-	os.Rename(GAppOption.AppExePath, fmt.Sprintf("%s/lancherlast", appExeDir))
 
-	fmt.Println("need restart ", GAppOption.AppExePath)
+	err = os.Rename(downLancherExe, currLancherExe)
+
+	if err != nil {
+		fmt.Println(err)
+		cb(-1, "安装失败!")
+		os.Rename(currLancherExeback, currLancherExe)
+		return
+	}
+
+	//4 启动lancher.exe
 	// 重启
-	cmd := exec.Command(GAppOption.AppExePath)
+	cmd := exec.Command(currLancherExe, "--np=14220")
 	cmd.Stdout = os.Stdout
 	cmd.Stderr = os.Stderr
 	err = cmd.Start()
 	if err != nil {
 		fmt.Printf("failed to call cmd.Start(): %v", err)
+		os.Rename(currLancherExeback, currLancherExe)
+		cb(-1, "新包启动失败!")
 		return
 	}
-	os.WriteFile(path.Join(appExeDir, "pid"), []byte(fmt.Sprintf("%d", cmd.Process.Pid)), fs.FileMode(os.O_TRUNC))
-	log.Printf("pid: %d", cmd.Process.Pid)
-	lancherbak := fmt.Sprintf("%s/%s", appExeDir, "lancherlast")
-	err = os.Remove(lancherbak)
-	if err != nil {
-		fmt.Println(err)
-	}
-	cmd.Process.Wait()
+	cb(1, "")
+
+	time.Sleep(time.Second)
+
+	os.Exit(0)
+}
+
+var installError = ""
+var installSucc = false
+
+func main() {
+	GAppOption.Parse()
+	fmt.Println(GAppOption)
 
 	// // Setup a separate channel to provide ticks to increment progress
-	// progressIncrementer = make(chan float32)
-	// go func() {
-	// 	for {
-	// 		time.Sleep(time.Second / 25)
-	// 		progressIncrementer <- 0.004
-	// 	}
-	// }()
-	// // task := make(map[string]string)
-	// // task["https://dl.softmgr.qq.com/original/game/WeGameSetup3.32.4.6183_gjwegame_0_0.exe"] = "wegame.exe"
-	// // for k, v := range task {
-	// // 	// wg.Add(1)
-	// // 	download.DownloadFile(k, v)
-	// // }
-
-	// go func() {
-	// 	// create new window
-	// 	w := app.NewWindow(
-	// 		app.Title("uploader"),
-	// 		app.Size(unit.Dp(600), unit.Dp(400)),
-	// 	)
-	// 	if err := draw(w); err != nil {
-	// 		log.Fatal(err)
-	// 	}
-	// 	os.Exit(0)
-	// }()
-
-	// app.Main()
+	progressIncrementer = make(chan float32)
+	go func() {
+		for {
+			time.Sleep(time.Second / 25)
+			progressIncrementer <- 0.004
+		}
+	}()
+
+	w := app.NewWindow(
+		app.Title("uploader"),
+		app.Size(unit.Dp(600), unit.Dp(400)),
+	)
+
+	go func() {
+		// create new window
+
+		if err := draw(w); err != nil {
+			log.Fatal(err)
+		}
+		os.Exit(0)
+	}()
+
+	go func() {
+		if len(GAppOption.Url) < 1 {
+			return
+		}
+
+		upgradeLancherExe(GAppOption.Url, func(i int, s string) {
+			fmt.Println("xxx=>", i, s)
+
+			if i == -1 {
+				installError = s
+				w.Invalidate()
+				return
+			}
+			installSucc = true
+			w.Invalidate()
+		})
+	}()
+
+	app.Main()
 }
 
 type C = layout.Context
@@ -189,7 +243,7 @@ func draw(w *app.Window) error {
 				return e.Err
 			}
 
-		// listen for events in the incrementor channel
+		//listen for events in the incrementor channel
 		case p := <-progressIncrementer:
 			if boiling && progress < 1 {
 				progress += p