mattermost/server/cmd/mmctl/commands/command.go
Ben Schumacher ae35dd93a9
[MM-64844] Remove deprecated mmctl commands and flags (#33435)
* Remove deprecated mmctl commands and flags

  Remove the following deprecated commands:
  - `channel add` (use `channel users add`)
  - `channel remove` (use `channel users remove`)
  - `channel restore` (use `channel unarchive`)
  - `channel make-private` (use `channel modify --private`)
  - `command delete` (use `command archive`)
  - `permissions show` (use `permissions role show`)

  Remove the following deprecated flags:
  - `--config-path` (use `--config`)
  - `--display_name` (use `--display-name`)
  - `--system_admin` (use `--system-admin`)
  - `--email_verified` (use `--email-verified`)
  - `--attachments` from export create
  - `--resume` from export download
2025-07-18 11:08:29 +02:00

343 lines
11 KiB
Go

// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
// See LICENSE.txt for license information.
package commands
import (
"context"
"errors"
"fmt"
"net/http"
"strings"
"github.com/hashicorp/go-multierror"
"github.com/mattermost/mattermost/server/public/model"
"github.com/spf13/cobra"
"github.com/mattermost/mattermost/server/v8/cmd/mmctl/client"
"github.com/mattermost/mattermost/server/v8/cmd/mmctl/printer"
)
var CommandCmd = &cobra.Command{
Use: "command",
Short: "Management of slash commands",
}
var CommandCreateCmd = &cobra.Command{
Use: "create [team]",
Short: "Create a custom slash command",
Long: `Create a custom slash command for the specified team.`,
Args: cobra.MinimumNArgs(1),
Example: ` command create myteam --title MyCommand --description "My Command Description" --trigger-word mycommand --url http://localhost:8000/my-slash-handler --creator myusername --response-username my-bot-username --icon http://localhost:8000/my-slash-handler-bot-icon.png --autocomplete --post`,
RunE: withClient(createCommandCmdF),
}
var CommandListCmd = &cobra.Command{
Use: "list [teams]",
Short: "List all commands on specified teams.",
Long: `List all commands on specified teams.`,
Example: ` command list myteam`,
RunE: withClient(listCommandCmdF),
}
var CommandArchiveCmd = &cobra.Command{
Use: "archive [commandID]",
Short: "Archive a slash command",
Long: `Archive a slash command. Commands can be specified by command ID.`,
Example: ` command archive commandID`,
Args: cobra.ExactArgs(1),
RunE: withClient(archiveCommandCmdF),
}
var CommandModifyCmd = &cobra.Command{
Use: "modify [commandID]",
Short: "Modify a slash command",
Long: `Modify a slash command. Commands can be specified by command ID.`,
Args: cobra.ExactArgs(1),
Example: ` command modify commandID --title MyModifiedCommand --description "My Modified Command Description" --trigger-word mycommand --url http://localhost:8000/my-slash-handler --creator myusername --response-username my-bot-username --icon http://localhost:8000/my-slash-handler-bot-icon.png --autocomplete --post`,
RunE: withClient(modifyCommandCmdF),
}
var CommandMoveCmd = &cobra.Command{
Use: "move [team] [commandID]",
Short: "Move a slash command to a different team",
Long: `Move a slash command to a different team. Commands can be specified by command ID.`,
Args: cobra.ExactArgs(2),
Example: ` command move newteam commandID`,
RunE: withClient(moveCommandCmdF),
}
var CommandShowCmd = &cobra.Command{
Use: "show [commandID]",
Short: "Show a custom slash command",
Long: `Show a custom slash command. Commands can be specified by command ID. Returns command ID, team ID, trigger word, display name and creator username.`,
Args: cobra.ExactArgs(1),
Example: ` command show commandID`,
RunE: withClient(showCommandCmdF),
}
func addCommandFieldsFlags(cmd *cobra.Command) {
cmd.Flags().String("title", "", "Command Title")
cmd.Flags().String("description", "", "Command Description")
cmd.Flags().String("trigger-word", "", "Command Trigger Word (required)")
cmd.Flags().String("url", "", "Command Callback URL (required)")
cmd.Flags().String("creator", "", "Command Creator's username, email or id (required)")
cmd.Flags().String("response-username", "", "Command Response Username")
cmd.Flags().String("icon", "", "Command Icon URL")
cmd.Flags().Bool("autocomplete", false, "Show Command in autocomplete list")
cmd.Flags().String("autocompleteDesc", "", "Short Command Description for autocomplete list")
cmd.Flags().String("autocompleteHint", "", "Command Arguments displayed as help in autocomplete list")
cmd.Flags().Bool("post", false, "Use POST method for Callback URL")
}
func init() {
cmds := []*cobra.Command{CommandCreateCmd, CommandModifyCmd}
for _, cmd := range cmds {
addCommandFieldsFlags(cmd)
}
_ = CommandCreateCmd.MarkFlagRequired("trigger-word")
_ = CommandCreateCmd.MarkFlagRequired("url")
_ = CommandCreateCmd.MarkFlagRequired("creator")
CommandCmd.AddCommand(
CommandCreateCmd,
CommandListCmd,
CommandModifyCmd,
CommandMoveCmd,
CommandShowCmd,
CommandArchiveCmd,
)
RootCmd.AddCommand(CommandCmd)
}
func createCommandCmdF(c client.Client, cmd *cobra.Command, args []string) error {
printer.SetSingle(true)
team := getTeamFromTeamArg(c, args[0])
if team == nil {
return errors.New("unable to find team '" + args[0] + "'")
}
// get the creator
creator, _ := cmd.Flags().GetString("creator")
user := getUserFromUserArg(c, creator)
if user == nil {
return errors.New("unable to find user '" + creator + "'")
}
title, _ := cmd.Flags().GetString("title")
description, _ := cmd.Flags().GetString("description")
trigger, _ := cmd.Flags().GetString("trigger-word")
if strings.HasPrefix(trigger, "/") {
return errors.New("a trigger word cannot begin with a /")
}
if strings.Contains(trigger, " ") {
return errors.New("a trigger word must not contain spaces")
}
url, _ := cmd.Flags().GetString("url")
responseUsername, _ := cmd.Flags().GetString("response-username")
icon, _ := cmd.Flags().GetString("icon")
autocomplete, _ := cmd.Flags().GetBool("autocomplete")
autocompleteDesc, _ := cmd.Flags().GetString("autocompleteDesc")
autocompleteHint, _ := cmd.Flags().GetString("autocompleteHint")
post, errp := cmd.Flags().GetBool("post")
method := "P"
if errp != nil || !post {
method = "G"
}
newCommand := &model.Command{
CreatorId: user.Id,
TeamId: team.Id,
Trigger: trigger,
Method: method,
Username: responseUsername,
IconURL: icon,
AutoComplete: autocomplete,
AutoCompleteDesc: autocompleteDesc,
AutoCompleteHint: autocompleteHint,
DisplayName: title,
Description: description,
URL: url,
}
createdCommand, _, err := c.CreateCommand(context.TODO(), newCommand)
if err != nil {
return errors.New("unable to create command '" + newCommand.DisplayName + "'. " + err.Error())
}
printer.PrintT("created command {{.DisplayName}}", createdCommand)
return nil
}
func listCommandCmdF(c client.Client, cmd *cobra.Command, args []string) error {
var teams []*model.Team
if len(args) < 1 {
teamList, err := getPages(func(page, numPerPage int, etag string) ([]*model.Team, *model.Response, error) {
return c.GetAllTeams(context.TODO(), etag, page, numPerPage)
}, DefaultPageSize)
if err != nil {
return err
}
teams = teamList
} else {
teams = getTeamsFromTeamArgs(c, args)
}
var errs *multierror.Error
for i, team := range teams {
if team == nil {
printer.PrintError("Unable to find team '" + args[i] + "'")
errs = multierror.Append(errs, fmt.Errorf("unable to find team '%s'", args[i]))
continue
}
commands, _, err := c.ListCommands(context.TODO(), team.Id, true)
if err != nil {
printer.PrintError("Unable to list commands for '" + team.Id + "'")
errs = multierror.Append(errs, fmt.Errorf("unable to list commands for '%s': %w", team.Id, err))
continue
}
for _, command := range commands {
printer.PrintT("{{.Id}}: {{.DisplayName}} (team: "+team.Name+")", command)
}
}
return errs.ErrorOrNil()
}
func archiveCommandCmdF(c client.Client, cmd *cobra.Command, args []string) error {
resp, err := c.DeleteCommand(context.TODO(), args[0])
if err != nil {
return errors.New("Unable to archive command '" + args[0] + "' error: " + err.Error())
}
if resp.StatusCode == http.StatusOK {
printer.PrintT("Status: {{.status}}", map[string]any{"status": "ok"})
} else {
printer.PrintT("Status: {{.status}}", map[string]any{"status": "error"})
}
return nil
}
func modifyCommandCmdF(c client.Client, cmd *cobra.Command, args []string) error {
printer.SetSingle(true)
command := getCommandFromCommandArg(c, args[0])
if command == nil {
return fmt.Errorf("unable to find command '%s'", args[0])
}
flags := cmd.Flags()
if flags.Changed("title") {
command.DisplayName, _ = flags.GetString("title")
}
if flags.Changed("description") {
command.Description, _ = flags.GetString("description")
}
if flags.Changed("trigger-word") {
trigger, _ := flags.GetString("trigger-word")
if strings.HasPrefix(trigger, "/") {
return errors.New("a trigger word cannot begin with a /")
}
if strings.Contains(trigger, " ") {
return errors.New("a trigger word must not contain spaces")
}
command.Trigger = trigger
}
if flags.Changed("url") {
command.URL, _ = flags.GetString("url")
}
if flags.Changed("creator") {
creator, _ := flags.GetString("creator")
user := getUserFromUserArg(c, creator)
if user == nil {
return fmt.Errorf("unable to find user '%s'", creator)
}
command.CreatorId = user.Id
}
if flags.Changed("response-username") {
command.Username, _ = flags.GetString("response-username")
}
if flags.Changed("icon") {
command.IconURL, _ = flags.GetString("icon")
}
if flags.Changed("autocomplete") {
command.AutoComplete, _ = flags.GetBool("autocomplete")
}
if flags.Changed("autocompleteDesc") {
command.AutoCompleteDesc, _ = flags.GetString("autocompleteDesc")
}
if flags.Changed("autocompleteHint") {
command.AutoCompleteHint, _ = flags.GetString("autocompleteHint")
}
if flags.Changed("post") {
post, _ := flags.GetBool("post")
if post {
command.Method = "P"
} else {
command.Method = "G"
}
}
modifiedCommand, _, err := c.UpdateCommand(context.TODO(), command)
if err != nil {
return fmt.Errorf("unable to modify command '%s': %w", command.DisplayName, err)
}
printer.PrintT("modified command {{.DisplayName}}", modifiedCommand)
return nil
}
func moveCommandCmdF(c client.Client, cmd *cobra.Command, args []string) error {
printer.SetSingle(true)
newTeam := getTeamFromTeamArg(c, args[0])
if newTeam == nil {
return fmt.Errorf("unable to find team '%s'", args[0])
}
command := getCommandFromCommandArg(c, args[1])
if command == nil {
return fmt.Errorf("unable to find command '%s'", args[1])
}
resp, err := c.MoveCommand(context.TODO(), newTeam.Id, command.Id)
if err != nil {
return fmt.Errorf("unable to move command '%s': %w", command.Id, err)
}
if resp.StatusCode == http.StatusOK {
printer.PrintT("Status: {{.status}}", map[string]any{"status": "ok"})
} else {
printer.PrintT("Status: {{.status}}", map[string]any{"status": "error"})
}
return nil
}
func showCommandCmdF(c client.Client, cmd *cobra.Command, args []string) error {
printer.SetSingle(true)
command := getCommandFromCommandArg(c, args[0])
if command == nil {
return fmt.Errorf("unable to find command '%s'", args[0])
}
template :=
`teamId: {{.TeamId}}
title: {{.DisplayName}}
description: {{.Description}}
trigger-word: {{.Trigger}}
URL: {{.URL}}
creatorId: {{.CreatorId}}
response-username: {{.Username}}
iconURL: {{.IconURL}}
autoComplete: {{.AutoComplete}}
autoCompleteDesc: {{.AutoCompleteDesc}}
autoCompleteHint: {{.AutoCompleteHint}}
method: {{.Method}}`
printer.PrintT(template, command)
return nil
}