dnote/pkg/cli/upgrade/upgrade.go
2025-10-31 23:41:21 -07:00

145 lines
3.5 KiB
Go

/* Copyright 2025 Dnote Authors
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package upgrade
import (
stdCtx "context"
"fmt"
"strings"
"time"
"github.com/dnote/dnote/pkg/cli/consts"
"github.com/dnote/dnote/pkg/cli/context"
"github.com/dnote/dnote/pkg/cli/log"
"github.com/dnote/dnote/pkg/cli/ui"
"github.com/google/go-github/github"
"github.com/pkg/errors"
)
// upgradeInterval is 8 weeks
var upgradeInterval int64 = 86400 * 7 * 8
// shouldCheckUpdate checks if update should be checked
func shouldCheckUpdate(ctx context.DnoteCtx) (bool, error) {
db := ctx.DB
var lastUpgrade int64
err := db.QueryRow("SELECT value FROM system WHERE key = ?", consts.SystemLastUpgrade).Scan(&lastUpgrade)
if err != nil {
return false, errors.Wrap(err, "getting last_udpate")
}
now := time.Now().Unix()
return now-lastUpgrade > upgradeInterval, nil
}
func touchLastUpgrade(ctx context.DnoteCtx) error {
db := ctx.DB
now := time.Now().Unix()
_, err := db.Exec("UPDATE system SET value = ? WHERE key = ?", now, consts.SystemLastUpgrade)
if err != nil {
return errors.Wrap(err, "updating last_upgrade")
}
return nil
}
func fetchLatestStableTag(gh *github.Client, page int) (string, error) {
params := github.ListOptions{
Page: page,
}
releases, resp, err := gh.Repositories.ListReleases(stdCtx.Background(), "dnote", "dnote", &params)
if err != nil {
return "", errors.Wrapf(err, "fetching releases page %d", page)
}
for _, release := range releases {
tag := release.GetTagName()
isStable := !release.GetPrerelease()
if strings.HasPrefix(tag, "cli-") && isStable {
return tag, nil
}
}
if page == resp.LastPage {
return "", errors.New("No CLI release was found")
}
return fetchLatestStableTag(gh, page+1)
}
func checkVersion(ctx context.DnoteCtx) error {
log.Infof("current version is %s\n", ctx.Version)
// Fetch the latest version
gh := github.NewClient(nil)
latestTag, err := fetchLatestStableTag(gh, 1)
if err != nil {
return errors.Wrap(err, "fetching the latest stable release")
}
// releases are tagged in a form of cli-v1.0.0
latestVersion := latestTag[5:]
log.Infof("latest version is %s\n", latestVersion)
if latestVersion == ctx.Version {
log.Success("you are up-to-date\n\n")
} else {
log.Infof("to upgrade, see https://github.com/dnote/dnote\n")
}
return nil
}
// Check triggers update if needed
func Check(ctx context.DnoteCtx) error {
// If upgrade check is not enabled, do not proceed further
if !ctx.EnableUpgradeCheck {
return nil
}
shouldCheck, err := shouldCheckUpdate(ctx)
if err != nil {
return errors.Wrap(err, "checking if dnote should check update")
}
if !shouldCheck {
return nil
}
err = touchLastUpgrade(ctx)
if err != nil {
return errors.Wrap(err, "updating the last upgrade timestamp")
}
fmt.Printf("\n")
willCheck, err := ui.Confirm("check for upgrade?", true)
if err != nil {
return errors.Wrap(err, "getting user confirmation")
}
if !willCheck {
return nil
}
err = checkVersion(ctx)
if err != nil {
return errors.Wrap(err, "checking version")
}
return nil
}