mirror of
https://github.com/diamondburned/arikawa.git
synced 2025-01-21 03:57:26 +00:00
521 lines
12 KiB
Go
521 lines
12 KiB
Go
package bot
|
|
|
|
import (
|
|
"reflect"
|
|
"strings"
|
|
|
|
"github.com/diamondburned/arikawa/api"
|
|
"github.com/diamondburned/arikawa/discord"
|
|
"github.com/diamondburned/arikawa/gateway"
|
|
"github.com/pkg/errors"
|
|
)
|
|
|
|
var (
|
|
typeMessageCreate = reflect.TypeOf((*gateway.MessageCreateEvent)(nil))
|
|
|
|
typeString = reflect.TypeOf("")
|
|
typeEmbed = reflect.TypeOf((*discord.Embed)(nil))
|
|
typeSend = reflect.TypeOf((*api.SendMessageData)(nil))
|
|
|
|
typeSubcmd = reflect.TypeOf((*Subcommand)(nil))
|
|
|
|
typeIError = reflect.TypeOf((*error)(nil)).Elem()
|
|
typeIManP = reflect.TypeOf((*ManualParser)(nil)).Elem()
|
|
typeICusP = reflect.TypeOf((*CustomParser)(nil)).Elem()
|
|
typeIParser = reflect.TypeOf((*Parser)(nil)).Elem()
|
|
typeIUsager = reflect.TypeOf((*Usager)(nil)).Elem()
|
|
typeSetupFn = func() reflect.Type {
|
|
method, _ := reflect.TypeOf((*CanSetup)(nil)).
|
|
Elem().
|
|
MethodByName("Setup")
|
|
return method.Type
|
|
}()
|
|
)
|
|
|
|
// HelpUnderline formats command arguments with an underline, similar to
|
|
// manpages.
|
|
var HelpUnderline = true
|
|
|
|
func underline(word string) string {
|
|
if HelpUnderline {
|
|
return "__" + word + "__"
|
|
}
|
|
return word
|
|
}
|
|
|
|
// Subcommand is any form of command, which could be a top-level command or a
|
|
// subcommand.
|
|
//
|
|
// Allowed method signatures
|
|
//
|
|
// These are the acceptable function signatures that would be parsed as commands
|
|
// or events. A return type <T> implies that return value will be ignored.
|
|
//
|
|
// func(*gateway.MessageCreateEvent, ...) (string, error)
|
|
// func(*gateway.MessageCreateEvent, ...) (*discord.Embed, error)
|
|
// func(*gateway.MessageCreateEvent, ...) (*api.SendMessageData, error)
|
|
// func(*gateway.MessageCreateEvent, ...) (T, error)
|
|
// func(*gateway.MessageCreateEvent, ...) error
|
|
// func(*gateway.MessageCreateEvent, ...)
|
|
// func(<AnyEvent>) (T, error)
|
|
// func(<AnyEvent>) error
|
|
// func(<AnyEvent>)
|
|
//
|
|
type Subcommand struct {
|
|
Description string
|
|
|
|
// Raw struct name, including the flag (only filled for actual subcommands,
|
|
// will be empty for Context):
|
|
StructName string
|
|
// Parsed command name:
|
|
Command string
|
|
|
|
// struct flags
|
|
Flag NameFlag
|
|
|
|
// SanitizeMessage is executed on the message content if the method returns
|
|
// a string content or a SendMessageData.
|
|
SanitizeMessage func(content string) string
|
|
|
|
// QuietUnknownCommand, if true, will not make the bot reply with an unknown
|
|
// command error into the chat. If this is set in Context, it will apply to
|
|
// all other subcommands.
|
|
QuietUnknownCommand bool
|
|
|
|
// Commands can actually return either a string, an embed, or a
|
|
// SendMessageData, with error as the second argument.
|
|
|
|
// All registered command contexts:
|
|
Commands []*CommandContext
|
|
Events []*CommandContext
|
|
|
|
// Middleware command contexts:
|
|
mwMethods []*CommandContext
|
|
|
|
// Plumb nameflag, use Commands[0] if true.
|
|
plumb bool
|
|
|
|
// Directly to struct
|
|
cmdValue reflect.Value
|
|
cmdType reflect.Type
|
|
|
|
// Pointer value
|
|
ptrValue reflect.Value
|
|
ptrType reflect.Type
|
|
|
|
// command interface as reference
|
|
command interface{}
|
|
}
|
|
|
|
// CommandContext is an internal struct containing fields to make this library
|
|
// work. As such, they're all unexported. Description, however, is exported for
|
|
// editing, and may be used to generate more informative help messages.
|
|
type CommandContext struct {
|
|
Description string
|
|
Flag NameFlag
|
|
|
|
MethodName string
|
|
Command string // empty if Plumb
|
|
|
|
// Hidden is true if the method has a hidden nameflag.
|
|
Hidden bool
|
|
|
|
value reflect.Value // Func
|
|
event reflect.Type // gateway.*Event
|
|
method reflect.Method
|
|
|
|
Arguments []Argument
|
|
}
|
|
|
|
// CanSetup is used for subcommands to change variables, such as Description.
|
|
// This method will be triggered when InitCommands is called, which is during
|
|
// New for Context and during RegisterSubcommand for subcommands.
|
|
type CanSetup interface {
|
|
// Setup should panic when it has an error.
|
|
Setup(*Subcommand)
|
|
}
|
|
|
|
func (cctx *CommandContext) Usage() []string {
|
|
if len(cctx.Arguments) == 0 {
|
|
return nil
|
|
}
|
|
|
|
var arguments = make([]string, len(cctx.Arguments))
|
|
for i, arg := range cctx.Arguments {
|
|
arguments[i] = arg.String
|
|
}
|
|
|
|
return arguments
|
|
}
|
|
|
|
// NewSubcommand is used to make a new subcommand. You usually wouldn't call
|
|
// this function, but instead use (*Context).RegisterSubcommand().
|
|
func NewSubcommand(cmd interface{}) (*Subcommand, error) {
|
|
var sub = Subcommand{
|
|
command: cmd,
|
|
SanitizeMessage: func(c string) string {
|
|
return c
|
|
},
|
|
}
|
|
|
|
if err := sub.reflectCommands(); err != nil {
|
|
return nil, errors.Wrap(err, "Failed to reflect commands")
|
|
}
|
|
|
|
if err := sub.parseCommands(); err != nil {
|
|
return nil, errors.Wrap(err, "Failed to parse commands")
|
|
}
|
|
|
|
return &sub, nil
|
|
}
|
|
|
|
// NeedsName sets the name for this subcommand. Like InitCommands, this
|
|
// shouldn't be called at all, rather you should use RegisterSubcommand.
|
|
func (sub *Subcommand) NeedsName() {
|
|
sub.StructName = sub.cmdType.Name()
|
|
|
|
flag, name := ParseFlag(sub.StructName)
|
|
|
|
if !flag.Is(Raw) {
|
|
name = lowerFirstLetter(name)
|
|
}
|
|
|
|
sub.Command = name
|
|
sub.Flag = flag
|
|
}
|
|
|
|
// FindCommand finds the command. Nil is returned if nothing is found. It's a
|
|
// better idea to not handle nil, as they would become very subtle bugs.
|
|
func (sub *Subcommand) FindCommand(methodName string) *CommandContext {
|
|
for _, c := range sub.Commands {
|
|
if c.MethodName != methodName {
|
|
continue
|
|
}
|
|
return c
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// ChangeCommandInfo changes the matched methodName's Command and Description.
|
|
// Empty means unchanged. The returned bool is true when the method is found.
|
|
func (sub *Subcommand) ChangeCommandInfo(methodName, cmd, desc string) bool {
|
|
for _, c := range sub.Commands {
|
|
if c.MethodName != methodName {
|
|
continue
|
|
}
|
|
|
|
if cmd != "" {
|
|
c.Command = cmd
|
|
}
|
|
if desc != "" {
|
|
c.Description = desc
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
func (sub *Subcommand) Help(indent string, hideAdmin bool) string {
|
|
if sub.Flag.Is(AdminOnly) && hideAdmin {
|
|
return ""
|
|
}
|
|
|
|
// The header part:
|
|
var header string
|
|
|
|
if sub.Command != "" {
|
|
header += "**" + sub.Command + "**"
|
|
}
|
|
|
|
if sub.Description != "" {
|
|
if header != "" {
|
|
header += ": "
|
|
}
|
|
|
|
header += sub.Description
|
|
}
|
|
|
|
header += "\n"
|
|
|
|
// The commands part:
|
|
var commands = ""
|
|
|
|
for i, cmd := range sub.Commands {
|
|
if cmd.Flag.Is(AdminOnly) && hideAdmin {
|
|
continue
|
|
}
|
|
|
|
switch {
|
|
case sub.Command != "" && cmd.Command != "":
|
|
commands += indent + sub.Command + " " + cmd.Command
|
|
case sub.Command != "":
|
|
commands += indent + sub.Command
|
|
default:
|
|
commands += indent + cmd.Command
|
|
}
|
|
|
|
// Write the usages first.
|
|
for _, usage := range cmd.Usage() {
|
|
commands += " " + underline(usage)
|
|
}
|
|
|
|
// Write the description if there's any.
|
|
if cmd.Description != "" {
|
|
commands += ": " + cmd.Description
|
|
}
|
|
|
|
// Add a new line if this isn't the last command.
|
|
if i != len(sub.Commands)-1 {
|
|
commands += "\n"
|
|
}
|
|
}
|
|
|
|
if commands == "" {
|
|
return ""
|
|
}
|
|
|
|
return header + commands
|
|
}
|
|
|
|
func (sub *Subcommand) reflectCommands() error {
|
|
t := reflect.TypeOf(sub.command)
|
|
v := reflect.ValueOf(sub.command)
|
|
|
|
if t.Kind() != reflect.Ptr {
|
|
return errors.New("sub is not a pointer")
|
|
}
|
|
|
|
// Set the pointer fields
|
|
sub.ptrValue = v
|
|
sub.ptrType = t
|
|
|
|
ts := t.Elem()
|
|
vs := v.Elem()
|
|
|
|
if ts.Kind() != reflect.Struct {
|
|
return errors.New("sub is not pointer to struct")
|
|
}
|
|
|
|
// Set the struct fields
|
|
sub.cmdValue = vs
|
|
sub.cmdType = ts
|
|
|
|
return nil
|
|
}
|
|
|
|
// InitCommands fills a Subcommand with a context. This shouldn't be called at
|
|
// all, rather you should use the RegisterSubcommand method of a Context.
|
|
func (sub *Subcommand) InitCommands(ctx *Context) error {
|
|
// Start filling up a *Context field
|
|
if err := sub.fillStruct(ctx); err != nil {
|
|
return err
|
|
}
|
|
|
|
// See if struct implements CanSetup:
|
|
if v, ok := sub.command.(CanSetup); ok {
|
|
v.Setup(sub)
|
|
}
|
|
|
|
// Finalize the subcommand:
|
|
for _, cmd := range sub.Commands {
|
|
// Inherit parent's flags
|
|
cmd.Flag |= sub.Flag
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (sub *Subcommand) fillStruct(ctx *Context) error {
|
|
for i := 0; i < sub.cmdValue.NumField(); i++ {
|
|
field := sub.cmdValue.Field(i)
|
|
|
|
if !field.CanSet() || !field.CanInterface() {
|
|
continue
|
|
}
|
|
|
|
if _, ok := field.Interface().(*Context); !ok {
|
|
continue
|
|
}
|
|
|
|
field.Set(reflect.ValueOf(ctx))
|
|
return nil
|
|
}
|
|
|
|
return errors.New("No fields with *bot.Context found")
|
|
}
|
|
|
|
func (sub *Subcommand) parseCommands() error {
|
|
var numMethods = sub.ptrValue.NumMethod()
|
|
|
|
for i := 0; i < numMethods; i++ {
|
|
method := sub.ptrValue.Method(i)
|
|
|
|
if !method.CanInterface() {
|
|
continue
|
|
}
|
|
|
|
methodT := method.Type()
|
|
numArgs := methodT.NumIn()
|
|
|
|
if numArgs == 0 {
|
|
// Doesn't meet the requirement for an event, continue.
|
|
continue
|
|
}
|
|
|
|
if methodT == typeSetupFn {
|
|
// Method is a setup method, continue.
|
|
continue
|
|
}
|
|
|
|
// Check number of returns:
|
|
numOut := methodT.NumOut()
|
|
|
|
// Returns can either be:
|
|
// Nothing - func()
|
|
// An error - func() error
|
|
// An error and something else - func() (T, error)
|
|
if numOut > 2 {
|
|
continue
|
|
}
|
|
|
|
// Check the last return's type if the method returns anything.
|
|
if numOut > 0 {
|
|
if i := methodT.Out(numOut - 1); i == nil || !i.Implements(typeIError) {
|
|
// Invalid, skip.
|
|
continue
|
|
}
|
|
}
|
|
|
|
var command = CommandContext{
|
|
method: sub.ptrType.Method(i),
|
|
value: method,
|
|
event: methodT.In(0), // parse event
|
|
}
|
|
|
|
// Parse the method name
|
|
flag, name := ParseFlag(command.method.Name)
|
|
|
|
// Set the method name, command, and flag:
|
|
command.MethodName = name
|
|
command.Command = name
|
|
command.Flag = flag
|
|
|
|
// Check if Raw is enabled for command:
|
|
if !flag.Is(Raw) {
|
|
command.Command = lowerFirstLetter(name)
|
|
}
|
|
|
|
// Middlewares shouldn't even have arguments.
|
|
if flag.Is(Middleware) {
|
|
sub.mwMethods = append(sub.mwMethods, &command)
|
|
continue
|
|
}
|
|
|
|
// TODO: allow more flexibility
|
|
if command.event != typeMessageCreate || flag.Is(Hidden) {
|
|
sub.Events = append(sub.Events, &command)
|
|
continue
|
|
}
|
|
|
|
// See if we know the first return type, if error's return is the
|
|
// second:
|
|
if numOut > 1 {
|
|
switch t := methodT.Out(0); t {
|
|
case typeString, typeEmbed, typeSend:
|
|
// noop, passes
|
|
default:
|
|
continue
|
|
}
|
|
}
|
|
|
|
// If a plumb method has been found:
|
|
if sub.plumb {
|
|
continue
|
|
}
|
|
|
|
// If the method only takes an event:
|
|
if numArgs == 1 {
|
|
sub.Commands = append(sub.Commands, &command)
|
|
continue
|
|
}
|
|
|
|
// The argument's second argument (the first is the event).
|
|
var inT = methodT.In(1)
|
|
var ptr bool
|
|
|
|
if inT.Kind() != reflect.Ptr {
|
|
inT = reflect.PtrTo(inT)
|
|
ptr = true
|
|
}
|
|
|
|
// If the second argument implements CustomParse()
|
|
if t := inT; t.Implements(typeICusP) {
|
|
mt, _ := inT.MethodByName("CustomParse")
|
|
|
|
if t.Kind() == reflect.Ptr {
|
|
t = t.Elem()
|
|
}
|
|
|
|
command.Arguments = []Argument{{
|
|
String: t.String(),
|
|
Type: t,
|
|
pointer: ptr,
|
|
custom: &mt,
|
|
}}
|
|
|
|
goto Done
|
|
}
|
|
|
|
// If the second argument implements ParseContent()
|
|
if t := inT; t.Implements(typeIManP) {
|
|
mt, _ := inT.MethodByName("ParseContent")
|
|
|
|
if t.Kind() == reflect.Ptr {
|
|
t = t.Elem()
|
|
}
|
|
|
|
command.Arguments = []Argument{{
|
|
String: t.String(),
|
|
Type: t,
|
|
pointer: ptr,
|
|
manual: &mt,
|
|
}}
|
|
|
|
goto Done
|
|
}
|
|
|
|
command.Arguments = make([]Argument, 0, numArgs)
|
|
|
|
// Fill up arguments
|
|
for i := 1; i < numArgs; i++ {
|
|
t := methodT.In(i)
|
|
a, err := getArgumentValueFn(t)
|
|
if err != nil {
|
|
return errors.Wrap(err, "Error parsing argument "+t.String())
|
|
}
|
|
|
|
command.Arguments = append(command.Arguments, *a)
|
|
}
|
|
|
|
Done:
|
|
// If the current event is a plumb event:
|
|
if flag.Is(Plumb) {
|
|
command.Command = "" // plumbers don't have names
|
|
sub.Commands = []*CommandContext{&command}
|
|
sub.plumb = true
|
|
continue
|
|
}
|
|
|
|
// Append
|
|
sub.Commands = append(sub.Commands, &command)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func lowerFirstLetter(name string) string {
|
|
return strings.ToLower(string(name[0])) + name[1:]
|
|
}
|