- Rename 'propose' group to 'proposal' in surface, leaf help, and routing - Keep 'hf propose' as backward-compatible alias via groupAliases - Add essential subcommand group: list, create, update, delete - Accept command now shows generated story tasks in output - Accept command supports --json output - Task create blocks story/* types with helpful error message - All help text updated to use 'proposal' terminology
407 lines
9.8 KiB
Go
407 lines
9.8 KiB
Go
package commands
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
|
|
"git.hangman-lab.top/zhi/HarborForge.Cli/internal/client"
|
|
"git.hangman-lab.top/zhi/HarborForge.Cli/internal/config"
|
|
"git.hangman-lab.top/zhi/HarborForge.Cli/internal/output"
|
|
)
|
|
|
|
// proposeResponse matches the backend ProposeResponse schema.
|
|
type proposeResponse struct {
|
|
ID int `json:"id"`
|
|
Code string `json:"code"`
|
|
Title string `json:"title"`
|
|
Description *string `json:"description"`
|
|
Status string `json:"status"`
|
|
ProjectCode string `json:"project_code"`
|
|
CreatedBy *string `json:"created_by"`
|
|
CreatedAt string `json:"created_at"`
|
|
}
|
|
|
|
// RunProposeList implements `hf propose list --project <project-code>`.
|
|
func RunProposeList(args []string, tokenFlag string) {
|
|
token := ResolveToken(tokenFlag)
|
|
|
|
query := ""
|
|
for i := 0; i < len(args); i++ {
|
|
switch args[i] {
|
|
case "--project":
|
|
if i+1 >= len(args) {
|
|
output.Error("--project requires a value")
|
|
}
|
|
i++
|
|
query = appendQuery(query, "project", args[i])
|
|
case "--status":
|
|
if i+1 >= len(args) {
|
|
output.Error("--status requires a value")
|
|
}
|
|
i++
|
|
query = appendQuery(query, "status", args[i])
|
|
case "--order-by":
|
|
if i+1 >= len(args) {
|
|
output.Error("--order-by requires a value")
|
|
}
|
|
i++
|
|
query = appendQuery(query, "order_by", args[i])
|
|
default:
|
|
output.Errorf("unknown flag: %s", args[i])
|
|
}
|
|
}
|
|
|
|
cfg, err := config.Load()
|
|
if err != nil {
|
|
output.Errorf("config error: %v", err)
|
|
}
|
|
c := client.New(cfg.BaseURL, token)
|
|
path := "/proposes"
|
|
if query != "" {
|
|
path += "?" + query
|
|
}
|
|
data, err := c.Get(path)
|
|
if err != nil {
|
|
output.Errorf("failed to list proposals: %v", err)
|
|
}
|
|
|
|
if output.JSONMode {
|
|
var raw json.RawMessage
|
|
if err := json.Unmarshal(data, &raw); err != nil {
|
|
output.Errorf("invalid JSON response: %v", err)
|
|
}
|
|
output.PrintJSON(raw)
|
|
return
|
|
}
|
|
|
|
var proposes []proposeResponse
|
|
if err := json.Unmarshal(data, &proposes); err != nil {
|
|
output.Errorf("cannot parse proposal list: %v", err)
|
|
}
|
|
|
|
headers := []string{"CODE", "TITLE", "STATUS", "PROJECT", "CREATED BY"}
|
|
var rows [][]string
|
|
for _, p := range proposes {
|
|
createdBy := ""
|
|
if p.CreatedBy != nil {
|
|
createdBy = *p.CreatedBy
|
|
}
|
|
title := p.Title
|
|
if len(title) > 40 {
|
|
title = title[:37] + "..."
|
|
}
|
|
rows = append(rows, []string{p.Code, title, p.Status, p.ProjectCode, createdBy})
|
|
}
|
|
output.PrintTable(headers, rows)
|
|
}
|
|
|
|
// RunProposeGet implements `hf propose get <propose-code>`.
|
|
func RunProposeGet(proposeCode, tokenFlag string) {
|
|
token := ResolveToken(tokenFlag)
|
|
cfg, err := config.Load()
|
|
if err != nil {
|
|
output.Errorf("config error: %v", err)
|
|
}
|
|
c := client.New(cfg.BaseURL, token)
|
|
data, err := c.Get("/proposes/" + proposeCode)
|
|
if err != nil {
|
|
output.Errorf("failed to get proposal: %v", err)
|
|
}
|
|
|
|
if output.JSONMode {
|
|
var raw json.RawMessage
|
|
if err := json.Unmarshal(data, &raw); err != nil {
|
|
output.Errorf("invalid JSON response: %v", err)
|
|
}
|
|
output.PrintJSON(raw)
|
|
return
|
|
}
|
|
|
|
var p proposeResponse
|
|
if err := json.Unmarshal(data, &p); err != nil {
|
|
output.Errorf("cannot parse proposal: %v", err)
|
|
}
|
|
|
|
desc := ""
|
|
if p.Description != nil {
|
|
desc = *p.Description
|
|
}
|
|
createdBy := ""
|
|
if p.CreatedBy != nil {
|
|
createdBy = *p.CreatedBy
|
|
}
|
|
output.PrintKeyValue(
|
|
"code", p.Code,
|
|
"title", p.Title,
|
|
"description", desc,
|
|
"status", p.Status,
|
|
"project", p.ProjectCode,
|
|
"created-by", createdBy,
|
|
"created", p.CreatedAt,
|
|
)
|
|
}
|
|
|
|
// RunProposeCreate implements `hf propose create`.
|
|
func RunProposeCreate(args []string, tokenFlag string) {
|
|
token := ResolveToken(tokenFlag)
|
|
|
|
project, title, desc := "", "", ""
|
|
for i := 0; i < len(args); i++ {
|
|
switch args[i] {
|
|
case "--project":
|
|
if i+1 >= len(args) {
|
|
output.Error("--project requires a value")
|
|
}
|
|
i++
|
|
project = args[i]
|
|
case "--title":
|
|
if i+1 >= len(args) {
|
|
output.Error("--title requires a value")
|
|
}
|
|
i++
|
|
title = args[i]
|
|
case "--desc":
|
|
if i+1 >= len(args) {
|
|
output.Error("--desc requires a value")
|
|
}
|
|
i++
|
|
desc = args[i]
|
|
default:
|
|
output.Errorf("unknown flag: %s", args[i])
|
|
}
|
|
}
|
|
|
|
if project == "" || title == "" || desc == "" {
|
|
output.Error("usage: hf propose create --project <project-code> --title <title> --desc <desc>")
|
|
}
|
|
|
|
payload := map[string]interface{}{
|
|
"project_code": project,
|
|
"title": title,
|
|
"description": desc,
|
|
}
|
|
|
|
body, err := json.Marshal(payload)
|
|
if err != nil {
|
|
output.Errorf("cannot marshal payload: %v", err)
|
|
}
|
|
|
|
cfg, err := config.Load()
|
|
if err != nil {
|
|
output.Errorf("config error: %v", err)
|
|
}
|
|
c := client.New(cfg.BaseURL, token)
|
|
data, err := c.Post("/proposes", bytes.NewReader(body))
|
|
if err != nil {
|
|
output.Errorf("failed to create proposal: %v", err)
|
|
}
|
|
|
|
if output.JSONMode {
|
|
var raw json.RawMessage
|
|
if err := json.Unmarshal(data, &raw); err != nil {
|
|
output.Errorf("invalid JSON response: %v", err)
|
|
}
|
|
output.PrintJSON(raw)
|
|
return
|
|
}
|
|
|
|
var p proposeResponse
|
|
if err := json.Unmarshal(data, &p); err != nil {
|
|
fmt.Printf("proposal created: %s\n", title)
|
|
return
|
|
}
|
|
fmt.Printf("proposal created: %s (code: %s)\n", p.Title, p.Code)
|
|
}
|
|
|
|
// RunProposeUpdate implements `hf propose update <propose-code>`.
|
|
func RunProposeUpdate(proposeCode string, args []string, tokenFlag string) {
|
|
token := ResolveToken(tokenFlag)
|
|
|
|
payload := make(map[string]interface{})
|
|
for i := 0; i < len(args); i++ {
|
|
switch args[i] {
|
|
case "--title":
|
|
if i+1 >= len(args) {
|
|
output.Error("--title requires a value")
|
|
}
|
|
i++
|
|
payload["title"] = args[i]
|
|
case "--desc":
|
|
if i+1 >= len(args) {
|
|
output.Error("--desc requires a value")
|
|
}
|
|
i++
|
|
payload["description"] = args[i]
|
|
default:
|
|
output.Errorf("unknown flag: %s", args[i])
|
|
}
|
|
}
|
|
|
|
if len(payload) == 0 {
|
|
output.Error("nothing to update — provide at least one flag")
|
|
}
|
|
|
|
body, err := json.Marshal(payload)
|
|
if err != nil {
|
|
output.Errorf("cannot marshal payload: %v", err)
|
|
}
|
|
|
|
cfg, err := config.Load()
|
|
if err != nil {
|
|
output.Errorf("config error: %v", err)
|
|
}
|
|
c := client.New(cfg.BaseURL, token)
|
|
_, err = c.Patch("/proposes/"+proposeCode, bytes.NewReader(body))
|
|
if err != nil {
|
|
output.Errorf("failed to update proposal: %v", err)
|
|
}
|
|
|
|
fmt.Printf("proposal updated: %s\n", proposeCode)
|
|
}
|
|
|
|
// acceptResponse holds the accept result including generated tasks.
|
|
type acceptResponse struct {
|
|
ProposalCode string `json:"proposal_code"`
|
|
Status string `json:"status"`
|
|
GeneratedTasks []generatedTask `json:"generated_tasks"`
|
|
}
|
|
|
|
type generatedTask struct {
|
|
TaskID int `json:"task_id"`
|
|
TaskCode *string `json:"task_code"`
|
|
Title string `json:"title"`
|
|
TaskType string `json:"task_type"`
|
|
TaskSubtype *string `json:"task_subtype"`
|
|
}
|
|
|
|
// RunProposeAccept implements `hf proposal accept <proposal-code> --milestone <milestone-code>`.
|
|
func RunProposeAccept(proposeCode string, args []string, tokenFlag string) {
|
|
token := ResolveToken(tokenFlag)
|
|
|
|
milestone := ""
|
|
for i := 0; i < len(args); i++ {
|
|
switch args[i] {
|
|
case "--milestone":
|
|
if i+1 >= len(args) {
|
|
output.Error("--milestone requires a value")
|
|
}
|
|
i++
|
|
milestone = args[i]
|
|
default:
|
|
output.Errorf("unknown flag: %s", args[i])
|
|
}
|
|
}
|
|
|
|
if milestone == "" {
|
|
output.Error("usage: hf proposal accept <proposal-code> --milestone <milestone-code>")
|
|
}
|
|
|
|
payload := map[string]interface{}{
|
|
"milestone_code": milestone,
|
|
}
|
|
body, err := json.Marshal(payload)
|
|
if err != nil {
|
|
output.Errorf("cannot marshal payload: %v", err)
|
|
}
|
|
|
|
cfg, err := config.Load()
|
|
if err != nil {
|
|
output.Errorf("config error: %v", err)
|
|
}
|
|
c := client.New(cfg.BaseURL, token)
|
|
data, err := c.Post("/proposes/"+proposeCode+"/accept", bytes.NewReader(body))
|
|
if err != nil {
|
|
output.Errorf("failed to accept proposal: %v", err)
|
|
}
|
|
|
|
if output.JSONMode {
|
|
var raw json.RawMessage
|
|
if err := json.Unmarshal(data, &raw); err != nil {
|
|
output.Errorf("invalid JSON response: %v", err)
|
|
}
|
|
output.PrintJSON(raw)
|
|
return
|
|
}
|
|
|
|
fmt.Printf("proposal accepted: %s\n", proposeCode)
|
|
|
|
// Try to parse and display generated tasks
|
|
var resp acceptResponse
|
|
if err := json.Unmarshal(data, &resp); err == nil && len(resp.GeneratedTasks) > 0 {
|
|
fmt.Printf("\nGenerated %d story task(s):\n", len(resp.GeneratedTasks))
|
|
for _, gt := range resp.GeneratedTasks {
|
|
code := ""
|
|
if gt.TaskCode != nil {
|
|
code = *gt.TaskCode
|
|
}
|
|
subtype := ""
|
|
if gt.TaskSubtype != nil {
|
|
subtype = "/" + *gt.TaskSubtype
|
|
}
|
|
fmt.Printf(" %s %s%s %s\n", code, gt.TaskType, subtype, gt.Title)
|
|
}
|
|
}
|
|
}
|
|
|
|
// RunProposeReject implements `hf propose reject <propose-code>`.
|
|
func RunProposeReject(proposeCode string, args []string, tokenFlag string) {
|
|
token := ResolveToken(tokenFlag)
|
|
|
|
reason := ""
|
|
for i := 0; i < len(args); i++ {
|
|
switch args[i] {
|
|
case "--reason":
|
|
if i+1 >= len(args) {
|
|
output.Error("--reason requires a value")
|
|
}
|
|
i++
|
|
reason = args[i]
|
|
default:
|
|
output.Errorf("unknown flag: %s", args[i])
|
|
}
|
|
}
|
|
|
|
var body io.Reader
|
|
if reason != "" {
|
|
payload := map[string]interface{}{
|
|
"reason": reason,
|
|
}
|
|
data, err := json.Marshal(payload)
|
|
if err != nil {
|
|
output.Errorf("cannot marshal payload: %v", err)
|
|
}
|
|
body = bytes.NewReader(data)
|
|
}
|
|
|
|
cfg, err := config.Load()
|
|
if err != nil {
|
|
output.Errorf("config error: %v", err)
|
|
}
|
|
c := client.New(cfg.BaseURL, token)
|
|
_, err = c.Post("/proposes/"+proposeCode+"/reject", body)
|
|
if err != nil {
|
|
output.Errorf("failed to reject proposal: %v", err)
|
|
}
|
|
|
|
fmt.Printf("proposal rejected: %s\n", proposeCode)
|
|
}
|
|
|
|
// RunProposeReopen implements `hf propose reopen <propose-code>`.
|
|
func RunProposeReopen(proposeCode, tokenFlag string) {
|
|
token := ResolveToken(tokenFlag)
|
|
|
|
cfg, err := config.Load()
|
|
if err != nil {
|
|
output.Errorf("config error: %v", err)
|
|
}
|
|
c := client.New(cfg.BaseURL, token)
|
|
_, err = c.Post("/proposes/"+proposeCode+"/reopen", nil)
|
|
if err != nil {
|
|
output.Errorf("failed to reopen proposal: %v", err)
|
|
}
|
|
|
|
fmt.Printf("proposal reopened: %s\n", proposeCode)
|
|
}
|