mox/rfc/link.go

217 lines
5.2 KiB
Go
Raw Normal View History

2023-01-30 16:27:06 +03:00
//go:build link
package main
// Read source files and RFC and errata files, and cross-link them.
// todo: also cross-reference typescript and possibly other files. switch from go parser to just reading the source as text.
2023-01-30 16:27:06 +03:00
import (
"bytes"
"flag"
"fmt"
"go/parser"
"go/token"
"log"
"os"
"path/filepath"
"regexp"
"strconv"
"strings"
)
func usage() {
log.Println("usage: link ../*.go ../*/*.go")
flag.PrintDefaults()
os.Exit(2)
}
func main() {
log.SetFlags(0)
flag.Usage = usage
flag.Parse()
args := flag.Args()
if len(args) == 0 {
usage()
}
type ref struct {
srcpath string
srclineno int
dstpath string
dstlineno int
dstisrfc bool
dstrfc string // e.g. "5322" or "6376-eid4810"
comment string // e.g. "todo" or "todo spec"
2023-01-30 16:27:06 +03:00
}
// RFC-file to RFC-line to references to list of file+line (possibly RFCs).
rfcLineSources := map[string]map[int][]ref{}
// Source-file to source-line to references of RFCs.
sourceLineRFCs := map[string]map[int][]ref{}
re := regexp.MustCompile(`((../)*)rfc/([0-9]{4,5})(-eid([1-9][0-9]*))?(:([1-9][0-9]*))?`)
addRef := func(m map[string]map[int][]ref, rfc string, lineno int, r ref) {
lineRefs := m[rfc]
if lineRefs == nil {
lineRefs = map[int][]ref{}
m[rfc] = lineRefs
}
lineRefs[lineno] = append(lineRefs[lineno], r)
}
// Parse all .go files on the cli, assumed to be relative to current dir.
fset := token.NewFileSet()
for _, arg := range args {
f, err := parser.ParseFile(fset, arg, nil, parser.ParseComments|parser.SkipObjectResolution)
if err != nil {
log.Fatalf("parse file %q: %s", arg, err)
}
for _, cg := range f.Comments {
for _, c := range cg.List {
lines := strings.Split(c.Text, "\n")
for i, line := range lines {
matches := re.FindAllStringSubmatch(line, -1)
if len(matches) == 0 {
continue
}
var comment string
if strings.HasPrefix(line, "// todo") {
s, _, have := strings.Cut(strings.TrimPrefix(line, "// "), ":")
if have {
comment = s
} else {
comment = "todo"
}
}
2023-01-30 16:27:06 +03:00
srcpath := arg
srclineno := fset.Position(c.Pos()).Line + i
dir := filepath.Dir(srcpath)
for _, m := range matches {
pre := m[1]
rfc := m[3]
eid := m[5]
lineStr := m[7]
if eid != "" && lineStr != "" {
log.Fatalf("%s:%d: cannot reference both errata (eid %q) to specified line number", srcpath, srclineno, eid)
}
var dstlineno int
if lineStr != "" {
v, err := strconv.ParseInt(lineStr, 10, 32)
if err != nil {
log.Fatalf("%s:%d: bad linenumber %q: %v", srcpath, srclineno, lineStr, err)
}
dstlineno = int(v)
}
if dstlineno <= 0 {
dstlineno = 1
}
if eid != "" {
rfc += "-eid" + eid
}
dstpath := filepath.Join(dir, pre+"rfc", rfc)
if _, err := os.Stat(dstpath); err != nil {
log.Fatalf("%s:%d: references %s: %v", srcpath, srclineno, dstpath, err)
}
r := ref{srcpath, srclineno, dstpath, dstlineno, true, rfc, comment}
2023-01-30 16:27:06 +03:00
addRef(sourceLineRFCs, r.srcpath, r.srclineno, r)
addRef(rfcLineSources, r.dstrfc, r.dstlineno, ref{r.dstrfc, r.dstlineno, r.srcpath, r.srclineno, false, "", comment})
2023-01-30 16:27:06 +03:00
}
}
}
}
}
files, err := os.ReadDir(".")
if err != nil {
log.Fatalf("readdir: %v", err)
}
for _, de := range files {
name := de.Name()
isrfc := isRFC(name)
iserrata := isErrata(name)
if !isrfc && !iserrata {
continue
}
oldBuf, err := os.ReadFile(name)
if err != nil {
log.Fatalf("readdir: %v", err)
}
old := string(oldBuf)
b := &bytes.Buffer{}
lineRefs := rfcLineSources[name]
lines := strings.Split(old, "\n")
if len(lines) > 0 && lines[len(lines)-1] == "" {
lines = lines[:len(lines)-1]
}
for i, line := range lines {
if !iserrata && len(line) > 80 {
line = strings.TrimRight(line[:80], " ")
}
refs := lineRefs[i+1]
if len(refs) > 0 {
if iserrata {
line = ""
} else {
line = fmt.Sprintf("%-80s", line)
}
// Lookup source files for rfc:line, so we can cross-link the rfcs.
done := map[string]bool{}
for _, r := range refs {
for _, xr := range sourceLineRFCs[r.dstpath][r.dstlineno] {
sref := fmt.Sprintf(" %s:%d", xr.dstrfc, xr.dstlineno)
if xr.dstrfc == name && xr.dstlineno == i+1 || done[sref] {
continue
}
line += sref
done[sref] = true
}
}
// Add link from rfc to source code.
for _, r := range refs {
comment := r.comment
if comment != "" {
comment += ": "
}
line += fmt.Sprintf(" %s%s:%d", comment, r.dstpath, r.dstlineno)
2023-01-30 16:27:06 +03:00
}
if iserrata {
line = line[1:]
}
}
line += "\n"
b.WriteString(line)
}
newBuf := b.Bytes()
if !bytes.Equal(oldBuf, newBuf) {
if err := os.WriteFile(name, newBuf, 0660); err != nil {
log.Printf("writefile %q: %s", name, err)
}
log.Print(name)
}
}
}
func isRFC(name string) bool {
if len(name) < 4 || len(name) > 5 {
return false
}
for _, c := range name {
if c < '0' || c > '9' {
return false
}
}
return true
}
func isErrata(name string) bool {
t := strings.Split(name, "-")
return len(t) == 2 && isRFC(t[0]) && strings.HasPrefix(t[1], "eid")
}