package shared import ( _ "embed" "bytes" "encoding/json" "fmt" "io" "io/ioutil" "log" "os" "os/exec" "os/user" "path" "path/filepath" "strconv" "strings" "time" "github.com/fatih/color" "github.com/google/uuid" "github.com/rodaine/table" ) const ( CONFIG_PATH = ".hishtory.config" ) var ( //go:embed config.sh CONFIG_SH_CONTENTS string ) func getCwd() (string, error) { cwd, err := os.Getwd() if err != nil { return "", fmt.Errorf("failed to get cwd for last command: %v", err) } homedir, err := os.UserHomeDir() if err != nil { return "", fmt.Errorf("failed to get user's home directory: %v", err) } if cwd == homedir { return "~/", nil } if strings.HasPrefix(cwd, homedir) { return strings.Replace(cwd, homedir, "~", 1), nil } return cwd, nil } func BuildHistoryEntry(args []string) (*HistoryEntry, error) { var entry HistoryEntry // exitCode exitCode, err := strconv.Atoi(args[2]) if err != nil { return nil, fmt.Errorf("failed to build history entry: %v", err) } entry.ExitCode = exitCode // user user, err := user.Current() if err != nil { return nil, fmt.Errorf("failed to build history entry: %v", err) } entry.LocalUsername = user.Username // cwd cwd, err := getCwd() if err != nil { return nil, fmt.Errorf("failed to build history entry: %v", err) } entry.CurrentWorkingDirectory = cwd // start time nanos, err := strconv.ParseInt(args[4], 10, 64) if err != nil { return nil, fmt.Errorf("failed to parse start time %s as int: %v", args[4], err) } entry.StartTime = time.Unix(0, nanos) // end time entry.EndTime = time.Now() // command cmd, err := getLastCommand(args[3]) if err != nil { return nil, fmt.Errorf("failed to build history entry: %v", err) } entry.Command = cmd // hostname hostname, err := os.Hostname() if err != nil { return nil, fmt.Errorf("failed to build history entry: %v", err) } entry.Hostname = hostname // user secret userSecret, err := GetUserSecret() if err != nil { return nil, fmt.Errorf("failed to build history entry: %v", err) } entry.UserSecret = userSecret return &entry, nil } func getLastCommand(history string) (string, error) { return strings.TrimSpace(strings.SplitN(strings.TrimSpace(history), " ", 2)[1]), nil } func GetUserSecret() (string, error) { config, err := GetConfig() if err != nil { return "", err } return config.UserSecret, nil } func Setup(args []string) error { userSecret := uuid.Must(uuid.NewRandom()).String() if len(args) > 2 && args[2] != "" { userSecret = args[2] } fmt.Println("Setting secret hishtory key to " + string(userSecret)) var config ClientConfig config.UserSecret = userSecret config.IsEnabled = true return SetConfig(config) } func DisplayResults(results []*HistoryEntry, displayHostname bool) { headerFmt := color.New(color.FgGreen, color.Underline).SprintfFunc() tbl := table.New("CWD", "Timestamp", "Runtime", "Exit Code", "Command") if displayHostname { tbl = table.New("Hostname", "CWD", "Timestamp", "Runtime", "Exit Code", "Command") } tbl.WithHeaderFormatter(headerFmt) for _, result := range results { timestamp := result.StartTime.Format("Jan 2 2006 15:04:05 MST") duration := result.EndTime.Sub(result.StartTime).Round(time.Millisecond).String() if displayHostname { tbl.AddRow(result.Hostname, result.CurrentWorkingDirectory, timestamp, duration, result.ExitCode, result.Command) } else { tbl.AddRow(result.CurrentWorkingDirectory, timestamp, duration, result.ExitCode, result.Command) } } tbl.Print() } type ClientConfig struct { UserSecret string `json:"user_secret"` IsEnabled bool `json:"is_enabled"` } func GetConfig() (ClientConfig, error) { homedir, err := os.UserHomeDir() if err != nil { return ClientConfig{}, fmt.Errorf("failed to retrieve homedir: %v", err) } data, err := os.ReadFile(path.Join(homedir, CONFIG_PATH)) if err != nil { return ClientConfig{}, fmt.Errorf("failed to read config file: %v", err) } var config ClientConfig err = json.Unmarshal(data, &config) if err != nil { return ClientConfig{}, fmt.Errorf("failed to parse config file: %v", err) } return config, nil } func SetConfig(config ClientConfig) error { serializedConfig, err := json.Marshal(config) if err != nil { return fmt.Errorf("failed to serialize config: %v", err) } homedir, err := os.UserHomeDir() if err != nil { return fmt.Errorf("failed to retrieve homedir: %v", err) } err = os.WriteFile(path.Join(homedir, CONFIG_PATH), serializedConfig, 0600) if err != nil { return fmt.Errorf("failed to write config: %v", err) } return nil } func IsEnabled() (bool, error) { config, err := GetConfig() if err != nil { return false, err } return config.IsEnabled, nil } func Enable() error { config, err := GetConfig() if err != nil { return err } config.IsEnabled = true return SetConfig(config) } func Disable() error { config, err := GetConfig() if err != nil { return err } config.IsEnabled = false return SetConfig(config) } func CheckFatalError(err error) { if err != nil { log.Fatalf("hishtory fatal error: %v", err) } } func Install() error { homedir, err := os.UserHomeDir() if err != nil { return fmt.Errorf("failed to get user's home directory: %v", err) } path, err := installBinary(homedir) if err != nil { return err } err = configureBashrc(homedir, path) if err != nil { return err } _, err = GetConfig() if err != nil { // No config, so set up a new installation return Setup(os.Args) } return nil } func configureBashrc(homedir, binaryPath string) error { // Create the file we're going to source in our bashrc. Do this no matter what in case there are updates to it. bashConfigPath := path.Join(filepath.Dir(binaryPath), "config.sh") err := ioutil.WriteFile(bashConfigPath, []byte(CONFIG_SH_CONTENTS), 0644) if err != nil { return fmt.Errorf("failed to write config.sh file: %v", err) } // Check if we need to configure the bashrc bashrc, err := ioutil.ReadFile(path.Join(homedir, ".bashrc")) if err != nil { return fmt.Errorf("failed to read bashrc: %v", err) } if strings.Contains(string(bashrc), "# Hishtory Config:") { return nil } // Add to bashrc f, err := os.OpenFile(path.Join(homedir, ".bashrc"), os.O_APPEND|os.O_WRONLY, 0644) if err != nil { return fmt.Errorf("failed to append to bashrc: %v", err) } defer f.Close() _, err = f.WriteString("\n# Hishtory Config:\nexport PATH=\"$PATH:" + filepath.Dir(binaryPath) + "\"\nsource " + bashConfigPath + "\n") if err != nil { return fmt.Errorf("failed to append to bashrc: %v", err) } return nil } func installBinary(homedir string) (string, error) { clientPath, err := exec.LookPath("hishtory") if err != nil { clientDir := path.Join(homedir, ".hishtory-bin") err = os.MkdirAll(clientDir, 0744) if err != nil { return "", fmt.Errorf("failed to create folder for hishtory binary: %v", err) } clientPath = path.Join(clientDir, "hishtory") } err = copyFile(os.Args[0], clientPath) if err != nil { return "", fmt.Errorf("failed to copy hishtory binary to $PATH: %v", err) } err = os.Chmod(clientPath, 0700) if err != nil { return "", fmt.Errorf("failed to set permissions on hishtory binary: %v", err) } return clientPath, nil } func copyFile(src, dst string) error { sourceFileStat, err := os.Stat(src) if err != nil { return err } if !sourceFileStat.Mode().IsRegular() { return fmt.Errorf("%s is not a regular file", src) } source, err := os.Open(src) if err != nil { return err } defer source.Close() destination, err := os.Create(dst) if err != nil { return err } defer destination.Close() _, err = io.Copy(destination, source) return err } func Update(url string) error { var stdout bytes.Buffer var stderr bytes.Buffer cmd := exec.Command("bash", "-c", "curl -o /tmp/hishtory-client "+url+"; chmod +x /tmp/hishtory-client; /tmp/hishtory-client install") cmd.Stdout = &stdout cmd.Stderr = &stderr err := cmd.Run() if err != nil { return fmt.Errorf("Failed to update: %v, stdout=%+v, stderr=%+v", err, stdout.String(), stderr.String()) } return nil }