generate.go 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336
  1. /*
  2. * ECAL
  3. *
  4. * Copyright 2020 Matthias Ladkau. All rights reserved.
  5. *
  6. * This Source Code Form is subject to the terms of the MIT
  7. * License, If a copy of the MIT License was not distributed with this
  8. * file, You can obtain one at https://opensource.org/licenses/MIT.
  9. */
  10. package main
  11. import (
  12. "bytes"
  13. "flag"
  14. "fmt"
  15. "go/ast"
  16. "go/build"
  17. "go/doc"
  18. "go/importer"
  19. goparser "go/parser"
  20. "go/token"
  21. "go/types"
  22. "io/ioutil"
  23. "os"
  24. "path/filepath"
  25. "sort"
  26. "unicode"
  27. "devt.de/krotik/common/errorutil"
  28. "devt.de/krotik/common/stringutil"
  29. )
  30. //go:generate echo Generating ECAL stdlib from Go functions ...
  31. //go:generate go run devt.de/krotik/ecal/stdlib/generate $PWD/stdlib
  32. /*
  33. Stdlib candidates modules:
  34. go list std | grep -v internal | grep -v '\.' | grep -v unsafe | grep -v syscall
  35. */
  36. // List underneath the Go symbols which should be available in ECAL's stdlib
  37. // e.g. var pkgNames = map[string][]string{ "fmt": {"Println", "Sprint"} }
  38. // Turn the generateDoc switch on or off to extract documentation from the
  39. // Go source.
  40. // =============EDIT HERE START=============
  41. var pkgNames = map[string][]string{
  42. // "math": {"Pi", "E", "Phi", "Pow"},
  43. // "fmt": {"Println", "Sprint"},
  44. }
  45. var generateDoc = true
  46. // ==============EDIT HERE END==============
  47. var filename = filepath.Join(os.Args[1], "stdlib_gen.go")
  48. var stderrPrint = fmt.Println
  49. var stdoutPrint = fmt.Println
  50. func main() {
  51. var err error
  52. var outbuf bytes.Buffer
  53. synopsis := make(map[string]string)
  54. pkgDocs := make(map[string]*doc.Package)
  55. flag.Parse()
  56. // Make sure we have at least an empty pkgName
  57. if len(pkgNames) == 0 {
  58. pkgNames["math"] = []string{"Pi"}
  59. }
  60. // Make sure pkgNames is sorted
  61. var importList []string
  62. for pkgName, names := range pkgNames {
  63. sort.Strings(names)
  64. importList = append(importList, pkgName)
  65. }
  66. sort.Strings(importList)
  67. outbuf.WriteString(`
  68. // Code generated by ecal/stdlib/generate; DO NOT EDIT.
  69. package stdlib
  70. `)
  71. outbuf.WriteString("import (\n")
  72. for _, pkgName := range importList {
  73. if generateDoc {
  74. syn, pkgDoc, err := getPackageDocs(pkgName)
  75. errorutil.AssertOk(err) // If this throws try not generating the docs!
  76. synopsis[pkgName] = syn
  77. pkgDocs[pkgName] = pkgDoc
  78. } else {
  79. synopsis[pkgName] = fmt.Sprintf("Package %v", pkgName)
  80. }
  81. outbuf.WriteString(fmt.Sprintf("\t\"%v\"\n", pkgName))
  82. }
  83. if stringutil.IndexOf("fmt", importList) == -1 {
  84. outbuf.WriteString(` "fmt"
  85. `)
  86. }
  87. if stringutil.IndexOf("reflect", importList) == -1 {
  88. outbuf.WriteString(` "reflect"
  89. )
  90. `)
  91. }
  92. outbuf.WriteString(`/*
  93. genStdlib contains all generated stdlib constructs.
  94. */
  95. `)
  96. outbuf.WriteString("var genStdlib = map[interface{}]interface{}{\n")
  97. for _, pkgName := range importList {
  98. if s, ok := synopsis[pkgName]; ok {
  99. outbuf.WriteString(fmt.Sprintf("\t\"%v-synopsis\" : %#v,\n", pkgName, s))
  100. }
  101. outbuf.WriteString(fmt.Sprintf("\t\"%v-const\" : %vConstMap,\n", pkgName, pkgName))
  102. outbuf.WriteString(fmt.Sprintf("\t\"%v-func\" : %vFuncMap,\n", pkgName, pkgName))
  103. outbuf.WriteString(fmt.Sprintf("\t\"%v-func-doc\" : %vFuncDocMap,\n", pkgName, pkgName))
  104. }
  105. outbuf.WriteString("}\n\n")
  106. for _, pkgName := range importList {
  107. var pkg *types.Package
  108. pkgSymbols := pkgNames[pkgName]
  109. if err == nil {
  110. pkg, err = importer.ForCompiler(fset, "source", nil).Import(pkgName)
  111. if err == nil {
  112. stdoutPrint("Generating adapter functions for", pkg)
  113. scope := pkg.Scope()
  114. // Write constants
  115. outbuf.WriteString(fmt.Sprintf(`/*
  116. %vConstMap contains the mapping of stdlib %v constants.
  117. */
  118. var %vConstMap = map[interface{}]interface{}{
  119. `, pkgName, pkgName, pkgName))
  120. for _, name := range scope.Names() {
  121. if !containsSymbol(pkgSymbols, name) {
  122. continue
  123. }
  124. switch obj := scope.Lookup(name).(type) {
  125. case *types.Const:
  126. if unicode.IsUpper([]rune(name)[0]) {
  127. line := fmt.Sprintf(` "%v": %v.%v,
  128. `, name, pkgName, obj.Name())
  129. if basicType, ok := obj.Type().(*types.Basic); ok {
  130. // Convert number constants so they can be used in calculations
  131. switch basicType.Kind() {
  132. case types.Int,
  133. types.Int8,
  134. types.Int16,
  135. types.Int32,
  136. types.Int64,
  137. types.Uint,
  138. types.Uint8,
  139. types.Uint16,
  140. types.Uint32,
  141. types.Uint64,
  142. types.Uintptr,
  143. types.Float32,
  144. types.UntypedInt,
  145. types.UntypedFloat:
  146. line = fmt.Sprintf(` "%v": float64(%v.%v),
  147. `, name, pkgName, obj.Name())
  148. }
  149. }
  150. outbuf.WriteString(line)
  151. }
  152. }
  153. }
  154. outbuf.WriteString("}\n\n")
  155. // Write function documentation
  156. outbuf.WriteString(fmt.Sprintf(`/*
  157. %vFuncDocMap contains the documentation of stdlib %v functions.
  158. */
  159. var %vFuncDocMap = map[interface{}]interface{}{
  160. `, pkgName, pkgName, pkgName))
  161. if pkgDoc, ok := pkgDocs[pkgName]; ok {
  162. for _, name := range scope.Names() {
  163. if !containsSymbol(pkgSymbols, name) {
  164. continue
  165. }
  166. for _, f := range pkgDoc.Funcs {
  167. if f.Name == name {
  168. outbuf.WriteString(
  169. fmt.Sprintf(` "%v": %#v,
  170. `, name, f.Doc))
  171. }
  172. }
  173. }
  174. } else {
  175. for _, name := range pkgSymbols {
  176. switch scope.Lookup(name).(type) {
  177. case *types.Func:
  178. outbuf.WriteString(
  179. fmt.Sprintf(` "%v": "Function: %v",
  180. `, name, name))
  181. }
  182. }
  183. }
  184. outbuf.WriteString("}\n\n")
  185. // Write functions
  186. outbuf.WriteString(fmt.Sprintf(`/*
  187. %vFuncMap contains the mapping of stdlib %v functions.
  188. */
  189. var %vFuncMap = map[interface{}]interface{}{
  190. `, pkgName, pkgName, pkgName))
  191. for _, name := range scope.Names() {
  192. if !containsSymbol(pkgSymbols, name) {
  193. continue
  194. }
  195. switch obj := scope.Lookup(name).(type) {
  196. case *types.Func:
  197. if unicode.IsUpper([]rune(name)[0]) {
  198. outbuf.WriteString(
  199. fmt.Sprintf(` %#v: &ECALFunctionAdapter{reflect.ValueOf(%v), fmt.Sprint(%vFuncDocMap[%#v])},
  200. `, name, obj.FullName(), pkgName, name))
  201. }
  202. }
  203. }
  204. outbuf.WriteString("}\n\n")
  205. }
  206. }
  207. }
  208. // Write dummy statement
  209. outbuf.WriteString("// Dummy statement to prevent declared and not used errors\n")
  210. outbuf.WriteString("var Dummy = fmt.Sprint(reflect.ValueOf(fmt.Sprint))\n\n")
  211. if err == nil {
  212. err = ioutil.WriteFile(filename, outbuf.Bytes(), 0644)
  213. }
  214. if err != nil {
  215. stderrPrint("Error:", err)
  216. }
  217. }
  218. var (
  219. fset = token.NewFileSet()
  220. ctx = &build.Default
  221. )
  222. /*
  223. getPackageDocs returns the source code documentation of as given Go package.
  224. Returns a short synopsis and a documentation object.
  225. */
  226. func getPackageDocs(pkgName string) (string, *doc.Package, error) {
  227. var synopsis string
  228. var pkgDoc *doc.Package
  229. var filenames []string
  230. bp, err := ctx.Import(pkgName, ".", 0)
  231. if err == nil {
  232. synopsis = bp.Doc
  233. // Get all go files of the package
  234. filenames = append(filenames, bp.GoFiles...)
  235. filenames = append(filenames, bp.CgoFiles...)
  236. // Build the ast package from Go source
  237. astPkg := &ast.Package{
  238. Name: bp.Name,
  239. Files: make(map[string]*ast.File),
  240. }
  241. for _, filename := range filenames {
  242. filepath := filepath.Join(bp.Dir, filename)
  243. astFile, _ := goparser.ParseFile(fset, filepath, nil, goparser.ParseComments)
  244. astPkg.Files[filepath] = astFile
  245. }
  246. // Build the package doc object
  247. pkgDoc = doc.New(astPkg, bp.Dir, doc.AllDecls)
  248. }
  249. return synopsis, pkgDoc, err
  250. }
  251. func containsSymbol(symbols []string, item string) bool {
  252. i := sort.SearchStrings(symbols, item)
  253. return i < len(symbols) && symbols[i] == item
  254. }