mirror of
https://github.com/larksuite/cli.git
synced 2026-07-04 06:29:52 +08:00
* fix(apps): read app object from data.app for +create and +update
The Miaoda OpenAPI returns the application object nested under
data.app for both POST /apps and PATCH /apps/{appId}. The CLI text
helper was reading common.GetString(data, "app_id"), which yields an
empty string against the wire format -- so `lark-cli apps +create
--format pretty` printed `created: ` with no ID.
Navigate the new nested path via GetString(data, "app", "app_id") for
both create and update. Update unit-test mocks to wrap the response
under `app`. Refresh the lark-apps skill references (example response
shape + jq paths) so agents reading them follow the right path.
Wire format is passed through to the user's JSON envelope untouched
-- no unwrapping in CLI. Consumers reading the response should use
.data.app.app_id.
The GET /apps list endpoint is unchanged: per the design doc its
items[] are flat objects, no wrapper.
* docs(apps): add required --app-type HTML to scenario 2 snippet
The "用户没有 app_id" snippet in lark-apps-html-publish.md was missing
the required --app-type flag, so copy-pasting it triggered Validate
("--app-type is required") and left $APP empty -- the following
+html-publish then failed with --app-id "". Bring the snippet in line
with every other apps +create example in the skill.
* docs(apps): simplify auth-recovery rule to error.type == missing_scope
Every apps shortcut declares Scopes, so the precheck path in
shortcuts/common/runner.go:825 is always the one that fires on scope
violations and the envelope's error.type is the stable discriminator.
Drop the keyword-sniffing of error.hint, the chain explanation, and the
bot caveat — they all reduce to one boolean: error.type == "missing_scope"
→ run `lark-cli auth login --domain apps`.
Also collapse the corresponding bullet in 快速决策 to point at this rule.
117 lines
3.5 KiB
Go
117 lines
3.5 KiB
Go
// Copyright (c) 2026 Lark Technologies Pte. Ltd.
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
package apps
|
|
|
|
import (
|
|
"encoding/json"
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/larksuite/cli/internal/httpmock"
|
|
)
|
|
|
|
func TestAppsUpdate_PartialFields(t *testing.T) {
|
|
factory, stdout, reg := newAppsExecuteFactory(t)
|
|
stub := &httpmock.Stub{
|
|
Method: "PATCH",
|
|
URL: "/open-apis/spark/v1/apps/app_x",
|
|
Body: map[string]interface{}{
|
|
"code": 0,
|
|
"data": map[string]interface{}{
|
|
"app": map[string]interface{}{
|
|
"app_id": "app_x",
|
|
"name": "renamed",
|
|
"updated_at": "2026-05-18T10:05:00Z",
|
|
},
|
|
},
|
|
},
|
|
}
|
|
reg.Register(stub)
|
|
|
|
if err := runAppsShortcut(t, AppsUpdate,
|
|
[]string{"+update", "--app-id", "app_x", "--name", "renamed", "--as", "user"},
|
|
factory, stdout); err != nil {
|
|
t.Fatalf("execute err=%v", err)
|
|
}
|
|
|
|
var sent map[string]interface{}
|
|
if err := json.Unmarshal(stub.CapturedBody, &sent); err != nil {
|
|
t.Fatalf("decode body: %v", err)
|
|
}
|
|
if sent["name"] != "renamed" {
|
|
t.Fatalf("body.name = %v", sent["name"])
|
|
}
|
|
if _, present := sent["description"]; present {
|
|
t.Fatalf("description should not be in body when not provided: %v", sent)
|
|
}
|
|
}
|
|
|
|
func TestAppsUpdate_RequiresAppID(t *testing.T) {
|
|
factory, stdout, _ := newAppsExecuteFactory(t)
|
|
err := runAppsShortcut(t, AppsUpdate,
|
|
[]string{"+update", "--name", "renamed", "--as", "user"}, factory, stdout)
|
|
// cobra Required:true may match "app-id" instead of "--app-id"
|
|
if err == nil || !strings.Contains(err.Error(), "app-id") {
|
|
t.Fatalf("expected --app-id required, got %v", err)
|
|
}
|
|
}
|
|
|
|
func TestAppsUpdate_RequiresAtLeastOneField(t *testing.T) {
|
|
factory, stdout, _ := newAppsExecuteFactory(t)
|
|
err := runAppsShortcut(t, AppsUpdate,
|
|
[]string{"+update", "--app-id", "app_x", "--as", "user"}, factory, stdout)
|
|
if err == nil {
|
|
t.Fatalf("expected error when no field provided")
|
|
}
|
|
}
|
|
|
|
func TestAppsUpdate_TrimsAppIDInPath(t *testing.T) {
|
|
// 钉死 --app-id 在拼进 URL 前要先 TrimSpace —— 与 create / access-scope-* 等保持一致,
|
|
// 避免 " app_x " 这种取值被原样 EncodePathSegment 编进 path 出现空格转义。
|
|
factory, stdout, reg := newAppsExecuteFactory(t)
|
|
stub := &httpmock.Stub{
|
|
Method: "PATCH",
|
|
URL: "/open-apis/spark/v1/apps/app_x",
|
|
Body: map[string]interface{}{
|
|
"code": 0,
|
|
"data": map[string]interface{}{
|
|
"app": map[string]interface{}{"app_id": "app_x"},
|
|
},
|
|
},
|
|
}
|
|
reg.Register(stub)
|
|
|
|
if err := runAppsShortcut(t, AppsUpdate,
|
|
[]string{"+update", "--app-id", " app_x ", "--name", "renamed", "--as", "user"},
|
|
factory, stdout); err != nil {
|
|
t.Fatalf("execute err=%v", err)
|
|
}
|
|
}
|
|
|
|
// TestAppsUpdate_PrettyOutputReadsNestedAppID exercises the prettyFn callback
|
|
// passed to OutFormat (only invoked under --format pretty) so the new
|
|
// data.app.app_id nesting is actually read by the text writer.
|
|
func TestAppsUpdate_PrettyOutputReadsNestedAppID(t *testing.T) {
|
|
factory, stdout, reg := newAppsExecuteFactory(t)
|
|
reg.Register(&httpmock.Stub{
|
|
Method: "PATCH",
|
|
URL: "/open-apis/spark/v1/apps/app_x",
|
|
Body: map[string]interface{}{
|
|
"code": 0,
|
|
"data": map[string]interface{}{
|
|
"app": map[string]interface{}{"app_id": "app_x", "name": "renamed"},
|
|
},
|
|
},
|
|
})
|
|
|
|
if err := runAppsShortcut(t, AppsUpdate,
|
|
[]string{"+update", "--app-id", "app_x", "--name", "renamed", "--format", "pretty", "--as", "user"},
|
|
factory, stdout); err != nil {
|
|
t.Fatalf("execute err=%v", err)
|
|
}
|
|
if got := stdout.String(); !strings.Contains(got, "updated: app_x") {
|
|
t.Fatalf("pretty output should read app_id from data.app.app_id, got: %q", got)
|
|
}
|
|
}
|