mattermost-community-enterp.../public/model/integration_action_test.go
Claude ec1f89217a Merge: Complete Mattermost Server with Community Enterprise
Full Mattermost server source with integrated Community Enterprise features.
Includes vendor directory for offline/air-gapped builds.

Structure:
- enterprise-impl/: Enterprise feature implementations
- enterprise-community/: Init files that register implementations
- enterprise/: Bridge imports (community_imports.go)
- vendor/: All dependencies for offline builds

Build (online):
  go build ./cmd/mattermost

Build (offline/air-gapped):
  go build -mod=vendor ./cmd/mattermost

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2025-12-17 23:59:07 +09:00

1472 lines
42 KiB
Go

// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
// See LICENSE.txt for license information.
package model
import (
"crypto"
"crypto/ecdsa"
"crypto/elliptic"
"crypto/rand"
"encoding/base64"
"io"
"math/big"
"strings"
"testing"
"time"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
)
// failingSigner is a test implementation of crypto.Signer that always returns an error
type failingSigner struct {
err error
}
func (f *failingSigner) Public() crypto.PublicKey {
return nil
}
func (f *failingSigner) Sign(rand io.Reader, digest []byte, opts crypto.SignerOpts) (signature []byte, err error) {
return nil, f.err
}
func TestPostAction_IsValid(t *testing.T) {
tests := map[string]struct {
action *PostAction
wantErr string
}{
"valid button action with http URL": {
action: &PostAction{
Id: "validid",
Name: "Test Button",
Type: PostActionTypeButton,
Integration: &PostActionIntegration{
URL: "http://localhost:8065",
},
},
wantErr: "",
},
"valid button action with http URL without Id": {
action: &PostAction{
Name: "Test Button",
Type: PostActionTypeButton,
Integration: &PostActionIntegration{
URL: "http://localhost:8065",
},
},
wantErr: "",
},
"valid button action with plugin path": {
action: &PostAction{
Id: "validid",
Name: "Test Button",
Type: PostActionTypeButton,
Integration: &PostActionIntegration{
URL: "/plugins/myplugin/action",
},
},
wantErr: "",
},
"valid button action with relative plugin path": {
action: &PostAction{
Id: "validid",
Name: "Test Button",
Type: PostActionTypeButton,
Integration: &PostActionIntegration{
URL: "plugins/myplugin/action",
},
},
wantErr: "",
},
"invalid integration URL": {
action: &PostAction{
Id: "validid",
Name: "Test Button",
Type: PostActionTypeButton,
Integration: &PostActionIntegration{
URL: "invalid-url",
},
},
wantErr: "action must have an valid integration URL",
},
"valid select action with datasource": {
action: &PostAction{
Id: "validid",
Name: "Test Select",
Type: PostActionTypeSelect,
DataSource: PostActionDataSourceUsers,
Integration: &PostActionIntegration{
URL: "http://localhost:8065",
},
},
wantErr: "",
},
"valid select action with options": {
action: &PostAction{
Id: "validid",
Name: "Test Select",
Type: PostActionTypeSelect,
Options: []*PostActionOptions{
{Text: "Opt1", Value: "opt1"},
},
Integration: &PostActionIntegration{
URL: "http://localhost:8065",
},
},
wantErr: "",
},
"select action with nil option": {
action: &PostAction{
Id: "validid",
Name: "Test Select",
Type: PostActionTypeSelect,
Options: []*PostActionOptions{
nil,
{Text: "Opt1", Value: "opt1"},
},
Integration: &PostActionIntegration{
URL: "http://localhost:8065",
},
},
wantErr: "select action contains nil option",
},
"missing name": {
action: &PostAction{
Id: "validid",
Type: PostActionTypeButton,
Integration: &PostActionIntegration{
URL: "http://localhost:8065",
},
},
wantErr: "action must have a name",
},
"invalid style": {
action: &PostAction{
Id: "validid",
Name: "Test Button",
Type: PostActionTypeButton,
Style: "invalid",
Integration: &PostActionIntegration{
URL: "http://localhost:8065",
},
},
wantErr: "invalid style 'invalid' - must be one of [default, primary, success, good, warning, danger] or a hex color",
},
"valid style": {
action: &PostAction{
Id: "validid",
Name: "Test Button",
Type: PostActionTypeButton,
Style: "primary",
Integration: &PostActionIntegration{
URL: "http://localhost:8065",
},
},
wantErr: "",
},
"button with options": {
action: &PostAction{
Id: "validid",
Name: "Test Button",
Type: PostActionTypeButton,
Options: []*PostActionOptions{
{Text: "Opt1", Value: "opt1"},
},
Integration: &PostActionIntegration{
URL: "http://localhost:8065",
},
},
wantErr: "button action must not have options",
},
"button with datasource": {
action: &PostAction{
Id: "validid",
Name: "Test Button",
Type: PostActionTypeButton,
DataSource: PostActionDataSourceUsers,
Integration: &PostActionIntegration{
URL: "http://localhost:8065",
},
},
wantErr: "button action must not have a data source",
},
"select without datasource or options": {
action: &PostAction{
Id: "validid",
Name: "Test Select",
Type: PostActionTypeSelect,
Integration: &PostActionIntegration{
URL: "http://localhost:8065",
},
},
wantErr: "select action must have either DataSource or Options set",
},
"select with both datasource and options": {
action: &PostAction{
Id: "validid",
Name: "Test Select",
Type: PostActionTypeSelect,
DataSource: PostActionDataSourceUsers,
Options: []*PostActionOptions{
{Text: "Opt1", Value: "opt1"},
},
Integration: &PostActionIntegration{
URL: "http://localhost:8065",
},
},
wantErr: "select action cannot have both DataSource and Options set",
},
"invalid datasource": {
action: &PostAction{
Id: "validid",
Name: "Test Select",
Type: PostActionTypeSelect,
DataSource: "invalid",
Integration: &PostActionIntegration{
URL: "http://localhost:8065",
},
},
wantErr: "invalid data_source 'invalid' for select action",
},
"missing integration": {
action: &PostAction{
Id: "validid",
Name: "Test Button",
Type: PostActionTypeButton,
},
wantErr: "action must have integration settings",
},
"missing integration URL": {
action: &PostAction{
Id: "validid",
Name: "Test Button",
Type: PostActionTypeButton,
Integration: &PostActionIntegration{},
},
wantErr: "action must have an integration URL",
},
"invalid type": {
action: &PostAction{
Id: "validid",
Name: "Test Action",
Type: "invalid",
Integration: &PostActionIntegration{
URL: "http://localhost:8065",
},
},
wantErr: "invalid action type: must be 'button' or 'select'",
},
}
for name, tc := range tests {
t.Run(name, func(t *testing.T) {
err := tc.action.IsValid()
if tc.wantErr == "" {
assert.NoError(t, err, name)
} else {
assert.ErrorContains(t, err, tc.wantErr, name)
}
})
}
}
func TestGenerateTriggerId(t *testing.T) {
t.Run("should succeed with valid key", func(t *testing.T) {
key, err := ecdsa.GenerateKey(elliptic.P256(), rand.Reader)
require.NoError(t, err)
userId := NewId()
clientTriggerId, triggerId, appErr := GenerateTriggerId(userId, key)
assert.Nil(t, appErr)
assert.NotEmpty(t, clientTriggerId)
assert.NotEmpty(t, triggerId)
})
t.Run("should handle signer that returns error", func(t *testing.T) {
// Create a signer that always returns an error
badSigner := &failingSigner{err: assert.AnError}
userId := NewId()
_, _, appErr := GenerateTriggerId(userId, badSigner)
require.NotNil(t, appErr)
assert.Equal(t, "interactive_message.generate_trigger_id.signing_failed", appErr.Id)
assert.NotEmpty(t, appErr.Error())
})
t.Run("should handle invalid ECDSA key with nil D value", func(t *testing.T) {
// Create an invalid ECDSA key with nil D (private key component)
// This would normally panic in crypto/ecdsa, but our recover handler catches it
invalidKey := &ecdsa.PrivateKey{
PublicKey: ecdsa.PublicKey{
Curve: elliptic.P256(),
X: nil,
Y: nil,
},
D: nil,
}
userId := NewId()
_, _, appErr := GenerateTriggerId(userId, invalidKey)
require.NotNil(t, appErr)
assert.Equal(t, "interactive_message.generate_trigger_id.signing_failed", appErr.Id)
assert.Contains(t, appErr.Error(), "invalid signing key")
})
t.Run("should handle ECDSA key with zero D value", func(t *testing.T) {
// Create an ECDSA key with D set to zero (invalid private key)
invalidKey := &ecdsa.PrivateKey{
PublicKey: ecdsa.PublicKey{
Curve: elliptic.P256(),
},
D: big.NewInt(0),
}
userId := NewId()
_, _, appErr := GenerateTriggerId(userId, invalidKey)
require.NotNil(t, appErr)
assert.Equal(t, "interactive_message.generate_trigger_id.signing_failed", appErr.Id)
})
}
func TestTriggerIdDecodeAndVerification(t *testing.T) {
key, err := ecdsa.GenerateKey(elliptic.P256(), rand.Reader)
require.NoError(t, err)
t.Run("should succeed decoding and validation", func(t *testing.T) {
userId := NewId()
clientTriggerId, triggerId, appErr := GenerateTriggerId(userId, key)
require.Nil(t, appErr)
decodedClientTriggerId, decodedUserId, appErr := DecodeAndVerifyTriggerId(triggerId, key, OutgoingIntegrationRequestsDefaultTimeout*time.Second)
assert.Nil(t, appErr)
assert.Equal(t, clientTriggerId, decodedClientTriggerId)
assert.Equal(t, userId, decodedUserId)
})
t.Run("should succeed decoding and validation through request structs", func(t *testing.T) {
actionReq := &PostActionIntegrationRequest{
UserId: NewId(),
}
clientTriggerId, triggerId, appErr := actionReq.GenerateTriggerId(key)
require.Nil(t, appErr)
dialogReq := &OpenDialogRequest{TriggerId: triggerId}
decodedClientTriggerId, decodedUserId, appErr := dialogReq.DecodeAndVerifyTriggerId(key, OutgoingIntegrationRequestsDefaultTimeout*time.Second)
assert.Nil(t, appErr)
assert.Equal(t, clientTriggerId, decodedClientTriggerId)
assert.Equal(t, actionReq.UserId, decodedUserId)
})
t.Run("should fail on base64 decode", func(t *testing.T) {
_, _, appErr := DecodeAndVerifyTriggerId("junk!", key, OutgoingIntegrationRequestsDefaultTimeout*time.Second)
require.NotNil(t, appErr)
assert.Equal(t, "interactive_message.decode_trigger_id.base64_decode_failed", appErr.Id)
})
t.Run("should fail on trigger parsing", func(t *testing.T) {
_, _, appErr := DecodeAndVerifyTriggerId(base64.StdEncoding.EncodeToString([]byte("junk!")), key, OutgoingIntegrationRequestsDefaultTimeout*time.Second)
require.NotNil(t, appErr)
assert.Equal(t, "interactive_message.decode_trigger_id.missing_data", appErr.Id)
})
t.Run("should fail on expired timestamp", func(t *testing.T) {
_, _, appErr := DecodeAndVerifyTriggerId(base64.StdEncoding.EncodeToString([]byte("some-trigger-id:some-user-id:1234567890:junksignature")), key, OutgoingIntegrationRequestsDefaultTimeout*time.Second)
require.NotNil(t, appErr)
assert.Equal(t, "interactive_message.decode_trigger_id.expired", appErr.Id)
})
t.Run("should fail on base64 decoding signature", func(t *testing.T) {
_, _, appErr := DecodeAndVerifyTriggerId(base64.StdEncoding.EncodeToString([]byte("some-trigger-id:some-user-id:12345678900000:junk!")), key, OutgoingIntegrationRequestsDefaultTimeout*time.Second)
require.NotNil(t, appErr)
assert.Equal(t, "interactive_message.decode_trigger_id.base64_decode_failed_signature", appErr.Id)
})
t.Run("should fail on bad signature", func(t *testing.T) {
_, _, appErr := DecodeAndVerifyTriggerId(base64.StdEncoding.EncodeToString([]byte("some-trigger-id:some-user-id:12345678900000:junk")), key, OutgoingIntegrationRequestsDefaultTimeout*time.Second)
require.NotNil(t, appErr)
assert.Equal(t, "interactive_message.decode_trigger_id.signature_decode_failed", appErr.Id)
})
t.Run("should fail on bad key", func(t *testing.T) {
_, triggerId, appErr := GenerateTriggerId(NewId(), key)
require.Nil(t, appErr)
newKey, keyErr := ecdsa.GenerateKey(elliptic.P256(), rand.Reader)
require.NoError(t, keyErr)
_, _, appErr = DecodeAndVerifyTriggerId(triggerId, newKey, OutgoingIntegrationRequestsDefaultTimeout*time.Second)
require.NotNil(t, appErr)
assert.Equal(t, "interactive_message.decode_trigger_id.verify_signature_failed", appErr.Id)
})
}
func TestPostActionIntegrationEquals(t *testing.T) {
t.Run("equal uncomparable types", func(t *testing.T) {
pa1 := &PostAction{
Integration: &PostActionIntegration{
Context: map[string]any{
"a": map[string]any{
"a": 0,
},
},
},
}
pa2 := &PostAction{
Integration: &PostActionIntegration{
Context: map[string]any{
"a": map[string]any{
"a": 0,
},
},
},
}
require.True(t, pa1.Equals(pa2))
})
t.Run("equal comparable types", func(t *testing.T) {
pa1 := &PostAction{
Integration: &PostActionIntegration{
Context: map[string]any{
"a": "test",
},
},
}
pa2 := &PostAction{
Integration: &PostActionIntegration{
Context: map[string]any{
"a": "test",
},
},
}
require.True(t, pa1.Equals(pa2))
})
t.Run("non-equal types", func(t *testing.T) {
pa1 := &PostAction{
Integration: &PostActionIntegration{
Context: map[string]any{
"a": map[string]any{
"a": 0,
},
},
},
}
pa2 := &PostAction{
Integration: &PostActionIntegration{
Context: map[string]any{
"a": "test",
},
},
}
require.False(t, pa1.Equals(pa2))
})
t.Run("nil check in input integration", func(t *testing.T) {
pa1 := &PostAction{
Integration: &PostActionIntegration{},
}
pa2 := &PostAction{
Integration: nil,
}
require.False(t, pa1.Equals(pa2))
})
t.Run("nil check in original integration", func(t *testing.T) {
pa1 := &PostAction{
Integration: nil,
}
pa2 := &PostAction{
Integration: &PostActionIntegration{},
}
require.False(t, pa1.Equals(pa2))
})
t.Run("both nil", func(t *testing.T) {
pa1 := &PostAction{
Integration: nil,
}
pa2 := &PostAction{
Integration: nil,
}
require.True(t, pa1.Equals(pa2))
})
}
func TestPostActionOptions_IsValid(t *testing.T) {
tests := map[string]struct {
options *PostActionOptions
wantErr string
}{
"valid options": {
options: &PostActionOptions{
Text: "Option 1",
Value: "opt1",
},
wantErr: "",
},
"missing text": {
options: &PostActionOptions{
Value: "opt1",
},
wantErr: "text is required",
},
"missing value": {
options: &PostActionOptions{
Text: "Option 1",
},
wantErr: "value is required",
},
}
for name, tc := range tests {
t.Run(name, func(t *testing.T) {
err := tc.options.IsValid()
if tc.wantErr == "" {
assert.NoError(t, err)
} else {
assert.ErrorContains(t, err, tc.wantErr)
}
})
}
}
func TestOpenDialogRequestIsValid(t *testing.T) {
getBaseOpenDialogRequest := func() OpenDialogRequest {
return OpenDialogRequest{
TriggerId: "triggerId",
URL: "http://localhost:8065",
Dialog: Dialog{
CallbackId: "callbackid",
Title: "Some Title",
Elements: []DialogElement{
{
DisplayName: "Element Name",
Name: "element_name",
Type: "text",
Placeholder: "Enter a value",
},
},
SubmitLabel: "Submit",
NotifyOnCancel: false,
State: "somestate",
},
}
}
t.Run("should pass validation", func(t *testing.T) {
request := getBaseOpenDialogRequest()
err := request.IsValid()
assert.NoError(t, err)
})
t.Run("should fail on empty url", func(t *testing.T) {
request := getBaseOpenDialogRequest()
request.URL = ""
err := request.IsValid()
assert.ErrorContains(t, err, "empty URL")
})
t.Run("should fail on empty trigger", func(t *testing.T) {
request := getBaseOpenDialogRequest()
request.TriggerId = ""
err := request.IsValid()
assert.ErrorContains(t, err, "empty trigger id")
})
t.Run("should fail on empty dialog title", func(t *testing.T) {
request := getBaseOpenDialogRequest()
request.Dialog.Title = ""
err := request.IsValid()
assert.ErrorContains(t, err, "invalid dialog title")
})
t.Run("should fail on wrong subtype and long dialog title", func(t *testing.T) {
request := getBaseOpenDialogRequest()
request.Dialog.Elements[0].SubType = "wrong SubType"
request.Dialog.Title = "Very very long Dialog Name"
err := request.IsValid()
assert.ErrorContains(t, err, "invalid subtype")
assert.ErrorContains(t, err, "invalid dialog title")
t.Cleanup(func() {
request.Dialog.Elements[0].SubType = ""
request.Dialog.Title = "Some Title"
})
})
t.Run("should fail on wrong dialog icon url", func(t *testing.T) {
request := getBaseOpenDialogRequest()
request.Dialog.IconURL = "wrong url"
err := request.IsValid()
assert.ErrorContains(t, err, "invalid icon url")
})
t.Run("should pass on empty dialog icon url", func(t *testing.T) {
request := getBaseOpenDialogRequest()
request.Dialog.IconURL = ""
err := request.IsValid()
assert.NoError(t, err)
})
t.Run("should fail on wrong minimal and maximal element length", func(t *testing.T) {
request := getBaseOpenDialogRequest()
request.Dialog.Elements[0].MinLength = 10
request.Dialog.Elements[0].MaxLength = 9
err := request.IsValid()
assert.ErrorContains(t, err, "field is not valid")
assert.ErrorContains(t, err, "min length should be less then max length")
})
t.Run("should fail on wrong element type", func(t *testing.T) {
request := getBaseOpenDialogRequest()
request.Dialog.Elements[0].Type = "wrong type"
err := request.IsValid()
assert.ErrorContains(t, err, "invalid element type")
})
t.Run("should fail on duplicate element name", func(t *testing.T) {
request := getBaseOpenDialogRequest()
request.Dialog.Elements = append(request.Dialog.Elements, DialogElement{
DisplayName: "Radio element name",
Name: "element_name",
Type: "radio",
})
err := request.IsValid()
assert.ErrorContains(t, err, "duplicate dialog element")
})
t.Run("should fail on wrong bool default value", func(t *testing.T) {
request := getBaseOpenDialogRequest()
request.Dialog.Elements = append(request.Dialog.Elements, DialogElement{
DisplayName: "Bool element name",
Name: "bool_element_name",
Type: "bool",
Default: "wrong default",
})
err := request.IsValid()
assert.ErrorContains(t, err, "invalid default of bool")
})
t.Run("should pass on bool default value", func(t *testing.T) {
request := getBaseOpenDialogRequest()
request.Dialog.Elements = append(request.Dialog.Elements, DialogElement{
DisplayName: "Bool element name",
Name: "bool_element_name",
Type: "bool",
Default: "true",
})
err := request.IsValid()
assert.NoError(t, err)
})
t.Run("should fail on wrong select datasource value", func(t *testing.T) {
request := getBaseOpenDialogRequest()
request.Dialog.Elements = append(request.Dialog.Elements, DialogElement{
DisplayName: "Select element name",
Name: "select_element_name",
Type: "select",
DataSource: "wrong DataSource",
})
err := request.IsValid()
assert.ErrorContains(t, err, "invalid data source")
})
t.Run("should fail on wrong select default value, and not fail with nil dereference", func(t *testing.T) {
request := getBaseOpenDialogRequest()
request.Dialog.Elements = append(request.Dialog.Elements, DialogElement{
DisplayName: "Select element name",
Name: "select_element_name",
Type: "select",
DataSource: "",
Default: "default",
Options: []*PostActionOptions{
nil,
},
})
err := request.IsValid()
assert.ErrorContains(t, err, "default value \"default\" doesn't exist in options")
})
t.Run("should fail on wrong radio default value", func(t *testing.T) {
request := getBaseOpenDialogRequest()
request.Dialog.Elements = append(request.Dialog.Elements, DialogElement{
DisplayName: "Radio element name",
Name: "radio_element_name",
Type: "radio",
Default: "default",
Options: []*PostActionOptions{
{
Text: "Text 1",
Value: "value 1",
},
},
})
err := request.IsValid()
assert.ErrorContains(t, err, "default value \"default\" doesn't exist in options")
})
t.Run("should fail on wrong radio default value, and not fail with nil dereference", func(t *testing.T) {
request := getBaseOpenDialogRequest()
request.Dialog.Elements = append(request.Dialog.Elements, DialogElement{
DisplayName: "Radio element name",
Name: "radio_element_name",
Type: "radio",
Default: "default",
Options: []*PostActionOptions{
nil,
},
})
err := request.IsValid()
assert.ErrorContains(t, err, "default value \"default\" doesn't exist in options")
})
t.Run("should pass radio default value", func(t *testing.T) {
request := getBaseOpenDialogRequest()
request.Dialog.Elements = append(request.Dialog.Elements, DialogElement{
DisplayName: "Radio element name",
Name: "radio_element_name",
Type: "radio",
Default: "default",
Options: []*PostActionOptions{
{
Text: "Text 1",
Value: "value 1",
},
{
Text: "Text 2",
Value: "default",
},
},
})
err := request.IsValid()
assert.NoError(t, err)
})
t.Run("should fail on too long text placeholder", func(t *testing.T) {
request := getBaseOpenDialogRequest()
request.Dialog.Elements[0].Placeholder = strings.Repeat("x", 151)
err := request.IsValid()
assert.ErrorContains(t, err, "Placeholder cannot be longer than 150 characters")
})
t.Run("should pass with select element with dynamic data_source", func(t *testing.T) {
request := getBaseOpenDialogRequest()
request.Dialog.Elements = append(request.Dialog.Elements, DialogElement{
DisplayName: "Dynamic data_source",
Name: "dynamic_field",
Type: "select",
DataSource: "dynamic",
DataSourceURL: "https://example.com/api/options",
})
err := request.IsValid()
assert.NoError(t, err)
})
t.Run("should fail dynamic data_source without data_source_url", func(t *testing.T) {
request := getBaseOpenDialogRequest()
request.Dialog.Elements = append(request.Dialog.Elements, DialogElement{
DisplayName: "Dynamic data_source",
Name: "dynamic_field",
Type: "select",
DataSource: "dynamic",
})
err := request.IsValid()
assert.ErrorContains(t, err, "dynamic data_source requires data_source_url")
})
t.Run("should fail dynamic data_source with malformed URL", func(t *testing.T) {
request := getBaseOpenDialogRequest()
request.Dialog.Elements = append(request.Dialog.Elements, DialogElement{
DisplayName: "Dynamic data_source",
Name: "dynamic_field",
Type: "select",
DataSource: "dynamic",
DataSourceURL: "not-a-valid-url",
})
err := request.IsValid()
assert.ErrorContains(t, err, "invalid data_source_url for dynamic select")
})
t.Run("should pass dynamic data_source with HTTP URL", func(t *testing.T) {
request := getBaseOpenDialogRequest()
request.Dialog.Elements = append(request.Dialog.Elements, DialogElement{
DisplayName: "Dynamic data_source",
Name: "dynamic_field",
Type: "select",
DataSource: "dynamic",
DataSourceURL: "http://example.com/api/options",
})
err := request.IsValid()
assert.NoError(t, err)
})
t.Run("should pass dynamic data_source with plugin URL", func(t *testing.T) {
request := getBaseOpenDialogRequest()
request.Dialog.Elements = append(request.Dialog.Elements, DialogElement{
DisplayName: "Dynamic data_source",
Name: "dynamic_field",
Type: "select",
DataSource: "dynamic",
DataSourceURL: "/plugins/myplugin/api/options",
})
err := request.IsValid()
assert.NoError(t, err)
})
t.Run("should fail dynamic data_source with static options", func(t *testing.T) {
request := getBaseOpenDialogRequest()
request.Dialog.Elements = append(request.Dialog.Elements, DialogElement{
DisplayName: "Dynamic data_source",
Name: "dynamic_field",
Type: "select",
DataSource: "dynamic",
DataSourceURL: "https://example.com/api/options",
Options: []*PostActionOptions{
{Text: "Option 1", Value: "opt1"},
},
})
err := request.IsValid()
assert.ErrorContains(t, err, "dynamic select element should not have static options")
})
}
func TestIsValidLookupURL(t *testing.T) {
tests := map[string]struct {
url string
expected bool
}{
"valid HTTPS URL": {
url: "https://example.com/api/lookup",
expected: true,
},
"valid HTTP URL": {
url: "http://example.com/api/lookup",
expected: true,
},
"valid plugin path": {
url: "/plugins/myplugin/lookup",
expected: true,
},
"empty URL": {
url: "",
expected: false,
},
"path traversal attack": {
url: "/plugins/../../../etc/passwd",
expected: false,
},
"double slash in plugin path": {
url: "/plugins//myplugin/lookup",
expected: false,
},
"invalid scheme": {
url: "ftp://example.com/lookup",
expected: false,
},
"relative path": {
url: "relative/path",
expected: false,
},
"localhost HTTPS": {
url: "https://localhost:8080/api/lookup",
expected: true,
},
"localhost HTTP": {
url: "http://localhost:8080/api/lookup",
expected: true,
},
"127.0.0.1 HTTP": {
url: "http://127.0.0.1:8080/api/lookup",
expected: true,
},
"malformed URL": {
url: "not-a-url",
expected: false,
},
}
for name, tc := range tests {
t.Run(name, func(t *testing.T) {
result := IsValidLookupURL(tc.url)
assert.Equal(t, tc.expected, result, "IsValidLookupURL(%q) = %v, want %v", tc.url, result, tc.expected)
})
}
}
func TestDialogSelectOption(t *testing.T) {
t.Run("should create valid option", func(t *testing.T) {
option := DialogSelectOption{
Text: "Test Option",
Value: "test_value",
}
assert.Equal(t, "Test Option", option.Text)
assert.Equal(t, "test_value", option.Value)
})
t.Run("should handle empty values", func(t *testing.T) {
option := DialogSelectOption{
Text: "",
Value: "",
}
assert.Equal(t, "", option.Text)
assert.Equal(t, "", option.Value)
})
}
func TestLookupDialogResponse(t *testing.T) {
t.Run("should create valid response", func(t *testing.T) {
response := LookupDialogResponse{
Items: []DialogSelectOption{
{Text: "Option 1", Value: "value1"},
{Text: "Option 2", Value: "value2"},
},
}
assert.Len(t, response.Items, 2)
assert.Equal(t, "Option 1", response.Items[0].Text)
assert.Equal(t, "value1", response.Items[0].Value)
})
t.Run("should handle empty response", func(t *testing.T) {
response := LookupDialogResponse{
Items: []DialogSelectOption{},
}
assert.Empty(t, response.Items)
})
t.Run("should handle nil items", func(t *testing.T) {
response := LookupDialogResponse{}
assert.Nil(t, response.Items)
})
}
func TestDialogElementMultiSelectValidation(t *testing.T) {
t.Run("should pass with multiselect on select element", func(t *testing.T) {
element := &DialogElement{
DisplayName: "Multi Select Element",
Name: "multi_select",
Type: "select",
MultiSelect: true,
Options: []*PostActionOptions{
{Text: "Option 1", Value: "opt1"},
{Text: "Option 2", Value: "opt2"},
},
}
err := element.IsValid()
assert.NoError(t, err)
})
t.Run("should fail with multiselect on non-select element", func(t *testing.T) {
element := &DialogElement{
DisplayName: "Text Element",
Name: "text_element",
Type: "text",
MultiSelect: true,
}
err := element.IsValid()
assert.ErrorContains(t, err, "multiselect can only be used with select elements, got type \"text\"")
})
t.Run("should fail with multiselect on radio element", func(t *testing.T) {
element := &DialogElement{
DisplayName: "Radio Element",
Name: "radio_element",
Type: "radio",
MultiSelect: true,
Options: []*PostActionOptions{
{Text: "Option 1", Value: "opt1"},
},
}
err := element.IsValid()
assert.ErrorContains(t, err, "multiselect can only be used with select elements, got type \"radio\"")
})
t.Run("should fail with multiselect on bool element", func(t *testing.T) {
element := &DialogElement{
DisplayName: "Bool Element",
Name: "bool_element",
Type: "bool",
MultiSelect: true,
}
err := element.IsValid()
assert.ErrorContains(t, err, "multiselect can only be used with select elements, got type \"bool\"")
})
t.Run("should pass with multiselect and valid comma-separated defaults", func(t *testing.T) {
element := &DialogElement{
DisplayName: "Multi Select Element",
Name: "multi_select",
Type: "select",
MultiSelect: true,
Default: "opt1,opt2",
Options: []*PostActionOptions{
{Text: "Option 1", Value: "opt1"},
{Text: "Option 2", Value: "opt2"},
{Text: "Option 3", Value: "opt3"},
},
}
err := element.IsValid()
assert.NoError(t, err)
})
t.Run("should pass with multiselect and valid spaced comma-separated defaults", func(t *testing.T) {
element := &DialogElement{
DisplayName: "Multi Select Element",
Name: "multi_select",
Type: "select",
MultiSelect: true,
Default: "opt1, opt2, opt3",
Options: []*PostActionOptions{
{Text: "Option 1", Value: "opt1"},
{Text: "Option 2", Value: "opt2"},
{Text: "Option 3", Value: "opt3"},
},
}
err := element.IsValid()
assert.NoError(t, err)
})
t.Run("should fail with multiselect and invalid default value not in options", func(t *testing.T) {
element := &DialogElement{
DisplayName: "Multi Select Element",
Name: "multi_select",
Type: "select",
MultiSelect: true,
Default: "opt1,invalid_opt",
Options: []*PostActionOptions{
{Text: "Option 1", Value: "opt1"},
{Text: "Option 2", Value: "opt2"},
},
}
err := element.IsValid()
assert.ErrorContains(t, err, "multiselect default value \"opt1,invalid_opt\" contains values not in options")
})
t.Run("should pass with multiselect and empty default", func(t *testing.T) {
element := &DialogElement{
DisplayName: "Multi Select Element",
Name: "multi_select",
Type: "select",
MultiSelect: true,
Default: "",
Options: []*PostActionOptions{
{Text: "Option 1", Value: "opt1"},
{Text: "Option 2", Value: "opt2"},
},
}
err := element.IsValid()
assert.NoError(t, err)
})
t.Run("should pass with multiselect and data source", func(t *testing.T) {
element := &DialogElement{
DisplayName: "Multi Select Element",
Name: "multi_select",
Type: "select",
MultiSelect: true,
DataSource: "users",
}
err := element.IsValid()
assert.NoError(t, err)
})
t.Run("should fail with single-select and comma-separated default values", func(t *testing.T) {
element := &DialogElement{
DisplayName: "Single Select Element",
Name: "single_select",
Type: "select",
MultiSelect: false,
Default: "opt1,opt2",
Options: []*PostActionOptions{
{Text: "Option 1", Value: "opt1"},
{Text: "Option 2", Value: "opt2"},
},
}
err := element.IsValid()
assert.ErrorContains(t, err, "default value \"opt1,opt2\" doesn't exist in options")
})
}
func TestIsMultiSelectDefaultInOptions(t *testing.T) {
options := []*PostActionOptions{
{Text: "Option 1", Value: "opt1"},
{Text: "Option 2", Value: "opt2"},
{Text: "Option 3", Value: "opt3"},
{Text: "Option 4", Value: "opt4"},
}
t.Run("should return true for empty default", func(t *testing.T) {
result := isMultiSelectDefaultInOptions("", options)
assert.True(t, result)
})
t.Run("should return true for single valid default", func(t *testing.T) {
result := isMultiSelectDefaultInOptions("opt1", options)
assert.True(t, result)
})
t.Run("should return true for multiple valid defaults", func(t *testing.T) {
result := isMultiSelectDefaultInOptions("opt1,opt2", options)
assert.True(t, result)
})
t.Run("should return true for multiple valid defaults with spaces", func(t *testing.T) {
result := isMultiSelectDefaultInOptions("opt1, opt2, opt3", options)
assert.True(t, result)
})
t.Run("should return true for all valid defaults", func(t *testing.T) {
result := isMultiSelectDefaultInOptions("opt1,opt2,opt3,opt4", options)
assert.True(t, result)
})
t.Run("should return false for single invalid default", func(t *testing.T) {
result := isMultiSelectDefaultInOptions("invalid", options)
assert.False(t, result)
})
t.Run("should return false for mixed valid and invalid defaults", func(t *testing.T) {
result := isMultiSelectDefaultInOptions("opt1,invalid,opt2", options)
assert.False(t, result)
})
t.Run("should return false for all invalid defaults", func(t *testing.T) {
result := isMultiSelectDefaultInOptions("invalid1,invalid2", options)
assert.False(t, result)
})
t.Run("should handle empty values in comma-separated string", func(t *testing.T) {
result := isMultiSelectDefaultInOptions("opt1,,opt2", options)
assert.True(t, result)
})
t.Run("should handle only commas", func(t *testing.T) {
result := isMultiSelectDefaultInOptions(",,", options)
assert.True(t, result)
})
t.Run("should return true for single comma", func(t *testing.T) {
result := isMultiSelectDefaultInOptions(",", options)
assert.True(t, result)
})
t.Run("should handle nil options", func(t *testing.T) {
result := isMultiSelectDefaultInOptions("opt1", nil)
assert.False(t, result)
})
t.Run("should handle options with nil entries", func(t *testing.T) {
optionsWithNil := []*PostActionOptions{
{Text: "Option 1", Value: "opt1"},
nil,
{Text: "Option 2", Value: "opt2"},
}
result := isMultiSelectDefaultInOptions("opt1,opt2", optionsWithNil)
assert.True(t, result)
})
t.Run("should return false when value matches nil option", func(t *testing.T) {
optionsWithNil := []*PostActionOptions{
{Text: "Option 1", Value: "opt1"},
nil,
}
result := isMultiSelectDefaultInOptions("opt1,opt2", optionsWithNil)
assert.False(t, result)
})
}
func TestSubmitDialogResponse_IsValid(t *testing.T) {
validDialog := &Dialog{
Title: "Test Dialog",
}
tests := map[string]struct {
response *SubmitDialogResponse
wantErr string
}{
"error takes precedence - with error field": {
response: &SubmitDialogResponse{
Error: "something went wrong",
Type: "invalid_type",
Form: validDialog,
},
wantErr: "",
},
"error takes precedence - with errors field": {
response: &SubmitDialogResponse{
Errors: map[string]string{"field1": "required"},
Type: "invalid_type",
Form: validDialog,
},
wantErr: "",
},
"valid empty type with no form": {
response: &SubmitDialogResponse{
Type: "",
},
wantErr: "",
},
"valid ok type with no form": {
response: &SubmitDialogResponse{
Type: "ok",
},
wantErr: "",
},
"valid navigate type with no form": {
response: &SubmitDialogResponse{
Type: "navigate",
},
wantErr: "",
},
"valid form type with valid form": {
response: &SubmitDialogResponse{
Type: "form",
Form: validDialog,
},
wantErr: "",
},
"invalid empty type with form": {
response: &SubmitDialogResponse{
Type: "",
Form: validDialog,
},
wantErr: "form field must be nil for type \"\"",
},
"invalid ok type with form": {
response: &SubmitDialogResponse{
Type: "ok",
Form: validDialog,
},
wantErr: "form field must be nil for type \"ok\"",
},
"invalid navigate type with form": {
response: &SubmitDialogResponse{
Type: "navigate",
Form: validDialog,
},
wantErr: "form field must be nil for type \"navigate\"",
},
"invalid form type with no form": {
response: &SubmitDialogResponse{
Type: "form",
},
wantErr: "form field is required for form type",
},
"invalid form type with invalid form": {
response: &SubmitDialogResponse{
Type: "form",
Form: &Dialog{}, // Invalid dialog
},
wantErr: "invalid form: 1 error occurred:\n\t* invalid dialog title \"\"",
},
"invalid type": {
response: &SubmitDialogResponse{
Type: "invalid",
},
wantErr: "invalid type \"invalid\", must be one of: empty, ok, form, navigate",
},
}
for name, tt := range tests {
t.Run(name, func(t *testing.T) {
err := tt.response.IsValid()
if tt.wantErr == "" {
assert.NoError(t, err)
} else {
assert.Error(t, err)
assert.Contains(t, err.Error(), tt.wantErr)
}
})
}
}
func TestValidateDateFormat(t *testing.T) {
tests := []struct {
name string
input string
expectError bool
}{
{"valid YYYY-MM-DD format", "2025-01-15", false},
{"valid date with leading zeros", "2025-01-01", false},
{"valid leap year date", "2024-02-29", false},
{"invalid format missing day", "2025-01", true},
{"invalid format with slashes", "2025/01/15", true},
{"invalid month", "2025-13-01", true},
{"invalid day", "2025-01-32", true},
{"invalid leap year", "2023-02-29", true},
{"empty string", "", false}, // Empty string is valid (no error)
{"partial date", "2025", true},
{"valid datetime format (should extract date)", "2025-01-15T10:30:00", true},
{"valid datetime with timezone", "2025-01-15T10:30:00Z", true},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
err := validateDateFormat(tt.input)
if tt.expectError {
assert.Error(t, err)
} else {
assert.NoError(t, err)
}
})
}
}
func TestValidateDateTimeFormat(t *testing.T) {
tests := []struct {
name string
input string
expectError bool
}{
{"valid RFC3339 format", "2025-01-15T10:30:00Z", false},
{"valid with timezone offset", "2025-01-15T10:30:00-05:00", false},
{"valid with positive timezone", "2025-01-15T10:30:00+02:00", false},
{"valid with milliseconds", "2025-01-15T10:30:00.123Z", false},
{"valid format without timezone", "2025-01-15T10:30:00", false},
{"valid format without seconds", "2025-01-15T10:30", false},
{"invalid date part", "2025-13-01T10:30:00Z", true},
{"invalid time part", "2025-01-15T25:30:00Z", true},
{"invalid timezone format", "2025-01-15T10:30:00GMT", true},
{"date only format", "2025-01-15", true},
{"empty string", "", false}, // Empty string is valid (no error)
{"invalid format with space", "2025-01-15 10:30:00", true},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
err := validateDateTimeFormat(tt.input)
if tt.expectError {
assert.Error(t, err)
} else {
assert.NoError(t, err)
}
})
}
}
func TestDialogElementDateTimeValidation(t *testing.T) {
t.Run("should validate DialogElement with date/datetime type", func(t *testing.T) {
element := DialogElement{
DisplayName: "Test Date",
Name: "test_date",
Type: "date",
MinDate: "2025-01-01",
MaxDate: "2025-12-31",
Optional: false,
}
err := element.IsValid()
assert.NoError(t, err)
})
t.Run("should validate DialogElement with datetime type and time properties", func(t *testing.T) {
element := DialogElement{
DisplayName: "Test DateTime",
Name: "test_datetime",
Type: "datetime",
MinDate: "2025-01-01",
MaxDate: "2025-12-31",
TimeInterval: 30,
Optional: false,
}
err := element.IsValid()
assert.NoError(t, err)
})
t.Run("should reject DialogElement with invalid min_date", func(t *testing.T) {
element := DialogElement{
DisplayName: "Test Date",
Name: "test_date",
Type: "date",
MinDate: "invalid-date",
Optional: false,
}
err := element.IsValid()
assert.Error(t, err)
assert.Contains(t, err.Error(), "invalid date format")
})
t.Run("should reject DialogElement with invalid time_interval", func(t *testing.T) {
element := DialogElement{
DisplayName: "Test DateTime",
Name: "test_datetime",
Type: "datetime",
TimeInterval: -1, // Invalid
Optional: false,
}
err := element.IsValid()
assert.Error(t, err)
assert.Contains(t, err.Error(), "time_interval")
})
t.Run("should reject DialogElement with time_interval that is not a divisor of 1440", func(t *testing.T) {
element := DialogElement{
DisplayName: "Test DateTime",
Name: "test_datetime",
Type: "datetime",
TimeInterval: 729, // Invalid - not a divisor of 1440
Optional: false,
}
err := element.IsValid()
assert.Error(t, err)
assert.Contains(t, err.Error(), "divisor of 1440")
})
t.Run("should accept DialogElement with valid time_interval divisors", func(t *testing.T) {
validIntervals := []int{1, 2, 3, 4, 5, 6, 8, 10, 12, 15, 20, 24, 30, 40, 60, 72, 90, 120, 180, 240, 360, 480, 720, 1440}
for _, interval := range validIntervals {
element := DialogElement{
DisplayName: "Test DateTime",
Name: "test_datetime",
Type: "datetime",
TimeInterval: interval,
Optional: false,
}
err := element.IsValid()
assert.NoError(t, err, "time_interval %d should be valid", interval)
}
})
t.Run("should reject DialogElement with invalid time_interval non-divisors", func(t *testing.T) {
invalidIntervals := []int{7, 11, 13, 17, 23, 25, 33, 37, 50, 55, 70, 100, 300, 500, 729, 1000}
for _, interval := range invalidIntervals {
element := DialogElement{
DisplayName: "Test DateTime",
Name: "test_datetime",
Type: "datetime",
TimeInterval: interval,
Optional: false,
}
err := element.IsValid()
assert.Error(t, err, "time_interval %d should be invalid", interval)
assert.Contains(t, err.Error(), "divisor of 1440")
}
})
t.Run("should use default time_interval of 60 minutes when zero", func(t *testing.T) {
// Valid with default 60-minute interval
element := DialogElement{
DisplayName: "Test DateTime",
Name: "test_datetime",
Type: "datetime",
TimeInterval: DefaultTimeIntervalMinutes,
Optional: false,
}
err := element.IsValid()
assert.NoError(t, err)
// Invalid with default 60-minute interval
element = DialogElement{
DisplayName: "Test DateTime",
Name: "test_datetime",
Type: "datetime",
TimeInterval: 0, // Should use default of 60
Optional: false,
}
err = element.IsValid()
assert.Error(t, err)
assert.Contains(t, err.Error(), "time_interval of 0 will be reset to default")
})
}