mirror of
https://github.com/mickael-menu/zk
synced 2024-11-11 07:10:25 +00:00
194 lines
6.4 KiB
Go
194 lines
6.4 KiB
Go
package cmd
|
|
|
|
import (
|
|
"strconv"
|
|
"time"
|
|
|
|
"github.com/mickael-menu/zk/core/note"
|
|
"github.com/mickael-menu/zk/core/zk"
|
|
"github.com/mickael-menu/zk/util/opt"
|
|
"github.com/tj/go-naturaldate"
|
|
)
|
|
|
|
// Filtering holds filtering options to select notes.
|
|
type Filtering struct {
|
|
Path []string `group:filter arg optional placeholder:PATH help:"Find notes matching the given path, including its descendants."`
|
|
|
|
Interactive bool `group:filter short:i help:"Select notes interactively with fzf."`
|
|
Limit int `group:filter short:n placeholder:COUNT help:"Limit the number of notes found."`
|
|
Match string `group:filter short:m placeholder:QUERY help:"Terms to search for in the notes."`
|
|
Exclude []string `group:filter short:x placeholder:PATH help:"Ignore notes matching the given path, including its descendants."`
|
|
Tag []string `group:filter short:t help:"Find notes tagged with the given tags."`
|
|
Mention []string `group:filter placeholder:PATH help:"Find notes mentioning the title of the given ones."`
|
|
LinkedBy []string `group:filter short:l placeholder:PATH help:"Find notes which are linked by the given ones." xor:link`
|
|
NoLinkedBy []string `group:filter placeholder:PATH help:"Find notes which are not linked by the given ones." xor:link`
|
|
LinkTo []string `group:filter short:L placeholder:PATH help:"Find notes which are linking to the given ones." xor:link`
|
|
NoLinkTo []string `group:filter placeholder:PATH help:"Find notes which are not linking to the given notes." xor:link`
|
|
Orphan bool `group:filter help:"Find notes which are not linked by any other note." xor:link`
|
|
Related []string `group:filter placeholder:PATH help:"Find notes which might be related to the given ones." xor:link`
|
|
MaxDistance int `group:filter placeholder:COUNT help:"Maximum distance between two linked notes."`
|
|
Recursive bool `group:filter short:r help:"Follow links recursively."`
|
|
Created string `group:filter placeholder:DATE help:"Find notes created on the given date."`
|
|
CreatedBefore string `group:filter placeholder:DATE help:"Find notes created before the given date."`
|
|
CreatedAfter string `group:filter placeholder:DATE help:"Find notes created after the given date."`
|
|
Modified string `group:filter placeholder:DATE help:"Find notes modified on the given date."`
|
|
ModifiedBefore string `group:filter placeholder:DATE help:"Find notes modified before the given date."`
|
|
ModifiedAfter string `group:filter placeholder:DATE help:"Find notes modified after the given date."`
|
|
}
|
|
|
|
// Sorting holds sorting options to order notes.
|
|
type Sorting struct {
|
|
Sort []string `group:sort short:s placeholder:TERM help:"Order the notes by the given criterion."`
|
|
}
|
|
|
|
// NewFinderOpts creates an instance of note.FinderOpts from a set of user flags.
|
|
func NewFinderOpts(zk *zk.Zk, filtering Filtering, sorting Sorting) (*note.FinderOpts, error) {
|
|
opts := note.FinderOpts{}
|
|
|
|
opts.Match = opt.NewNotEmptyString(filtering.Match)
|
|
|
|
if paths, ok := relPaths(zk, filtering.Path); ok {
|
|
opts.IncludePaths = paths
|
|
}
|
|
|
|
if paths, ok := relPaths(zk, filtering.Exclude); ok {
|
|
opts.ExcludePaths = paths
|
|
}
|
|
|
|
if len(filtering.Tag) > 0 {
|
|
opts.Tags = filtering.Tag
|
|
}
|
|
|
|
if len(filtering.Mention) > 0 {
|
|
opts.Mention = filtering.Mention
|
|
}
|
|
|
|
if paths, ok := relPaths(zk, filtering.LinkedBy); ok {
|
|
opts.LinkedBy = ¬e.LinkedByFilter{
|
|
Paths: paths,
|
|
Negate: false,
|
|
Recursive: filtering.Recursive,
|
|
MaxDistance: filtering.MaxDistance,
|
|
}
|
|
} else if paths, ok := relPaths(zk, filtering.NoLinkedBy); ok {
|
|
opts.LinkedBy = ¬e.LinkedByFilter{
|
|
Paths: paths,
|
|
Negate: true,
|
|
}
|
|
}
|
|
|
|
if paths, ok := relPaths(zk, filtering.LinkTo); ok {
|
|
opts.LinkTo = ¬e.LinkToFilter{
|
|
Paths: paths,
|
|
Negate: false,
|
|
Recursive: filtering.Recursive,
|
|
MaxDistance: filtering.MaxDistance,
|
|
}
|
|
} else if paths, ok := relPaths(zk, filtering.NoLinkTo); ok {
|
|
opts.LinkTo = ¬e.LinkToFilter{
|
|
Paths: paths,
|
|
Negate: true,
|
|
}
|
|
}
|
|
|
|
if paths, ok := relPaths(zk, filtering.Related); ok {
|
|
opts.Related = paths
|
|
}
|
|
|
|
opts.Orphan = filtering.Orphan
|
|
|
|
if filtering.Created != "" {
|
|
start, end, err := parseDayRange(filtering.Created)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
opts.CreatedStart = &start
|
|
opts.CreatedEnd = &end
|
|
} else {
|
|
if filtering.CreatedBefore != "" {
|
|
date, err := parseDate(filtering.CreatedBefore)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
opts.CreatedEnd = &date
|
|
}
|
|
if filtering.CreatedAfter != "" {
|
|
date, err := parseDate(filtering.CreatedAfter)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
opts.CreatedStart = &date
|
|
}
|
|
}
|
|
|
|
if filtering.Modified != "" {
|
|
start, end, err := parseDayRange(filtering.Modified)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
opts.ModifiedStart = &start
|
|
opts.ModifiedEnd = &end
|
|
} else {
|
|
if filtering.ModifiedBefore != "" {
|
|
date, err := parseDate(filtering.ModifiedBefore)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
opts.ModifiedEnd = &date
|
|
}
|
|
if filtering.ModifiedAfter != "" {
|
|
date, err := parseDate(filtering.ModifiedAfter)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
opts.ModifiedStart = &date
|
|
}
|
|
}
|
|
|
|
opts.Interactive = filtering.Interactive
|
|
|
|
sorters, err := note.SortersFromStrings(sorting.Sort)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
opts.Sorters = sorters
|
|
|
|
opts.Limit = filtering.Limit
|
|
|
|
return &opts, nil
|
|
}
|
|
|
|
func relPaths(zk *zk.Zk, paths []string) ([]string, bool) {
|
|
relPaths := make([]string, 0)
|
|
for _, p := range paths {
|
|
path, err := zk.RelPath(p)
|
|
if err == nil {
|
|
relPaths = append(relPaths, path)
|
|
}
|
|
}
|
|
return relPaths, len(relPaths) > 0
|
|
}
|
|
|
|
func parseDate(date string) (time.Time, error) {
|
|
if i, err := strconv.ParseInt(date, 10, 0); err == nil && i >= 1000 && i < 5000 {
|
|
return time.Date(int(i), time.January, 0, 0, 0, 0, 0, time.UTC), nil
|
|
}
|
|
return naturaldate.Parse(date, time.Now().UTC(), naturaldate.WithDirection(naturaldate.Past))
|
|
}
|
|
|
|
func parseDayRange(date string) (start time.Time, end time.Time, err error) {
|
|
day, err := parseDate(date)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
start = startOfDay(day)
|
|
end = start.AddDate(0, 0, 1)
|
|
return start, end, nil
|
|
}
|
|
|
|
func startOfDay(t time.Time) time.Time {
|
|
year, month, day := t.Date()
|
|
return time.Date(year, month, day, 0, 0, 0, 0, t.Location())
|
|
}
|