mirror of
https://github.com/larksuite/cli.git
synced 2026-07-03 14:02:43 +08:00
Task commands now return structured, typed errors instead of the legacy exit-code envelope: every failure carries a stable category, subtype, and recovery hint, so callers can branch on the error class instead of parsing messages. Exit codes derive from the error category — input validation exits 2, a permission denial exits 3, other API errors exit 1. Batch operations (adding tasks to a tasklist, creating a tasklist with tasks) now report partial failure honestly: the per-item successes and failures stay on stdout and the command exits non-zero instead of masking failures as a success.
143 lines
4.4 KiB
Go
143 lines
4.4 KiB
Go
// Copyright (c) 2026 Lark Technologies Pte. Ltd.
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
package task
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"strings"
|
|
|
|
"github.com/larksuite/cli/internal/output"
|
|
"github.com/larksuite/cli/shortcuts/common"
|
|
)
|
|
|
|
const (
|
|
relatedTasksDefaultPageLimit = 20
|
|
relatedTasksMaxPageLimit = 40
|
|
relatedTasksPageSize = 100
|
|
)
|
|
|
|
var GetRelatedTasks = common.Shortcut{
|
|
Service: "task",
|
|
Command: "+get-related-tasks",
|
|
Description: "list tasks related to me",
|
|
Risk: "read",
|
|
Scopes: []string{"task:task:read"},
|
|
AuthTypes: []string{"user"},
|
|
HasFormat: true,
|
|
Flags: []common.Flag{
|
|
{Name: "include-complete", Type: "bool", Desc: "default true; set false to return only incomplete tasks"},
|
|
{Name: "page-all", Type: "bool", Desc: "automatically paginate through all pages (max 40)"},
|
|
{Name: "page-limit", Type: "int", Default: "20", Desc: "max page limit (default 20, max 40)"},
|
|
{Name: "page-token", Desc: "page token / updated_at cursor in microseconds"},
|
|
{Name: "created-by-me", Type: "bool", Desc: "client-side filter to tasks created by me; pagination still follows upstream related-task pages"},
|
|
{Name: "followed-by-me", Type: "bool", Desc: "client-side filter to tasks followed by me; pagination still follows upstream related-task pages"},
|
|
},
|
|
DryRun: func(ctx context.Context, runtime *common.RuntimeContext) *common.DryRunAPI {
|
|
params := map[string]interface{}{
|
|
"user_id_type": "open_id",
|
|
"page_size": relatedTasksPageSize,
|
|
}
|
|
if runtime.Cmd.Flags().Changed("include-complete") && !runtime.Bool("include-complete") {
|
|
params["completed"] = false
|
|
}
|
|
if pageToken := runtime.Str("page-token"); pageToken != "" {
|
|
params["page_token"] = pageToken
|
|
}
|
|
return common.NewDryRunAPI().
|
|
GET("/open-apis/task/v2/task_v2/list_related_task").
|
|
Params(params)
|
|
},
|
|
Execute: func(ctx context.Context, runtime *common.RuntimeContext) error {
|
|
params := map[string]interface{}{
|
|
"user_id_type": "open_id",
|
|
"page_size": relatedTasksPageSize,
|
|
}
|
|
if runtime.Cmd.Flags().Changed("include-complete") && !runtime.Bool("include-complete") {
|
|
params["completed"] = "false"
|
|
}
|
|
if pageToken := runtime.Str("page-token"); pageToken != "" {
|
|
params["page_token"] = pageToken
|
|
}
|
|
|
|
pageLimit := runtime.Int("page-limit")
|
|
if pageLimit <= 0 {
|
|
pageLimit = relatedTasksDefaultPageLimit
|
|
}
|
|
if runtime.Bool("page-all") {
|
|
pageLimit = relatedTasksMaxPageLimit
|
|
}
|
|
if pageLimit > relatedTasksMaxPageLimit {
|
|
pageLimit = relatedTasksMaxPageLimit
|
|
}
|
|
|
|
var allItems []interface{}
|
|
var lastPageToken string
|
|
var lastHasMore bool
|
|
for page := 0; page < pageLimit; page++ {
|
|
data, err := callTaskAPITyped(runtime, http.MethodGet, "/open-apis/task/v2/task_v2/list_related_task", params, nil)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
items, _ := data["items"].([]interface{})
|
|
allItems = append(allItems, items...)
|
|
lastHasMore, _ = data["has_more"].(bool)
|
|
lastPageToken, _ = data["page_token"].(string)
|
|
if !lastHasMore || lastPageToken == "" {
|
|
break
|
|
}
|
|
params["page_token"] = lastPageToken
|
|
}
|
|
|
|
userOpenID := runtime.UserOpenId()
|
|
filtered := make([]map[string]interface{}, 0, len(allItems))
|
|
for _, item := range allItems {
|
|
task, ok := item.(map[string]interface{})
|
|
if !ok {
|
|
continue
|
|
}
|
|
if runtime.Bool("created-by-me") {
|
|
creator, _ := task["creator"].(map[string]interface{})
|
|
if creatorID, _ := creator["id"].(string); creatorID != userOpenID {
|
|
continue
|
|
}
|
|
}
|
|
if runtime.Bool("followed-by-me") && !taskFollowedBy(task, userOpenID) {
|
|
continue
|
|
}
|
|
filtered = append(filtered, outputRelatedTask(task))
|
|
}
|
|
|
|
outData := map[string]interface{}{
|
|
"items": filtered,
|
|
"page_token": lastPageToken,
|
|
"has_more": lastHasMore,
|
|
}
|
|
runtime.OutFormat(outData, &output.Meta{Count: len(filtered)}, func(w io.Writer) {
|
|
if len(filtered) == 0 {
|
|
fmt.Fprintln(w, "No related tasks found.")
|
|
return
|
|
}
|
|
io.WriteString(w, renderRelatedTasksPretty(filtered, lastHasMore, lastPageToken))
|
|
})
|
|
return nil
|
|
},
|
|
}
|
|
|
|
func taskFollowedBy(task map[string]interface{}, userOpenID string) bool {
|
|
members, _ := task["members"].([]interface{})
|
|
for _, member := range members {
|
|
memberObj, _ := member.(map[string]interface{})
|
|
role, _ := memberObj["role"].(string)
|
|
id, _ := memberObj["id"].(string)
|
|
if strings.EqualFold(role, "follower") && id == userOpenID {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|