package main import ( "fmt" "log" "os" "path/filepath" "strings" "text/template" "github.com/PuerkitoBio/goquery" "github.com/davecgh/go-spew/spew" ) type ( Class struct { ClassName string Fields []Field Methods []Method Constructors []Constructor } Field struct { Name string Type string Comment string } Method struct { Name string Params []Param Returns []Return Comment string } Param struct { Name string Type string Comment string } Return struct { Type string Comment string } Constructor struct { Params []Param Comment string } ) func (c *Class) GetOutFile(root string) (*os.File, error) { if c.ClassName == "" { return nil, fmt.Errorf("ClassName is empty") } filename := fmt.Sprintf("%s.lua", c.ClassName) filename = strings.ReplaceAll(filename, " ", "") filename = strings.ReplaceAll(filename, "-", "") filename = strings.ReplaceAll(filename, ",", "") filename = strings.ReplaceAll(filename, ":", "") filePath := filepath.Join(root, filename) filePath = filepath.Clean(filePath) f, err := os.Create(filePath) if err != nil { return nil, err } return f, nil } func (c *Class) Write(root string, tmpl *template.Template) error { outfile, err := c.GetOutFile(root) if err != nil { return fmt.Errorf("error creating output file %v: %v", c.ClassName, err) } err = tmpl.Execute(outfile, c) if err != nil { return fmt.Errorf("error writing output file %v: %v", c.ClassName, err) } return nil } func ParseClass(file string) (*Class, error) { log.Printf("Parsing file: '%s'", file) res := Class{ Fields: []Field{}, Methods: []Method{}, Constructors: []Constructor{}, } filehandle, err := os.Open(file) if err != nil { return nil, fmt.Errorf("error opening file: %w", err) } defer filehandle.Close() doc, err := goquery.NewDocumentFromReader(filehandle) if err != nil { return nil, fmt.Errorf("error parsing file: %w", err) } class := doc.Find("div.floatright > h1") if class.Length() == 0 { return nil, fmt.Errorf("no class found") } res.ClassName = strings.TrimSpace(class.Text()) codeblocks := doc.Find("div.floatright > div.codecontainer") if codeblocks.Length() == 0 { return nil, fmt.Errorf("no codeblocks found") } // The first code block should be the constructor // So far I have not found any classes with overloaded constructors... // So I can not handle that case yet constructorBlock := codeblocks.Eq(0) constructor := constructorBlock.Find("div.function") paramTypes := constructor.Find("span.type") paramNames := constructor.Find("span.parameter") resConstructor := Constructor{} paramTypes.Each(func(i int, s *goquery.Selection) { pname := strings.TrimSpace(paramNames.Eq(i).Text()) ptype := strings.TrimSpace(paramTypes.Eq(i).Text()) ptype = MapType(ptype) resConstructor.Params = append(resConstructor.Params, Param{ Name: pname, Type: ptype, Comment: "", }) }) res.Constructors = append(res.Constructors, resConstructor) properties := doc.Find("div.floatright > div.codecontainer#Properties") properties.ChildrenFiltered("div").Each(func(i int, s *goquery.Selection) { property := Field{} property.Name = strings.TrimSpace(s.Find("span.property").Text()) property.Type = strings.TrimSpace(s.Find("span.type").Text()) property.Type = MapType(property.Type) comment := s.Find("td[align='right']").Text() if comment != "" { property.Comment = strings.TrimSpace(comment) } res.Fields = append(res.Fields, property) }) codeblocks.ChildrenFiltered("div.function").Each(func(i int, s *goquery.Selection) { method := Method{} method.Name = strings.TrimSpace(s.AttrOr("id", "")) method.Comment = strings.TrimSpace(s.Find("span.comment").Text()) types := s.Find("span.type") parameters := s.Find("span.parameter") types.Each(func(i int, s *goquery.Selection) { param := Param{} param.Name = strings.TrimSpace(parameters.Eq(i).Text()) param.Type = strings.TrimSpace(types.Eq(i).Text()) param.Type = MapType(param.Type) method.Params = append(method.Params, param) }) res.Methods = append(res.Methods, method) }) spew.Dump(res) return &res, nil }