mirror of
https://github.com/ddworken/hishtory.git
synced 2025-01-25 07:38:52 +01:00
6575c8ae42
* Fix handling of new lines in commands for #163 * Move code for table from lib.go to query.go * Update goldens
225 lines
6.6 KiB
Go
225 lines
6.6 KiB
Go
package cmd
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"math/rand"
|
|
"os"
|
|
"strings"
|
|
|
|
"github.com/ddworken/hishtory/client/data"
|
|
"github.com/ddworken/hishtory/client/hctx"
|
|
"github.com/ddworken/hishtory/client/lib"
|
|
"github.com/ddworken/hishtory/client/tui"
|
|
"github.com/fatih/color"
|
|
"github.com/muesli/termenv"
|
|
"github.com/rodaine/table"
|
|
"github.com/spf13/cobra"
|
|
)
|
|
|
|
var EXAMPLE_QUERIES string = `Example queries:
|
|
'hishtory SUBCOMMAND apt-get' # Find shell commands containing 'apt-get'
|
|
'hishtory SUBCOMMAND apt-get install' # Find shell commands containing 'apt-get' and 'install'
|
|
'hishtory SUBCOMMAND curl cwd:/tmp/' # Find shell commands containing 'curl' run in '/tmp/'
|
|
'hishtory SUBCOMMAND curl user:david' # Find shell commands containing 'curl' run by 'david'
|
|
'hishtory SUBCOMMAND curl host:x1' # Find shell commands containing 'curl' run on 'x1'
|
|
'hishtory SUBCOMMAND exit_code:1' # Find shell commands that exited with status code 1
|
|
'hishtory SUBCOMMAND before:2022-02-01' # Find shell commands run before 2022-02-01
|
|
`
|
|
|
|
var GROUP_ID_QUERYING string = "group_id:querying"
|
|
|
|
var queryCmd = &cobra.Command{
|
|
Use: "query",
|
|
Short: "Query your shell history and display the results in an ASCII art table",
|
|
GroupID: GROUP_ID_QUERYING,
|
|
Long: strings.ReplaceAll(EXAMPLE_QUERIES, "SUBCOMMAND", "query"),
|
|
DisableFlagParsing: true,
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
ctx := hctx.MakeContext()
|
|
lib.CheckFatalError(lib.ProcessDeletionRequests(ctx))
|
|
query(ctx, strings.Join(args, " "))
|
|
},
|
|
}
|
|
|
|
var tqueryCmd = &cobra.Command{
|
|
Use: "tquery",
|
|
Short: "Interactively query your shell history in a TUI interface",
|
|
GroupID: GROUP_ID_QUERYING,
|
|
Long: strings.ReplaceAll(EXAMPLE_QUERIES, "SUBCOMMAND", "tquery"),
|
|
DisableFlagParsing: true,
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
ctx := hctx.MakeContext()
|
|
lib.CheckFatalError(tui.TuiQuery(ctx, strings.Join(args, " ")))
|
|
},
|
|
}
|
|
|
|
var exportCmd = &cobra.Command{
|
|
Use: "export",
|
|
Short: "Export your shell history and display just the raw commands",
|
|
GroupID: GROUP_ID_QUERYING,
|
|
Long: strings.ReplaceAll(EXAMPLE_QUERIES, "SUBCOMMAND", "export"),
|
|
DisableFlagParsing: true,
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
ctx := hctx.MakeContext()
|
|
lib.CheckFatalError(lib.ProcessDeletionRequests(ctx))
|
|
export(ctx, strings.Join(args, " "))
|
|
},
|
|
}
|
|
|
|
var getColorSupportCmd = &cobra.Command{
|
|
Use: "getColorSupport",
|
|
Hidden: true,
|
|
Short: "[Internal-only] Get the supported color format (returned as an exit code)",
|
|
GroupID: GROUP_ID_QUERYING,
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
switch termenv.ColorProfile() {
|
|
case termenv.TrueColor:
|
|
os.Exit(1)
|
|
case termenv.ANSI256:
|
|
os.Exit(2)
|
|
case termenv.ANSI:
|
|
os.Exit(3)
|
|
case termenv.Ascii:
|
|
os.Exit(4)
|
|
default:
|
|
os.Exit(0)
|
|
}
|
|
},
|
|
}
|
|
|
|
var updateLocalDbFromRemoteCmd = &cobra.Command{
|
|
Use: "updateLocalDbFromRemote",
|
|
Hidden: true,
|
|
Short: "[Internal-only] Update local DB from remote",
|
|
GroupID: GROUP_ID_QUERYING,
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
// Periodically, run a query so as to ensure that the local DB stays mostly up to date and that we don't
|
|
// accumulate a large number of entries that this device doesn't know about. This ensures that queries
|
|
// are always reasonably complete and fast (even when offline).
|
|
ctx := hctx.MakeContext()
|
|
config := hctx.GetConf(ctx)
|
|
if config.IsOffline {
|
|
return
|
|
}
|
|
// Do it a random percent of the time, which should be approximately often enough.
|
|
if rand.Intn(20) == 0 && !config.IsOffline {
|
|
err := lib.RetrieveAdditionalEntriesFromRemote(ctx, "preload")
|
|
if config.BetaMode {
|
|
lib.CheckFatalError(err)
|
|
} else if err != nil {
|
|
hctx.GetLogger().Infof("updateLocalDbFromRemote: Failed to RetrieveAdditionalEntriesFromRemote: %v", err)
|
|
}
|
|
err = lib.ProcessDeletionRequests(ctx)
|
|
if config.BetaMode {
|
|
lib.CheckFatalError(err)
|
|
} else if err != nil {
|
|
hctx.GetLogger().Infof("updateLocalDbFromRemote: Failed to ProcessDeletionRequests: %v", err)
|
|
}
|
|
}
|
|
},
|
|
}
|
|
|
|
func export(ctx context.Context, query string) {
|
|
db := hctx.GetDb(ctx)
|
|
err := lib.RetrieveAdditionalEntriesFromRemote(ctx, "export")
|
|
if err != nil {
|
|
if lib.IsOfflineError(ctx, err) {
|
|
fmt.Println("Warning: hishtory is offline so this may be missing recent results from your other machines!")
|
|
} else {
|
|
lib.CheckFatalError(err)
|
|
}
|
|
}
|
|
data, err := lib.Search(ctx, db, query, 0)
|
|
lib.CheckFatalError(err)
|
|
for i := len(data) - 1; i >= 0; i-- {
|
|
fmt.Println(data[i].Command)
|
|
}
|
|
}
|
|
|
|
func query(ctx context.Context, query string) {
|
|
db := hctx.GetDb(ctx)
|
|
err := lib.RetrieveAdditionalEntriesFromRemote(ctx, "query")
|
|
if err != nil {
|
|
if lib.IsOfflineError(ctx, err) {
|
|
fmt.Println("Warning: hishtory is offline so this may be missing recent results from your other machines!")
|
|
} else {
|
|
lib.CheckFatalError(err)
|
|
}
|
|
}
|
|
lib.CheckFatalError(displayBannerIfSet(ctx))
|
|
numResults := 25
|
|
data, err := lib.Search(ctx, db, query, numResults*5)
|
|
lib.CheckFatalError(err)
|
|
lib.CheckFatalError(DisplayResults(ctx, data, numResults))
|
|
}
|
|
|
|
func DisplayResults(ctx context.Context, results []*data.HistoryEntry, numResults int) error {
|
|
config := hctx.GetConf(ctx)
|
|
headerFmt := color.New(color.FgGreen, color.Underline).SprintfFunc()
|
|
|
|
columns := make([]any, 0)
|
|
for _, c := range config.DisplayedColumns {
|
|
columns = append(columns, c)
|
|
}
|
|
tbl := table.New(columns...)
|
|
tbl.WithHeaderFormatter(headerFmt)
|
|
|
|
numRows := 0
|
|
|
|
var seenCommands = make(map[string]bool)
|
|
|
|
for _, entry := range results {
|
|
if config.FilterDuplicateCommands && entry != nil {
|
|
cmd := strings.TrimSpace(entry.Command)
|
|
if seenCommands[cmd] {
|
|
continue
|
|
}
|
|
seenCommands[cmd] = true
|
|
}
|
|
|
|
row, err := lib.BuildTableRow(ctx, config.DisplayedColumns, *entry, func(s string) string { return s })
|
|
if err != nil {
|
|
return err
|
|
}
|
|
tbl.AddRow(stringArrayToAnyArray(row)...)
|
|
numRows += 1
|
|
if numRows >= numResults {
|
|
break
|
|
}
|
|
}
|
|
|
|
tbl.Print()
|
|
return nil
|
|
}
|
|
|
|
func stringArrayToAnyArray(arr []string) []any {
|
|
ret := make([]any, 0)
|
|
for _, item := range arr {
|
|
ret = append(ret, item)
|
|
}
|
|
return ret
|
|
}
|
|
|
|
func displayBannerIfSet(ctx context.Context) error {
|
|
respBody, err := lib.GetBanner(ctx)
|
|
if lib.IsOfflineError(ctx, err) {
|
|
return nil
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if len(respBody) > 0 {
|
|
fmt.Println(string(respBody))
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func init() {
|
|
rootCmd.AddCommand(queryCmd)
|
|
rootCmd.AddCommand(tqueryCmd)
|
|
rootCmd.AddCommand(exportCmd)
|
|
rootCmd.AddCommand(updateLocalDbFromRemoteCmd)
|
|
rootCmd.AddCommand(getColorSupportCmd)
|
|
}
|