bug.go

  1package bugcmd
  2
  3import (
  4	"fmt"
  5	"regexp"
  6	"strings"
  7	"time"
  8
  9	text "github.com/MichaelMure/go-term-text"
 10	"github.com/spf13/cobra"
 11
 12	"github.com/MichaelMure/git-bug/cache"
 13	"github.com/MichaelMure/git-bug/commands/cmdjson"
 14	"github.com/MichaelMure/git-bug/commands/completion"
 15	"github.com/MichaelMure/git-bug/commands/execenv"
 16	"github.com/MichaelMure/git-bug/entities/bug"
 17	"github.com/MichaelMure/git-bug/entities/common"
 18	"github.com/MichaelMure/git-bug/query"
 19	"github.com/MichaelMure/git-bug/util/colors"
 20)
 21
 22type bugOptions struct {
 23	statusQuery      []string
 24	authorQuery      []string
 25	metadataQuery    []string
 26	participantQuery []string
 27	actorQuery       []string
 28	labelQuery       []string
 29	titleQuery       []string
 30	noQuery          []string
 31	sortBy           string
 32	sortDirection    string
 33	outputFormat     string
 34}
 35
 36func NewBugCommand() *cobra.Command {
 37	env := execenv.NewEnv()
 38	options := bugOptions{}
 39
 40	cmd := &cobra.Command{
 41		Use:   "bug [QUERY]",
 42		Short: "List bugs",
 43		Long: `Display a summary of each bugs.
 44
 45You can pass an additional query to filter and order the list. This query can be expressed either with a simple query language, flags, a natural language full text search, or a combination of the aforementioned.`,
 46		Example: `List open bugs sorted by last edition with a query:
 47git bug status:open sort:edit-desc
 48
 49List closed bugs sorted by creation with flags:
 50git bug --status closed --by creation
 51
 52Do a full text search of all bugs:
 53git bug "foo bar" baz
 54
 55Use queries, flags, and full text search:
 56git bug status:open --by creation "foo bar" baz
 57`,
 58		PreRunE: execenv.LoadBackend(env),
 59		RunE: execenv.CloseBackend(env, func(cmd *cobra.Command, args []string) error {
 60			return runBug(env, options, args)
 61		}),
 62		ValidArgsFunction: completion.Ls(env),
 63	}
 64
 65	flags := cmd.Flags()
 66	flags.SortFlags = false
 67
 68	flags.StringSliceVarP(&options.statusQuery, "status", "s", nil,
 69		"Filter by status. Valid values are [open,closed]")
 70	cmd.RegisterFlagCompletionFunc("status", completion.From([]string{"open", "closed"}))
 71	flags.StringSliceVarP(&options.authorQuery, "author", "a", nil,
 72		"Filter by author")
 73	flags.StringSliceVarP(&options.metadataQuery, "metadata", "m", nil,
 74		"Filter by metadata. Example: github-url=URL")
 75	cmd.RegisterFlagCompletionFunc("author", completion.UserForQuery(env))
 76	flags.StringSliceVarP(&options.participantQuery, "participant", "p", nil,
 77		"Filter by participant")
 78	cmd.RegisterFlagCompletionFunc("participant", completion.UserForQuery(env))
 79	flags.StringSliceVarP(&options.actorQuery, "actor", "A", nil,
 80		"Filter by actor")
 81	cmd.RegisterFlagCompletionFunc("actor", completion.UserForQuery(env))
 82	flags.StringSliceVarP(&options.labelQuery, "label", "l", nil,
 83		"Filter by label")
 84	cmd.RegisterFlagCompletionFunc("label", completion.Label(env))
 85	flags.StringSliceVarP(&options.titleQuery, "title", "t", nil,
 86		"Filter by title")
 87	flags.StringSliceVarP(&options.noQuery, "no", "n", nil,
 88		"Filter by absence of something. Valid values are [label]")
 89	cmd.RegisterFlagCompletionFunc("no", completion.Label(env))
 90	flags.StringVarP(&options.sortBy, "by", "b", "creation",
 91		"Sort the results by a characteristic. Valid values are [id,creation,edit]")
 92	cmd.RegisterFlagCompletionFunc("by", completion.From([]string{"id", "creation", "edit"}))
 93	flags.StringVarP(&options.sortDirection, "direction", "d", "asc",
 94		"Select the sorting direction. Valid values are [asc,desc]")
 95	cmd.RegisterFlagCompletionFunc("direction", completion.From([]string{"asc", "desc"}))
 96	flags.StringVarP(&options.outputFormat, "format", "f", "default",
 97		"Select the output formatting style. Valid values are [default,plain,compact,id,json,org-mode]")
 98	cmd.RegisterFlagCompletionFunc("format",
 99		completion.From([]string{"default", "plain", "compact", "id", "json", "org-mode"}))
100
101	const selectGroup = "select"
102	cmd.AddGroup(&cobra.Group{ID: selectGroup, Title: "Implicit selection"})
103
104	addCmdWithGroup := func(child *cobra.Command, groupID string) {
105		cmd.AddCommand(child)
106		child.GroupID = groupID
107	}
108
109	addCmdWithGroup(newBugDeselectCommand(), selectGroup)
110	addCmdWithGroup(newBugSelectCommand(), selectGroup)
111
112	cmd.AddCommand(newBugCommentCommand())
113	cmd.AddCommand(newBugLabelCommand())
114	cmd.AddCommand(newBugNewCommand())
115	cmd.AddCommand(newBugRmCommand())
116	cmd.AddCommand(newBugShowCommand())
117	cmd.AddCommand(newBugStatusCommand())
118	cmd.AddCommand(newBugTitleCommand())
119
120	return cmd
121}
122
123func runBug(env *execenv.Env, opts bugOptions, args []string) error {
124	var q *query.Query
125	var err error
126
127	if len(args) >= 1 {
128		// either the shell or cobra remove the quotes, we need them back for the query parsing
129		assembled := repairQuery(args)
130
131		q, err = query.Parse(assembled)
132		if err != nil {
133			return err
134		}
135	} else {
136		q = query.NewQuery()
137	}
138
139	err = completeQuery(q, opts)
140	if err != nil {
141		return err
142	}
143
144	allIds, err := env.Backend.Bugs().Query(q)
145	if err != nil {
146		return err
147	}
148
149	bugExcerpt := make([]*cache.BugExcerpt, len(allIds))
150	for i, id := range allIds {
151		b, err := env.Backend.Bugs().ResolveExcerpt(id)
152		if err != nil {
153			return err
154		}
155		bugExcerpt[i] = b
156	}
157
158	switch opts.outputFormat {
159	case "org-mode":
160		return bugsOrgmodeFormatter(env, bugExcerpt)
161	case "plain":
162		return bugsPlainFormatter(env, bugExcerpt)
163	case "json":
164		return bugsJsonFormatter(env, bugExcerpt)
165	case "compact":
166		return bugsCompactFormatter(env, bugExcerpt)
167	case "id":
168		return bugsIDFormatter(env, bugExcerpt)
169	case "default":
170		return bugsDefaultFormatter(env, bugExcerpt)
171	default:
172		return fmt.Errorf("unknown format %s", opts.outputFormat)
173	}
174}
175
176func repairQuery(args []string) string {
177	for i, arg := range args {
178		split := strings.Split(arg, ":")
179		for j, s := range split {
180			if strings.Contains(s, " ") {
181				split[j] = fmt.Sprintf("\"%s\"", s)
182			}
183		}
184		args[i] = strings.Join(split, ":")
185	}
186	return strings.Join(args, " ")
187}
188
189func bugsJsonFormatter(env *execenv.Env, bugExcerpts []*cache.BugExcerpt) error {
190	jsonBugs := make([]cmdjson.BugExcerpt, len(bugExcerpts))
191	for i, b := range bugExcerpts {
192		jsonBug, err := cmdjson.NewBugExcerpt(env.Backend, b)
193		if err != nil {
194			return err
195		}
196		jsonBugs[i] = jsonBug
197	}
198	return env.Out.PrintJSON(jsonBugs)
199}
200
201func bugsCompactFormatter(env *execenv.Env, bugExcerpts []*cache.BugExcerpt) error {
202	for _, b := range bugExcerpts {
203		author, err := env.Backend.Identities().ResolveExcerpt(b.AuthorId)
204		if err != nil {
205			return err
206		}
207
208		var labelsTxt strings.Builder
209		for _, l := range b.Labels {
210			lc256 := l.Color().Term256()
211			labelsTxt.WriteString(lc256.Escape())
212			labelsTxt.WriteString("◼")
213			labelsTxt.WriteString(lc256.Unescape())
214		}
215
216		env.Out.Printf("%s %s %s %s %s\n",
217			colors.Cyan(b.Id().Human()),
218			colors.Yellow(b.Status),
219			text.LeftPadMaxLine(strings.TrimSpace(b.Title), 40, 0),
220			text.LeftPadMaxLine(labelsTxt.String(), 5, 0),
221			colors.Magenta(text.TruncateMax(author.DisplayName(), 15)),
222		)
223	}
224	return nil
225}
226
227func bugsIDFormatter(env *execenv.Env, bugExcerpts []*cache.BugExcerpt) error {
228	for _, b := range bugExcerpts {
229		env.Out.Println(b.Id().String())
230	}
231
232	return nil
233}
234
235func bugsDefaultFormatter(env *execenv.Env, bugExcerpts []*cache.BugExcerpt) error {
236	for _, b := range bugExcerpts {
237		author, err := env.Backend.Identities().ResolveExcerpt(b.AuthorId)
238		if err != nil {
239			return err
240		}
241
242		var labelsTxt strings.Builder
243		for _, l := range b.Labels {
244			lc256 := l.Color().Term256()
245			labelsTxt.WriteString(lc256.Escape())
246			labelsTxt.WriteString(" ◼")
247			labelsTxt.WriteString(lc256.Unescape())
248		}
249
250		// truncate + pad if needed
251		labelsFmt := text.TruncateMax(labelsTxt.String(), 10)
252		titleFmt := text.LeftPadMaxLine(strings.TrimSpace(b.Title), 50-text.Len(labelsFmt), 0)
253		authorFmt := text.LeftPadMaxLine(author.DisplayName(), 15, 0)
254
255		comments := fmt.Sprintf("%3d 💬", b.LenComments-1)
256		if b.LenComments-1 <= 0 {
257			comments = ""
258		}
259		if b.LenComments-1 > 999 {
260			comments = "  ∞ 💬"
261		}
262
263		env.Out.Printf("%s\t%s\t%s\t%s\t%s\n",
264			colors.Cyan(b.Id().Human()),
265			colors.Yellow(b.Status),
266			titleFmt+labelsFmt,
267			colors.Magenta(authorFmt),
268			comments,
269		)
270	}
271	return nil
272}
273
274func bugsPlainFormatter(env *execenv.Env, bugExcerpts []*cache.BugExcerpt) error {
275	for _, b := range bugExcerpts {
276		env.Out.Printf("%s [%s] %s\n", b.Id().Human(), b.Status, strings.TrimSpace(b.Title))
277	}
278	return nil
279}
280
281func bugsOrgmodeFormatter(env *execenv.Env, bugExcerpts []*cache.BugExcerpt) error {
282	// see https://orgmode.org/manual/Tags.html
283	orgTagRe := regexp.MustCompile("[^[:alpha:]_@]")
284	formatTag := func(l bug.Label) string {
285		return orgTagRe.ReplaceAllString(l.String(), "_")
286	}
287
288	formatTime := func(time time.Time) string {
289		return time.Format("[2006-01-02 Mon 15:05]")
290	}
291
292	env.Out.Println("#+TODO: OPEN | CLOSED")
293
294	for _, b := range bugExcerpts {
295		status := strings.ToUpper(b.Status.String())
296
297		var title string
298		if link, ok := b.CreateMetadata["github-url"]; ok {
299			title = fmt.Sprintf("[[%s][%s]]", link, b.Title)
300		} else {
301			title = b.Title
302		}
303
304		author, err := env.Backend.Identities().ResolveExcerpt(b.AuthorId)
305		if err != nil {
306			return err
307		}
308
309		var labels strings.Builder
310		labels.WriteString(":")
311		for i, l := range b.Labels {
312			if i > 0 {
313				labels.WriteString(":")
314			}
315			labels.WriteString(formatTag(l))
316		}
317		labels.WriteString(":")
318
319		env.Out.Printf("* %-6s %s %s %s: %s %s\n",
320			status,
321			b.Id().Human(),
322			formatTime(b.CreateTime()),
323			author.DisplayName(),
324			title,
325			labels.String(),
326		)
327
328		env.Out.Printf("** Last Edited: %s\n", formatTime(b.EditTime()))
329
330		env.Out.Printf("** Actors:\n")
331		for _, element := range b.Actors {
332			actor, err := env.Backend.Identities().ResolveExcerpt(element)
333			if err != nil {
334				return err
335			}
336
337			env.Out.Printf(": %s %s\n",
338				actor.Id().Human(),
339				actor.DisplayName(),
340			)
341		}
342
343		env.Out.Printf("** Participants:\n")
344		for _, element := range b.Participants {
345			participant, err := env.Backend.Identities().ResolveExcerpt(element)
346			if err != nil {
347				return err
348			}
349
350			env.Out.Printf(": %s %s\n",
351				participant.Id().Human(),
352				participant.DisplayName(),
353			)
354		}
355	}
356
357	return nil
358}
359
360// Finish the command flags transformation into the query.Query
361func completeQuery(q *query.Query, opts bugOptions) error {
362	for _, str := range opts.statusQuery {
363		status, err := common.StatusFromString(str)
364		if err != nil {
365			return err
366		}
367		q.Status = append(q.Status, status)
368	}
369
370	q.Author = append(q.Author, opts.authorQuery...)
371	for _, str := range opts.metadataQuery {
372		tokens := strings.Split(str, "=")
373		if len(tokens) < 2 {
374			return fmt.Errorf("no \"=\" in key=value metadata markup")
375		}
376		var pair query.StringPair
377		pair.Key = tokens[0]
378		pair.Value = tokens[1]
379		q.Metadata = append(q.Metadata, pair)
380	}
381	q.Participant = append(q.Participant, opts.participantQuery...)
382	q.Actor = append(q.Actor, opts.actorQuery...)
383	q.Label = append(q.Label, opts.labelQuery...)
384	q.Title = append(q.Title, opts.titleQuery...)
385
386	for _, no := range opts.noQuery {
387		switch no {
388		case "label":
389			q.NoLabel = true
390		default:
391			return fmt.Errorf("unknown \"no\" filter %s", no)
392		}
393	}
394
395	switch opts.sortBy {
396	case "id":
397		q.OrderBy = query.OrderById
398	case "creation":
399		q.OrderBy = query.OrderByCreation
400	case "edit":
401		q.OrderBy = query.OrderByEdit
402	default:
403		return fmt.Errorf("unknown sort flag %s", opts.sortBy)
404	}
405
406	switch opts.sortDirection {
407	case "asc":
408		q.OrderDirection = query.OrderAscending
409	case "desc":
410		q.OrderDirection = query.OrderDescending
411	default:
412		return fmt.Errorf("unknown sort direction %s", opts.sortDirection)
413	}
414
415	return nil
416}