mirror of
https://github.com/diamondburned/arikawa.git
synced 2025-01-09 13:37:02 +00:00
54cadd2f45
This commit refactors the whole package gateway as well as utils/ws (formerly utils/wsutil) and voice/voicegateway. The new refactor utilizes a design pattern involving a concurrent loop and an arriving event channel. An additional change was made to the way gateway events are typed. Before, pretty much any type will satisfy a gateway event type, since the actual type was just interface{}. The new refactor defines a concrete interface that events can implement: type Event interface { Op() OpCode EventType() EventType } Using this interface, the user can easily add custom gateway events independently of the library without relying on string maps. This adds a lot of type safety into the library and makes type-switching on Event types much more reasonable. Gateway error callbacks are also almost entirely removed in favor of custom gateway events. A catch-all can easily be added like this: s.AddHandler(func(err error) { log.Println("gateway error:, err") })
173 lines
3.2 KiB
Go
173 lines
3.2 KiB
Go
package main
|
|
|
|
import (
|
|
"bytes"
|
|
"flag"
|
|
"go/format"
|
|
"log"
|
|
"os"
|
|
"regexp"
|
|
"strconv"
|
|
"strings"
|
|
"text/template"
|
|
"unicode"
|
|
|
|
_ "embed"
|
|
|
|
"github.com/pkg/errors"
|
|
)
|
|
|
|
var (
|
|
pkg = "gateway"
|
|
out = "-"
|
|
)
|
|
|
|
type registry struct {
|
|
PackageName string
|
|
EventTypes []EventType
|
|
}
|
|
|
|
type EventType struct {
|
|
StructName string
|
|
EventName string
|
|
IsDispatch bool
|
|
OpCode int
|
|
}
|
|
|
|
func (t *EventType) MethodRecv() string {
|
|
if len(t.StructName) == 0 {
|
|
return "e"
|
|
}
|
|
return string(unicode.ToLower([]rune(t.StructName)[0]))
|
|
}
|
|
|
|
//go:embed template.tmpl
|
|
var packageTmpl string
|
|
|
|
var tmpl = template.Must(template.New("").Parse(packageTmpl))
|
|
|
|
const eventStructRegex = "(?m)" +
|
|
`^// ([A-Za-z]+(?:Event|Command)) is (a dispatch event|an event|a command)` +
|
|
`(?:` +
|
|
` for ([A-Z_]+)` + "|" +
|
|
` for Op (\d+)` +
|
|
`)?` +
|
|
`\.(?:.|\n)*?\ntype ([A-Za-z]+(?:Event|Command)) .*`
|
|
|
|
func main() {
|
|
flag.StringVar(&pkg, "p", pkg, "the package name to use")
|
|
flag.StringVar(&out, "o", out, "output file, - for stdout")
|
|
flag.Parse()
|
|
|
|
log.Println(eventStructRegex)
|
|
|
|
r := registry{
|
|
PackageName: pkg,
|
|
}
|
|
|
|
files, err := os.ReadDir(".")
|
|
if err != nil {
|
|
log.Fatalln("failed to read current directory:", err)
|
|
}
|
|
|
|
for _, file := range files {
|
|
if file.IsDir() || !strings.HasSuffix(file.Name(), ".go") {
|
|
continue
|
|
}
|
|
if err := r.CrawlFile(file.Name()); err != nil {
|
|
log.Fatalln("failed to crawl file:", err)
|
|
}
|
|
}
|
|
|
|
buf := bytes.Buffer{}
|
|
if err := tmpl.Execute(&buf, &r); err != nil {
|
|
log.Fatalln("failed to execute template:", err)
|
|
}
|
|
|
|
b, err := format.Source(buf.Bytes())
|
|
if err != nil {
|
|
log.Fatalln("failed to fmt:", err)
|
|
}
|
|
|
|
output := os.Stdout
|
|
if out != "-" {
|
|
f, err := os.Create(out)
|
|
if err != nil {
|
|
log.Fatalln("failed to create output:", err)
|
|
}
|
|
defer f.Close()
|
|
|
|
output = f
|
|
}
|
|
|
|
if _, err := output.Write(b); err != nil {
|
|
log.Fatalln("failed to write rendered:", err)
|
|
}
|
|
}
|
|
|
|
var reEventStruct = regexp.MustCompile(eventStructRegex)
|
|
|
|
func (r *registry) CrawlFile(name string) error {
|
|
f, err := os.ReadFile(name)
|
|
if err != nil {
|
|
return errors.Wrap(err, "failed to read file")
|
|
}
|
|
|
|
for _, match := range reEventStruct.FindAllSubmatch(f, -1) {
|
|
// Validity check.
|
|
if string(match[1]) != string(match[5]) {
|
|
continue
|
|
}
|
|
|
|
if strings.HasSuffix(string(match[1]), "Command") && string(match[2]) != "a command" {
|
|
log.Println(string(match[1]), "has invalid comment %q", string(match[2]))
|
|
continue
|
|
}
|
|
|
|
t := EventType{
|
|
StructName: string(match[1]),
|
|
EventName: string(match[3]),
|
|
IsDispatch: string(match[2]) == "a dispatch event",
|
|
OpCode: -1,
|
|
}
|
|
|
|
if op := string(match[4]); op != "" && !t.IsDispatch {
|
|
i, err := strconv.Atoi(op)
|
|
if err != nil {
|
|
log.Printf("error at struct %s: error parsing Op %v", t.StructName, err)
|
|
}
|
|
t.OpCode = i
|
|
}
|
|
|
|
if t.IsDispatch && t.EventName == "" {
|
|
t.EventName = guessEventName(t.StructName)
|
|
}
|
|
|
|
r.EventTypes = append(r.EventTypes, t)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func guessEventName(structName string) string {
|
|
name := strings.TrimSuffix(structName, "Event")
|
|
|
|
var newName strings.Builder
|
|
newName.Grow(len(name) * 2)
|
|
|
|
for i, r := range name {
|
|
if unicode.IsLower(r) {
|
|
newName.WriteRune(unicode.ToUpper(r))
|
|
continue
|
|
}
|
|
|
|
if i > 0 {
|
|
newName.WriteByte('_')
|
|
}
|
|
|
|
newName.WriteRune(r)
|
|
}
|
|
|
|
return newName.String()
|
|
}
|