/* * itd uses bluetooth low energy to communicate with InfiniTime devices * Copyright (C) 2021 Arsen Musayelyan * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see <http://www.gnu.org/licenses/>. */ package firmware import ( "fmt" "time" "github.com/cheggaaa/pb/v3" "github.com/rs/zerolog/log" "github.com/spf13/cobra" "github.com/spf13/viper" "go.arsenm.dev/itd/api" "go.arsenm.dev/itd/internal/types" ) type DFUProgress struct { Received int64 `mapstructure:"recvd"` Total int64 `mapstructure:"total"` } // upgradeCmd represents the upgrade command var upgradeCmd = &cobra.Command{ Use: "upgrade", Short: "Upgrade InfiniTime firmware using files or archive", Aliases: []string{"upg"}, Run: func(cmd *cobra.Command, args []string) { start := time.Now() client := viper.Get("client").(*api.Client) var upgType api.UpgradeType var files []string // Get relevant data struct if viper.GetString("archive") != "" { // Get archive data struct upgType = types.UpgradeTypeArchive files = []string{viper.GetString("archive")} } else if viper.GetString("initPkt") != "" && viper.GetString("firmware") != "" { // Get files data struct upgType = types.UpgradeTypeFiles files = []string{viper.GetString("initPkt"), viper.GetString("firmware")} } else { cmd.Usage() log.Warn().Msg("Upgrade command requires either archive or init packet and firmware.") return } progress, err := client.FirmwareUpgrade(upgType, files...) if err != nil { log.Fatal().Err(err).Msg("Error initiating DFU") } // Create progress bar template barTmpl := `{{counters . }} B {{bar . "|" "-" (cycle .) " " "|"}} {{percent . }} {{rtime . "%s"}}` // Start full bar at 0 total bar := pb.ProgressBarTemplate(barTmpl).Start(0) // Create new scanner of connection for event := range progress { // Set total bytes in progress bar bar.SetTotal(event.Total) // Set amount of bytes received in progress bar bar.SetCurrent(event.Received) // If transfer finished, break if event.Sent == event.Total { break } } // Finish progress bar bar.Finish() fmt.Printf("Transferred %d B in %s.\n", bar.Total(), time.Since(start)) fmt.Println("Remember to validate the new firmware in the InfiniTime settings.") }, } func init() { firmwareCmd.AddCommand(upgradeCmd) // Register flags upgradeCmd.Flags().StringP("archive", "a", "", "Path to firmware archive") upgradeCmd.Flags().StringP("init-pkt", "i", "", "Path to init packet (.dat file)") upgradeCmd.Flags().StringP("firmware", "f", "", "Path to firmware image (.bin file)") // Bind flags to viper keys viper.BindPFlag("archive", upgradeCmd.Flags().Lookup("archive")) viper.BindPFlag("initPkt", upgradeCmd.Flags().Lookup("init-pkt")) viper.BindPFlag("firmware", upgradeCmd.Flags().Lookup("firmware")) }