git-project-manager/cmd/util/util_fzf.go

187 lines
5.5 KiB
Go
Raw Normal View History

2024-12-30 19:54:32 +00:00
package util
2023-12-08 21:52:26 +00:00
import (
"context"
fzf "github.com/ktr0731/go-fuzzyfinder"
2024-10-01 18:29:14 +00:00
"golang.org/x/exp/slices"
2024-12-19 19:55:49 +00:00
"gitea.libretechconsulting.com/rmcguire/git-project-manager/internal/cache"
"gitea.libretechconsulting.com/rmcguire/git-project-manager/internal/remotes/projects"
2023-12-08 21:52:26 +00:00
)
2024-12-30 19:54:32 +00:00
type FzfProjectOpts struct {
Ctx context.Context
Search string
MustHaveAlias bool
Remotes []string
}
2023-12-10 16:15:52 +00:00
// This will try to find a project by alias if a search term
// is given, otherwise will fuzzy find by project
2024-12-30 19:54:32 +00:00
func (u *Utils) FzfFindProject(opts *FzfProjectOpts) *projects.Project {
2024-01-15 20:39:35 +00:00
var project *projects.Project
2023-12-08 21:52:26 +00:00
if opts.Search != "" {
2024-12-30 19:54:32 +00:00
project = u.FzfSearchProjectAliases(opts)
2023-12-08 21:52:26 +00:00
} else {
var err error
2024-12-30 19:54:32 +00:00
project, err = u.FzfProject(opts)
2023-12-08 21:52:26 +00:00
if project == nil || err != nil {
return nil
}
}
return project
}
// If . is given as a project, will open project from the
// current working directory. Otherwise, will attempt to fuzzy-find
// a project given a search term if provided
2024-12-30 19:54:32 +00:00
func (u *Utils) FzfCwdOrSearchProjectAliases(opts *FzfProjectOpts) *projects.Project {
2024-01-15 20:39:35 +00:00
var project *projects.Project
if opts.Search == "." {
2024-12-30 19:54:32 +00:00
project, _ = u.Cache().GetProjectFromCwd()
} else {
2024-12-30 19:54:32 +00:00
project = u.FzfSearchProjectAliases(opts)
}
return project
}
2023-12-10 16:15:52 +00:00
// This will fuzzy search only aliases, preferring an exact
// match if one is given
2024-12-30 19:54:32 +00:00
func (u *Utils) FzfSearchProjectAliases(opts *FzfProjectOpts) *projects.Project {
2024-01-15 20:39:35 +00:00
var project *projects.Project
2024-01-15 21:02:15 +00:00
var alias *cache.ProjectAlias
2024-12-30 19:54:32 +00:00
if alias = u.Cache().GetAliasByName(opts.Search, opts.Remotes...); alias != nil {
project = u.Cache().GetProjectByAlias(alias)
u.Logger().Info("Perfect alias match... flawless")
2023-12-08 21:52:26 +00:00
} else {
// Get fuzzy if we don't have an exact match
2024-12-30 19:54:32 +00:00
aliases := u.Cache().FuzzyFindAlias(opts.Search)
2023-12-08 21:52:26 +00:00
if len(aliases) > 1 {
2023-12-10 16:15:52 +00:00
// If multiple aliases were found, switch over to project
// by alias mode with merging
// alias = fzfAliasFromAliases(rootCmd.Context(), aliases)
2024-12-30 19:54:32 +00:00
project, _ = u.FzfProjectFromAliases(opts, aliases)
2023-12-08 21:52:26 +00:00
} else if len(aliases) == 1 {
alias = aliases[0]
2024-12-30 19:54:32 +00:00
project = u.Cache().GetProjectByAlias(alias)
2023-12-08 21:52:26 +00:00
}
}
return project
}
2023-12-10 16:15:52 +00:00
// Given a list of aliases, will fuzzy-find and return
// a single one. Replaced by fzfProjectFromAliases in fzfSearchProjectAliases
// as merging is preferred, but can be used if it's ever desirable to
// return a single alias from all aliases
2024-12-30 19:54:32 +00:00
func (u *Utils) FzfAliasFromAliases(opts *FzfProjectOpts, aliases []*cache.ProjectAlias) *cache.ProjectAlias {
2024-01-15 21:02:15 +00:00
var alias *cache.ProjectAlias
2023-12-08 21:52:26 +00:00
i, err := fzf.Find(
aliases,
func(i int) string {
2024-12-30 19:54:32 +00:00
return aliases[i].Alias + " -> " + u.Cache().GetProjectByAlias(aliases[i]).PathWithNamespace
2023-12-08 21:52:26 +00:00
},
fzf.WithContext(opts.Ctx),
2023-12-08 21:52:26 +00:00
fzf.WithHeader("Choose an Alias"),
)
if err != nil {
2024-12-30 19:54:32 +00:00
u.Logger().Error("Failed to fzf alias slice", u.Logger().Args("error", err))
2023-12-08 21:52:26 +00:00
} else {
alias = aliases[i]
}
return alias
}
2023-12-10 16:15:52 +00:00
// Given a list of aliases, merge them together and use the resulting
// list of projects to return a project
2024-12-30 19:54:32 +00:00
func (u *Utils) FzfProjectFromAliases(opts *FzfProjectOpts, aliases []*cache.ProjectAlias) (
2024-10-01 18:29:14 +00:00
*projects.Project, error,
) {
2024-12-30 19:54:32 +00:00
mergedProjects := u.projectsFromAliases(aliases)
if len(mergedProjects) == 1 {
return mergedProjects[0], nil
}
2024-12-30 19:54:32 +00:00
return u.FzfProjectFromProjects(opts, mergedProjects)
2023-12-10 16:15:52 +00:00
}
2024-12-30 19:54:32 +00:00
func (u *Utils) projectsFromAliases(aliases []*cache.ProjectAlias) []*projects.Project {
2024-10-01 18:29:14 +00:00
projects := make([]*projects.Project, 0, len(aliases))
2023-12-10 16:15:52 +00:00
for _, a := range aliases {
2024-12-30 19:54:32 +00:00
project := u.Cache().GetProjectByAlias(a)
2024-10-01 18:29:14 +00:00
if project != nil && !slices.Contains(projects, project) {
projects = append(projects, project)
2023-12-10 16:15:52 +00:00
}
}
2024-10-01 18:29:14 +00:00
return slices.Clip(projects)
2023-12-10 16:15:52 +00:00
}
// If opts.MustHaveAlias, will only allow selection of projects
2023-12-09 04:13:17 +00:00
// that have at least one alias defined
2024-12-30 19:54:32 +00:00
func (u *Utils) FzfProject(opts *FzfProjectOpts) (*projects.Project, error) {
2024-01-15 20:39:35 +00:00
var searchableProjects []*projects.Project
if opts.MustHaveAlias {
2024-12-30 19:54:32 +00:00
searchableProjects = u.Cache().GetProjectsWithAliases()
2023-12-09 04:13:17 +00:00
} else {
2024-12-30 19:54:32 +00:00
searchableProjects = u.Cache().Projects
2023-12-09 04:13:17 +00:00
}
// Filter out unwanted remotes if provided
2024-12-30 19:54:32 +00:00
searchableProjects = u.FilterProjectsWithRemotes(searchableProjects, opts.Remotes...)
return u.FzfProjectFromProjects(opts, searchableProjects)
2023-12-10 16:15:52 +00:00
}
2023-12-09 04:13:17 +00:00
2023-12-10 16:15:52 +00:00
// Takes a list of projects and performs a fuzzyfind
2024-12-30 19:54:32 +00:00
func (u *Utils) FzfProjectFromProjects(opts *FzfProjectOpts, projects []*projects.Project) (
2024-10-01 18:29:14 +00:00
*projects.Project, error,
) {
2023-12-10 16:15:52 +00:00
i, err := fzf.Find(projects,
2023-12-09 04:13:17 +00:00
func(i int) string {
2023-12-10 16:15:52 +00:00
// Display the project along with its aliases
2024-12-30 19:54:32 +00:00
return u.Cache().GetProjectStringWithAliases(projects[i])
2023-12-09 04:13:17 +00:00
},
fzf.WithPreviewWindow(
func(i, width, height int) string {
2024-12-30 19:54:32 +00:00
return u.Cache().ProjectString(projects[i])
2023-12-09 04:13:17 +00:00
},
),
fzf.WithContext(opts.Ctx),
2023-12-08 21:52:26 +00:00
fzf.WithHeader("Fuzzy find yourself a project"),
)
if err != nil || i < 0 {
return nil, err
}
2023-12-10 16:15:52 +00:00
return projects[i], nil
2023-12-08 21:52:26 +00:00
}
2024-12-30 19:54:32 +00:00
func (u *Utils) FzfPreviewWindow(i, _, _ int) string {
p := u.Cache().Projects[i]
return u.Cache().ProjectString(p)
2023-12-08 21:52:26 +00:00
}
2024-12-30 19:54:32 +00:00
func (u *Utils) FilterProjectsWithRemotes(gitProjects []*projects.Project, remotes ...string) []*projects.Project {
2024-01-15 20:39:35 +00:00
filteredProjects := make([]*projects.Project, 0, len(gitProjects))
if len(remotes) > 0 {
2024-01-15 20:39:35 +00:00
for _, p := range gitProjects {
if slices.Contains(remotes, p.Remote) {
filteredProjects = append(filteredProjects, p)
}
}
} else {
2024-01-15 20:39:35 +00:00
filteredProjects = gitProjects
}
return filteredProjects
}
// Nearly useless function that simply returns either an
// empty string, or a string from the first arg if one is provided
2024-12-30 19:54:32 +00:00
func (u *Utils) SearchStringFromArgs(args []string) string {
var term string
if len(args) > 0 {
term = args[0]
}
return term
}