| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118 |
- package update
- import (
- "context"
- "encoding/json"
- "fmt"
- "io"
- "net/http"
- "regexp"
- "strings"
- "time"
- )
- const (
- githubApiUrl = "https://api.github.com/repos/charmbracelet/crush/releases/latest"
- userAgent = "crush/1.0"
- )
- // Default is the default [Client].
- var Default Client = &github{}
- // Info contains information about an available update.
- type Info struct {
- Current string
- Latest string
- URL string
- }
- // Matches a version string like:
- // v0.0.0-0.20251231235959-06c807842604
- var goInstallRegexp = regexp.MustCompile(`^v?\d+\.\d+\.\d+-\d+\.\d{14}-[0-9a-f]{12}$`)
- func (i Info) IsDevelopment() bool {
- return i.Current == "devel" || i.Current == "unknown" || strings.Contains(i.Current, "dirty") || goInstallRegexp.MatchString(i.Current)
- }
- // Available returns true if there's an update available.
- //
- // If both current and latest are stable versions, returns true if versions are
- // different.
- // If current is a pre-release and latest isn't, returns true.
- // If latest is a pre-release and current isn't, returns false.
- func (i Info) Available() bool {
- cpr := strings.Contains(i.Current, "-")
- lpr := strings.Contains(i.Latest, "-")
- // current is pre release && latest isn't a prerelease
- if cpr && !lpr {
- return true
- }
- // latest is pre release && current isn't a prerelease
- if lpr && !cpr {
- return false
- }
- return i.Current != i.Latest
- }
- // Check checks if a new version is available.
- func Check(ctx context.Context, current string, client Client) (Info, error) {
- info := Info{
- Current: current,
- Latest: current,
- }
- release, err := client.Latest(ctx)
- if err != nil {
- return info, fmt.Errorf("failed to fetch latest release: %w", err)
- }
- info.Latest = strings.TrimPrefix(release.TagName, "v")
- info.Current = strings.TrimPrefix(info.Current, "v")
- info.URL = release.HTMLURL
- return info, nil
- }
- // Release represents a GitHub release.
- type Release struct {
- TagName string `json:"tag_name"`
- HTMLURL string `json:"html_url"`
- }
- // Client is a client that can get the latest release.
- type Client interface {
- Latest(ctx context.Context) (*Release, error)
- }
- type github struct{}
- // Latest implements [Client].
- func (c *github) Latest(ctx context.Context) (*Release, error) {
- client := &http.Client{
- Timeout: 30 * time.Second,
- }
- req, err := http.NewRequestWithContext(ctx, "GET", githubApiUrl, nil)
- if err != nil {
- return nil, err
- }
- req.Header.Set("User-Agent", userAgent)
- req.Header.Set("Accept", "application/vnd.github.v3+json")
- resp, err := client.Do(req)
- if err != nil {
- return nil, err
- }
- defer resp.Body.Close()
- if resp.StatusCode != http.StatusOK {
- body, _ := io.ReadAll(resp.Body)
- return nil, fmt.Errorf("GitHub API returned status %d: %s", resp.StatusCode, string(body))
- }
- var release Release
- if err := json.NewDecoder(resp.Body).Decode(&release); err != nil {
- return nil, err
- }
- return &release, nil
- }
|