mirror of
https://github.com/StackExchange/dnscontrol.git
synced 2025-01-10 09:27:46 +08:00
38b0be5d3f
Co-authored-by: str4d <4993799+str4d@users.noreply.github.com> Co-authored-by: Tom Limoncelli <tlimoncelli@stackoverflow.com>
265 lines
6.5 KiB
Go
265 lines
6.5 KiB
Go
package main
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"os"
|
|
"regexp"
|
|
"sort"
|
|
"strings"
|
|
|
|
"golang.org/x/text/cases"
|
|
"golang.org/x/text/language"
|
|
"gopkg.in/yaml.v3"
|
|
)
|
|
|
|
func join(parts ...string) string {
|
|
return strings.Join(parts, string(os.PathSeparator))
|
|
}
|
|
|
|
// removes repeated blank lines, replacing them with a single blank line.
|
|
func fixRuns(s string) string {
|
|
lines := strings.Split(s, "\n")
|
|
var out []string
|
|
for _, line := range lines {
|
|
if len(line) == 0 {
|
|
if len(out) > 0 && len(out[len(out)-1]) == 0 {
|
|
continue
|
|
}
|
|
}
|
|
out = append(out, line)
|
|
}
|
|
return strings.Join(out, "\n")
|
|
}
|
|
|
|
var delimiterRegex = regexp.MustCompile(`(?m)^---\n`)
|
|
|
|
func readDocFile(fPath string) (map[string]interface{}, string, error) {
|
|
content, err := os.ReadFile(fPath)
|
|
if err != nil {
|
|
return nil, "", err
|
|
}
|
|
frontMatter, body, err := parseFrontMatter(string(content))
|
|
if err != nil {
|
|
return nil, "", err
|
|
}
|
|
|
|
lines := strings.Split(body, "\n")
|
|
|
|
body = ""
|
|
|
|
for _, line := range lines {
|
|
if strings.HasPrefix(line, "{%") && strings.HasSuffix(line, "%}") {
|
|
continue
|
|
}
|
|
body += line + "\n"
|
|
}
|
|
|
|
body = strings.ReplaceAll(body, "**NOTE**", "NOTE")
|
|
body = strings.ReplaceAll(body, "**WARNING**", "WARNING")
|
|
body = fixRuns(body)
|
|
return frontMatter, body, nil
|
|
}
|
|
|
|
func parseFrontMatter(content string) (map[string]interface{}, string, error) {
|
|
delimiterIndices := delimiterRegex.FindAllStringIndex(content, 2)
|
|
if len(delimiterIndices) < 1 {
|
|
return nil, "", fmt.Errorf("failed to parse file. Remove it and try again")
|
|
}
|
|
startIndex := delimiterIndices[0][0]
|
|
endIndex := delimiterIndices[1][0]
|
|
yamlString := content[startIndex+4 : endIndex]
|
|
var frontMatter map[string]interface{}
|
|
err := yaml.Unmarshal([]byte(yamlString), &frontMatter)
|
|
if err != nil {
|
|
return nil, "", err
|
|
}
|
|
return frontMatter, content[endIndex+4:], nil
|
|
}
|
|
|
|
var returnTypes = map[string]string{
|
|
"domain": "DomainModifier",
|
|
"global": "void",
|
|
"record": "RecordModifier",
|
|
}
|
|
|
|
var categories = map[string]string{
|
|
"domain": "domain-modifiers",
|
|
"global": "top-level-functions",
|
|
"record": "record-modifiers",
|
|
}
|
|
|
|
var providerNames = map[string]string{
|
|
"AKAMAIEDGEDNS": "akamai-edge-dns",
|
|
"ROUTE53": "amazon-route-53",
|
|
"AZURE_DNS": "azure-dns",
|
|
"CLOUDFLAREAPI": "cloudflare-dns",
|
|
"CLOUDNS": "cloudns",
|
|
"NS1": "ns1",
|
|
}
|
|
|
|
func generateFunctionTypes() (string, error) {
|
|
funcs := []Function{}
|
|
|
|
srcRoot := join("documentation", "functions")
|
|
types, err := os.ReadDir(srcRoot)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
for _, t := range types {
|
|
if !t.IsDir() {
|
|
return "", errors.New("not a directory: " + join(srcRoot, t.Name()))
|
|
}
|
|
tPath := join(srcRoot, t.Name())
|
|
funcNames, err := os.ReadDir(tPath)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
for _, f := range funcNames {
|
|
fPath := join(tPath, f.Name())
|
|
if f.IsDir() {
|
|
return "", errors.New("not a file: " + fPath)
|
|
}
|
|
// println("Processing", fPath)
|
|
frontMatter, body, err := readDocFile(fPath)
|
|
if err != nil {
|
|
println("Error parsing front matter in", fPath, "error: ", err.Error())
|
|
continue
|
|
|
|
}
|
|
if frontMatter["ts_ignore"] == true {
|
|
continue
|
|
}
|
|
|
|
paramNames := []string{}
|
|
if frontMatter["parameters"] != nil {
|
|
for _, p := range frontMatter["parameters"].([]interface{}) {
|
|
paramNames = append(paramNames, p.(string))
|
|
}
|
|
}
|
|
|
|
suppliedParamTypes := map[string]string{}
|
|
if frontMatter["parameter_types"] != nil {
|
|
rawTypes := frontMatter["parameter_types"].(map[string]interface{})
|
|
for k, v := range rawTypes {
|
|
suppliedParamTypes[k] = v.(string)
|
|
}
|
|
}
|
|
|
|
params := []Param{}
|
|
for _, p := range paramNames {
|
|
// start with supplied type, fall back to defaultParamType
|
|
paramType := suppliedParamTypes[p]
|
|
if paramType == "" {
|
|
println("WARNING:", fPath+":", "no type for parameter ", "'"+p+"'")
|
|
paramType = "unknown"
|
|
}
|
|
params = append(params, Param{Name: p, Type: paramType})
|
|
}
|
|
|
|
returnType := returnTypes[t.Name()]
|
|
if frontMatter["ts_return"] != nil {
|
|
returnType = frontMatter["ts_return"].(string)
|
|
} else if frontMatter["return"] != nil {
|
|
returnType = frontMatter["return"].(string)
|
|
}
|
|
|
|
if len(params) == 0 {
|
|
if frontMatter["ts_is_function"] != true {
|
|
params = nil
|
|
}
|
|
}
|
|
|
|
category := categories[t.Name()]
|
|
if frontMatter["provider"] != nil {
|
|
category += "/service-provider-specific/" + providerNames[frontMatter["provider"].(string)]
|
|
}
|
|
|
|
funcs = append(funcs, Function{
|
|
Name: frontMatter["name"].(string),
|
|
Category: category,
|
|
Params: params,
|
|
ObjectParam: frontMatter["parameters_object"] == true,
|
|
Deprecated: frontMatter["deprecated"] == true,
|
|
ReturnType: returnType,
|
|
Description: strings.TrimSpace(body),
|
|
})
|
|
}
|
|
}
|
|
|
|
sort.Slice(funcs, func(i, j int) bool {
|
|
return funcs[i].Name < funcs[j].Name
|
|
})
|
|
|
|
content := ""
|
|
for _, f := range funcs {
|
|
content += f.String()
|
|
}
|
|
return content, nil
|
|
}
|
|
|
|
// Function is a struct the stores information about functions.
|
|
type Function struct {
|
|
Name string
|
|
Category string
|
|
Params []Param
|
|
ObjectParam bool
|
|
Deprecated bool
|
|
ReturnType string
|
|
Description string
|
|
}
|
|
|
|
// Param is a struct that stores a parameter.
|
|
type Param struct {
|
|
Name string
|
|
Type string
|
|
}
|
|
|
|
var caser = cases.Title(language.AmericanEnglish)
|
|
|
|
func (f Function) formatParams() string {
|
|
var params []string
|
|
for _, p := range f.Params {
|
|
name := p.Name
|
|
if strings.HasSuffix(name, "...") {
|
|
name = "..." + name[:len(name)-3]
|
|
}
|
|
if strings.Contains(name, " ") {
|
|
name = strings.ReplaceAll(caser.String(name), " ", "")
|
|
name = strings.ToLower(name[:1]) + name[1:]
|
|
}
|
|
|
|
typeName := p.Type
|
|
if strings.HasSuffix(typeName, "?") {
|
|
typeName = typeName[:len(typeName)-1]
|
|
name += "?"
|
|
}
|
|
|
|
params = append(params, fmt.Sprintf("%s: %s", name, typeName))
|
|
}
|
|
if f.ObjectParam {
|
|
return "opts: { " + strings.Join(params, "; ") + " }"
|
|
}
|
|
return strings.Join(params, ", ")
|
|
}
|
|
|
|
func (f Function) docs() string {
|
|
content := f.Description
|
|
if f.Deprecated {
|
|
content += "\n\n@deprecated"
|
|
}
|
|
content += fmt.Sprintf("\n\n@see https://docs.dnscontrol.org/language-reference/%s/%s", f.Category, strings.ToLower(f.Name))
|
|
return "/**\n * " + strings.ReplaceAll(content, "\n", "\n * ") + "\n */"
|
|
}
|
|
|
|
func (f Function) formatMain() string {
|
|
if f.Params == nil {
|
|
return fmt.Sprintf("declare const %s: %s", f.Name, f.ReturnType)
|
|
}
|
|
return fmt.Sprintf("declare function %s(%s): %s", f.Name, f.formatParams(), f.ReturnType)
|
|
}
|
|
|
|
func (f Function) String() string {
|
|
return fmt.Sprintf("%s\n%s;\n\n", f.docs(), f.formatMain())
|
|
}
|