mirror of
https://github.com/larksuite/cli.git
synced 2026-07-03 14:02:43 +08:00
Adds the apps domain to lark-cli for managing Miaoda (妙搭) applications: 6 shortcuts covering the full lifecycle (+create / +update / +list / +access-scope-set / +access-scope-get / +html-publish). Aligned with the OAPI v2 design — app_type enum (currently HTML), string scope enum (All / Tenant / Range), cursor pagination, in-memory tar.gz multipart publish flow. Namespace registered at /open-apis/spark/v1/ with spark:app.* scopes. --------- Co-authored-by: wangjiangwen-gif <286006750+wangjiangwen-gif@users.noreply.github.com>
171 lines
5.1 KiB
Go
171 lines
5.1 KiB
Go
// Copyright (c) 2026 Lark Technologies Pte. Ltd.
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
package apps
|
|
|
|
import (
|
|
"context"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
clie2e "github.com/larksuite/cli/tests/cli_e2e"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
"github.com/tidwall/gjson"
|
|
)
|
|
|
|
// TestAppsCreateDryRun pins the request shape and Validate behavior for
|
|
// `apps +create`. The shortcut is UAT-only and posts to the registered
|
|
// /open-apis/spark/v1 namespace; both are checked here.
|
|
func TestAppsCreateDryRun(t *testing.T) {
|
|
setAppsDryRunEnv(t)
|
|
|
|
t.Run("HappyPath_HTMLAppType", func(t *testing.T) {
|
|
ctx, cancel := context.WithTimeout(context.Background(), 30*time.Second)
|
|
t.Cleanup(cancel)
|
|
|
|
result, err := clie2e.RunCmd(ctx, clie2e.Request{
|
|
Args: []string{
|
|
"apps", "+create",
|
|
"--name", "Demo",
|
|
"--app-type", "HTML",
|
|
"--dry-run",
|
|
},
|
|
DefaultAs: "user",
|
|
})
|
|
require.NoError(t, err)
|
|
result.AssertExitCode(t, 0)
|
|
|
|
assert.Equal(t, "POST", gjson.Get(result.Stdout, "api.0.method").String())
|
|
assert.Equal(t, "/open-apis/spark/v1/apps", gjson.Get(result.Stdout, "api.0.url").String())
|
|
assert.Equal(t, "Demo", gjson.Get(result.Stdout, "api.0.body.name").String())
|
|
assert.Equal(t, "HTML", gjson.Get(result.Stdout, "api.0.body.app_type").String())
|
|
// Optional fields stay omitted when not provided.
|
|
assert.False(t, gjson.Get(result.Stdout, "api.0.body.description").Exists())
|
|
assert.False(t, gjson.Get(result.Stdout, "api.0.body.icon_url").Exists())
|
|
})
|
|
|
|
t.Run("AllFields", func(t *testing.T) {
|
|
ctx, cancel := context.WithTimeout(context.Background(), 30*time.Second)
|
|
t.Cleanup(cancel)
|
|
|
|
result, err := clie2e.RunCmd(ctx, clie2e.Request{
|
|
Args: []string{
|
|
"apps", "+create",
|
|
"--name", "Demo",
|
|
"--app-type", "HTML",
|
|
"--description", "survey app",
|
|
"--icon-url", "https://example.com/icon.svg",
|
|
"--dry-run",
|
|
},
|
|
DefaultAs: "user",
|
|
})
|
|
require.NoError(t, err)
|
|
result.AssertExitCode(t, 0)
|
|
|
|
assert.Equal(t, "Demo", gjson.Get(result.Stdout, "api.0.body.name").String())
|
|
assert.Equal(t, "HTML", gjson.Get(result.Stdout, "api.0.body.app_type").String())
|
|
assert.Equal(t, "survey app", gjson.Get(result.Stdout, "api.0.body.description").String())
|
|
assert.Equal(t, "https://example.com/icon.svg", gjson.Get(result.Stdout, "api.0.body.icon_url").String())
|
|
})
|
|
|
|
t.Run("RejectsMissingName", func(t *testing.T) {
|
|
ctx, cancel := context.WithTimeout(context.Background(), 30*time.Second)
|
|
t.Cleanup(cancel)
|
|
|
|
result, err := clie2e.RunCmd(ctx, clie2e.Request{
|
|
Args: []string{
|
|
"apps", "+create",
|
|
"--app-type", "HTML",
|
|
"--dry-run",
|
|
},
|
|
DefaultAs: "user",
|
|
})
|
|
require.NoError(t, err)
|
|
// cobra Required failures exit with code 1 (distinct from output.ErrValidation
|
|
// at code 2). Message goes to stderr as plain text, but we read combined output
|
|
// to stay robust to future runner changes.
|
|
result.AssertExitCode(t, 1)
|
|
assert.Contains(t, result.Stdout+result.Stderr, `required flag(s) "name" not set`)
|
|
})
|
|
|
|
t.Run("RejectsBlankName", func(t *testing.T) {
|
|
ctx, cancel := context.WithTimeout(context.Background(), 30*time.Second)
|
|
t.Cleanup(cancel)
|
|
|
|
result, err := clie2e.RunCmd(ctx, clie2e.Request{
|
|
Args: []string{
|
|
"apps", "+create",
|
|
"--name", " ",
|
|
"--app-type", "HTML",
|
|
"--dry-run",
|
|
},
|
|
DefaultAs: "user",
|
|
})
|
|
require.NoError(t, err)
|
|
result.AssertExitCode(t, 2)
|
|
msg := validateErrorMessage(result)
|
|
assert.Contains(t, msg, "--name is required")
|
|
})
|
|
|
|
t.Run("RejectsMissingAppType", func(t *testing.T) {
|
|
ctx, cancel := context.WithTimeout(context.Background(), 30*time.Second)
|
|
t.Cleanup(cancel)
|
|
|
|
result, err := clie2e.RunCmd(ctx, clie2e.Request{
|
|
Args: []string{
|
|
"apps", "+create",
|
|
"--name", "Demo",
|
|
"--dry-run",
|
|
},
|
|
DefaultAs: "user",
|
|
})
|
|
require.NoError(t, err)
|
|
result.AssertExitCode(t, 1)
|
|
assert.Contains(t, result.Stdout+result.Stderr, `required flag(s) "app-type" not set`)
|
|
})
|
|
|
|
t.Run("RejectsInvalidAppType", func(t *testing.T) {
|
|
ctx, cancel := context.WithTimeout(context.Background(), 30*time.Second)
|
|
t.Cleanup(cancel)
|
|
|
|
result, err := clie2e.RunCmd(ctx, clie2e.Request{
|
|
Args: []string{
|
|
"apps", "+create",
|
|
"--name", "Demo",
|
|
"--app-type", "spa",
|
|
"--dry-run",
|
|
},
|
|
DefaultAs: "user",
|
|
})
|
|
require.NoError(t, err)
|
|
result.AssertExitCode(t, 2)
|
|
msg := validateErrorMessage(result)
|
|
assert.Contains(t, msg, "not supported")
|
|
assert.Contains(t, msg, "HTML")
|
|
})
|
|
|
|
t.Run("RejectsLowercaseAppType", func(t *testing.T) {
|
|
// app-type is case-sensitive; lowercase "html" must be rejected even though
|
|
// it differs from the allowed "HTML" by case alone.
|
|
ctx, cancel := context.WithTimeout(context.Background(), 30*time.Second)
|
|
t.Cleanup(cancel)
|
|
|
|
result, err := clie2e.RunCmd(ctx, clie2e.Request{
|
|
Args: []string{
|
|
"apps", "+create",
|
|
"--name", "Demo",
|
|
"--app-type", "html",
|
|
"--dry-run",
|
|
},
|
|
DefaultAs: "user",
|
|
})
|
|
require.NoError(t, err)
|
|
result.AssertExitCode(t, 2)
|
|
msg := validateErrorMessage(result)
|
|
assert.True(t, strings.Contains(msg, `"html"`) && strings.Contains(msg, "not supported"),
|
|
"expected case-sensitive rejection, got: %s", msg)
|
|
})
|
|
}
|