package main import ( "context" "errors" "fmt" "os" "runtime" "github.com/charmbracelet/fang" "github.com/charmbracelet/lipgloss/v2" "github.com/spf13/cobra" "github.com/taciturnaxolotl/akami/wakatime" "gopkg.in/ini.v1" ) func main() { // init our cobra command with a name and description cmd := &cobra.Command{ Use: "akami", Short: "🌷 the cutsie hackatime helper", } // add our lipgloss styles fancy := lipgloss.NewStyle().Foreground(lipgloss.Magenta).Bold(true).Italic(true) muted := lipgloss.NewStyle().Foreground(lipgloss.BrightBlue).Italic(true) bad := lipgloss.NewStyle().Foreground(lipgloss.BrightRed).Bold(true) // root diagnose command cmd.AddCommand(&cobra.Command{ Use: "doc", Short: "diagnose potential hackatime issues", RunE: func(c *cobra.Command, _ []string) error { // check our os os_name := runtime.GOOS user_dir, err := os.UserHomeDir() if err != nil { return errors.New("somehow your user doesn't exist? fairly sure this should never happen; plz report this to @krn on slack or via email at me@dunkirk.sh") } hackatime_path := user_dir + "/.wakatime.cfg" switch os_name { case "linux": case "darwin": case "windows": default: return errors.New("hmm you don't seem to be running a recognized os? you are listed as running " + fancy.Render(os_name) + "; can you plz report this to @krn on slack or via email at me@dunkirk.sh?") } c.Println("Looks like you are running", fancy.Render(os_name), "so lets take a look at", muted.Render(hackatime_path), "for your config") rawCfg, err := os.ReadFile(hackatime_path) if errors.Is(err, os.ErrNotExist) { return errors.New("you don't have a wakatime config file! go check https://hackatime.hackclub.com/my/wakatime_setup for the instructions and then try this again") } cfg, err := ini.Load(rawCfg) if err != nil { return errors.New(err.Error()) } settings, err := cfg.GetSection("settings") if err != nil { return errors.New("wow! your config file seems to be messed up and doesn't have a settings heading; can you follow the instructions at https://hackatime.hackclub.com/my/wakatime_setup to regenerate it?\n\nThe raw error we got was: " + err.Error()) } api_key := settings.Key("api_key").String() api_url := settings.Key("api_url").String() if api_key == "" { return errors.New("hmm 🤔 looks like you don't have an api_key in your config file? are you sure you have followed the setup instructions at https://hackatime.hackclub.com/my/wakatime_setup correctly?") } if api_url == "" { return errors.New("hmm 🤔 looks like you don't have an api_url in your config file? are you sure you have followed the setup instructions at https://hackatime.hackclub.com/my/wakatime_setup correctly?") } if api_url != "https://hackatime.hackclub.com/api/hackatime/v1" { if api_url == "https://api.wakatime.com/api/v1" { client := wakatime.NewClient(api_key) _, err := client.GetStatusBar() if !errors.Is(err, wakatime.ErrUnauthorized) { return errors.New("turns out you were connected to wakatime.com instead of hackatime; since your key seems to work if you would like to keep syncing data to wakatime.com as well as to hackatime you can either setup a realy serve like " + muted.Render("https://github.com/JasonLovesDoggo/multitime") + " or you can wait for https://github.com/hackclub/hackatime/issues/85 to get merged in hackatime and have it synced there :)\n\nIf you want to import your wakatime.com data into hackatime then you can use hackatime v1 temporarily to connect your wakatime account and import (in settings under integrations at https://waka.hackclub.com) and then click the import from hackatime v1 button at https://hackatime.hackclub.com/my/settings.\n\n If you have more questions feel free to reach out to me (hackatime v1 creator) on slack (at @krn) or via email at me@dunkirk.sh") } else { return errors.New("turns out your config is connected to the wrong api url and is trying to use wakatime.com to sync time but you don't have a working api key from them. Go to https://hackatime.hackclub.com/my/wakatime_setup to run the setup script and fix your config file") } } c.Println("\nYour api url", muted.Render(api_url), "doesn't match the expected url of", muted.Render("https://hackatime.hackclub.com/api/hackatime/v1"), "however if you are using a custom forwarder or are sure you know what you are doing then you are probably fine") } client := wakatime.NewClientWithOptions(api_key, api_url) duration, err := client.GetStatusBar() if err != nil { if errors.Is(err, wakatime.ErrUnauthorized) { return errors.New("Your config file looks mostly correct and you have the correct api url but when we tested your api_key it looks like it is invalid? Can you double check if the key in your config file is the same as at https://hackatime.hackclub.com/my/wakatime_setup?") } return errors.New("Something weird happened with the hackatime api; if the error doesn't make sense then please contact @krn on slack or via email at me@dunkirk.sh\n\n" + bad.Render("Full error: "+err.Error())) } // Convert seconds to a formatted time string (hours, minutes, seconds) totalSeconds := duration.Data.GrandTotal.TotalSeconds hours := totalSeconds / 3600 minutes := (totalSeconds % 3600) / 60 seconds := totalSeconds % 60 formattedTime := "" if hours > 0 { formattedTime += fmt.Sprintf("%d hours, ", hours) } if minutes > 0 || hours > 0 { formattedTime += fmt.Sprintf("%d minutes, ", minutes) } formattedTime += fmt.Sprintf("%d seconds", seconds) c.Println("\nSweet!!! Looks like your hackatime is configured properly! Looks like you have coded today for", fancy.Render(formattedTime)) return nil }, }) // this is where we get the fancy fang magic ✨ if err := fang.Execute( context.Background(), cmd, ); err != nil { os.Exit(1) } }