Convert all fmt.Fprintf(os.Stderr, ...) + os.Exit(1) patterns to use FatalErrorRespectJSON for consistent JSON error output: - dep.go: dependency commands (add, remove, tree, cycles) - label.go: label commands (add, remove, list, list-all) - comments.go: comment commands (list, add) - epic.go: epic commands (status, close-eligible) 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
242 lines
6.2 KiB
Go
242 lines
6.2 KiB
Go
package main
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"os"
|
|
"os/user"
|
|
"strings"
|
|
|
|
"github.com/spf13/cobra"
|
|
"github.com/steveyegge/beads/internal/rpc"
|
|
"github.com/steveyegge/beads/internal/types"
|
|
"github.com/steveyegge/beads/internal/utils"
|
|
)
|
|
|
|
var commentsCmd = &cobra.Command{
|
|
Use: "comments [issue-id]",
|
|
GroupID: "issues",
|
|
Short: "View or manage comments on an issue",
|
|
Long: `View or manage comments on an issue.
|
|
|
|
Examples:
|
|
# List all comments on an issue
|
|
bd comments bd-123
|
|
|
|
# List comments in JSON format
|
|
bd comments bd-123 --json
|
|
|
|
# Add a comment
|
|
bd comments add bd-123 "This is a comment"
|
|
|
|
# Add a comment from a file
|
|
bd comments add bd-123 -f notes.txt`,
|
|
Args: cobra.MinimumNArgs(1),
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
issueID := args[0]
|
|
|
|
comments := make([]*types.Comment, 0)
|
|
usedDaemon := false
|
|
if daemonClient != nil {
|
|
resp, err := daemonClient.ListComments(&rpc.CommentListArgs{ID: issueID})
|
|
if err != nil {
|
|
if isUnknownOperationError(err) {
|
|
if err := fallbackToDirectMode("daemon does not support comment_list RPC"); err != nil {
|
|
FatalErrorRespectJSON("getting comments: %v", err)
|
|
}
|
|
} else {
|
|
FatalErrorRespectJSON("getting comments: %v", err)
|
|
}
|
|
} else {
|
|
if err := json.Unmarshal(resp.Data, &comments); err != nil {
|
|
FatalErrorRespectJSON("decoding comments: %v", err)
|
|
}
|
|
usedDaemon = true
|
|
}
|
|
}
|
|
|
|
if !usedDaemon {
|
|
if err := ensureStoreActive(); err != nil {
|
|
FatalErrorRespectJSON("getting comments: %v", err)
|
|
}
|
|
ctx := rootCtx
|
|
fullID, err := utils.ResolvePartialID(ctx, store, issueID)
|
|
if err != nil {
|
|
FatalErrorRespectJSON("resolving %s: %v", issueID, err)
|
|
}
|
|
issueID = fullID
|
|
|
|
result, err := store.GetIssueComments(ctx, issueID)
|
|
if err != nil {
|
|
FatalErrorRespectJSON("getting comments: %v", err)
|
|
}
|
|
comments = result
|
|
}
|
|
|
|
// Normalize nil to empty slice for consistent JSON output
|
|
if comments == nil {
|
|
comments = make([]*types.Comment, 0)
|
|
}
|
|
|
|
if jsonOutput {
|
|
data, err := json.MarshalIndent(comments, "", " ")
|
|
if err != nil {
|
|
FatalErrorRespectJSON("encoding JSON: %v", err)
|
|
}
|
|
fmt.Println(string(data))
|
|
return
|
|
}
|
|
|
|
// Human-readable output
|
|
if len(comments) == 0 {
|
|
fmt.Printf("No comments on %s\n", issueID)
|
|
return
|
|
}
|
|
|
|
fmt.Printf("\nComments on %s:\n\n", issueID)
|
|
for _, comment := range comments {
|
|
fmt.Printf("[%s] %s at %s\n", comment.Author, comment.Text, comment.CreatedAt.Format("2006-01-02 15:04"))
|
|
fmt.Println()
|
|
}
|
|
},
|
|
}
|
|
|
|
var commentsAddCmd = &cobra.Command{
|
|
Use: "add [issue-id] [text]",
|
|
Short: "Add a comment to an issue",
|
|
Long: `Add a comment to an issue.
|
|
|
|
Examples:
|
|
# Add a comment
|
|
bd comments add bd-123 "Working on this now"
|
|
|
|
# Add a comment from a file
|
|
bd comments add bd-123 -f notes.txt`,
|
|
Args: cobra.MinimumNArgs(1),
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
CheckReadonly("comment add")
|
|
issueID := args[0]
|
|
|
|
// Get comment text from flag or argument
|
|
commentText, _ := cmd.Flags().GetString("file")
|
|
if commentText != "" {
|
|
// Read from file
|
|
data, err := os.ReadFile(commentText) // #nosec G304 - user-provided file path is intentional
|
|
if err != nil {
|
|
FatalErrorRespectJSON("reading file: %v", err)
|
|
}
|
|
commentText = string(data)
|
|
} else if len(args) < 2 {
|
|
FatalErrorRespectJSON("comment text required (use -f to read from file)")
|
|
} else {
|
|
commentText = args[1]
|
|
}
|
|
|
|
// Get author from author flag, BD_ACTOR var, or system USER var
|
|
author, _ := cmd.Flags().GetString("author")
|
|
if author == "" {
|
|
author = os.Getenv("BD_ACTOR")
|
|
if author == "" {
|
|
author = os.Getenv("USER")
|
|
}
|
|
if author == "" {
|
|
if u, err := user.Current(); err == nil {
|
|
author = u.Username
|
|
} else {
|
|
author = "unknown"
|
|
}
|
|
}
|
|
}
|
|
|
|
var comment *types.Comment
|
|
if daemonClient != nil {
|
|
resp, err := daemonClient.AddComment(&rpc.CommentAddArgs{
|
|
ID: issueID,
|
|
Author: author,
|
|
Text: commentText,
|
|
})
|
|
if err != nil {
|
|
if isUnknownOperationError(err) {
|
|
if err := fallbackToDirectMode("daemon does not support comment_add RPC"); err != nil {
|
|
FatalErrorRespectJSON("adding comment: %v", err)
|
|
}
|
|
} else {
|
|
FatalErrorRespectJSON("adding comment: %v", err)
|
|
}
|
|
} else {
|
|
var parsed types.Comment
|
|
if err := json.Unmarshal(resp.Data, &parsed); err != nil {
|
|
FatalErrorRespectJSON("decoding comment: %v", err)
|
|
}
|
|
comment = &parsed
|
|
}
|
|
}
|
|
|
|
if comment == nil {
|
|
if err := ensureStoreActive(); err != nil {
|
|
FatalErrorRespectJSON("adding comment: %v", err)
|
|
}
|
|
ctx := rootCtx
|
|
|
|
fullID, err := utils.ResolvePartialID(ctx, store, issueID)
|
|
if err != nil {
|
|
FatalErrorRespectJSON("resolving %s: %v", issueID, err)
|
|
}
|
|
issueID = fullID
|
|
|
|
comment, err = store.AddIssueComment(ctx, issueID, author, commentText)
|
|
if err != nil {
|
|
FatalErrorRespectJSON("adding comment: %v", err)
|
|
}
|
|
}
|
|
|
|
if jsonOutput {
|
|
data, err := json.MarshalIndent(comment, "", " ")
|
|
if err != nil {
|
|
FatalErrorRespectJSON("encoding JSON: %v", err)
|
|
}
|
|
fmt.Println(string(data))
|
|
return
|
|
}
|
|
|
|
fmt.Printf("Comment added to %s\n", issueID)
|
|
},
|
|
}
|
|
|
|
// commentCmd is a top-level alias for commentsAddCmd
|
|
var commentCmd = &cobra.Command{
|
|
Use: "comment [issue-id] [text]",
|
|
GroupID: "issues",
|
|
Short: "Add a comment to an issue (alias for 'comments add')",
|
|
Long: `Add a comment to an issue. This is a convenient alias for 'bd comments add'.
|
|
|
|
Examples:
|
|
# Add a comment
|
|
bd comment bd-123 "Working on this now"
|
|
|
|
# Add a comment from a file
|
|
bd comment bd-123 -f notes.txt`,
|
|
Args: cobra.MinimumNArgs(1),
|
|
Run: commentsAddCmd.Run,
|
|
}
|
|
|
|
func init() {
|
|
commentsCmd.AddCommand(commentsAddCmd)
|
|
commentsAddCmd.Flags().StringP("file", "f", "", "Read comment text from file")
|
|
commentsAddCmd.Flags().StringP("author", "a", "", "Add author to comment")
|
|
|
|
// Add the same flags to the alias
|
|
commentCmd.Flags().StringP("file", "f", "", "Read comment text from file")
|
|
commentCmd.Flags().StringP("author", "a", "", "Add author to comment")
|
|
|
|
rootCmd.AddCommand(commentsCmd)
|
|
rootCmd.AddCommand(commentCmd)
|
|
}
|
|
|
|
func isUnknownOperationError(err error) bool {
|
|
if err == nil {
|
|
return false
|
|
}
|
|
return strings.Contains(err.Error(), "unknown operation")
|
|
}
|