Skip to content

Commit

Permalink
Upgrade for the native protocols spec page
Browse files Browse the repository at this point in the history
  • Loading branch information
tengu-alt committed Sep 3, 2024
1 parent dddece2 commit 2bb65c0
Show file tree
Hide file tree
Showing 12 changed files with 603 additions and 14 deletions.
12 changes: 12 additions & 0 deletions cqlprotodoc/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
Comes from https://github.com/martin-sucha/cqlprotodoc

cqlprotodoc converts
[CQL protocol specification files](https://github.com/apache/cassandra/tree/trunk/doc)
to HTML for easier browsing.

Usage:

1. Update template-notice.txt based on your Cassandra distribution.
2. `cqlprotodoc <path_to_cassandra_doc_dir> <output_dir>`

The generated files are published at https://martin-sucha.github.io/cqlprotodoc/
5 changes: 5 additions & 0 deletions cqlprotodoc/go.mod
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
module cqlprotodoc

go 1.17

require github.com/mvdan/xurls v1.1.0
2 changes: 2 additions & 0 deletions cqlprotodoc/go.sum
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
github.com/mvdan/xurls v1.1.0 h1:OpuDelGQ1R1ueQ6sSryzi6P+1RtBpfQHM8fJwlE45ww=
github.com/mvdan/xurls v1.1.0/go.mod h1:tQlNn3BED8bE/15hnSL2HLkDeLWpNPAwtw7wkEq44oU=
177 changes: 177 additions & 0 deletions cqlprotodoc/main.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,177 @@
package main

import (
"cqlprotodoc/spec"
"embed"
"fmt"
"html/template"
"os"
"path/filepath"
"strings"
)

//go:embed template.gohtml
var templateFS embed.FS

//go:embed template-notice.txt
var notice []byte

type TOCNode struct {
spec.TOCEntry
Exists bool
Children []*TOCNode
}

func buildTOCTree(entries []spec.TOCEntry, sectionNumbers map[string]struct{}) []*TOCNode {
var root TOCNode
stack := []*TOCNode{&root}
for _, e := range entries {
level := strings.Count(e.Number, ".") + 1
if len(stack) > level {
stack = stack[:level]
}
parent := stack[len(stack)-1]
_, exists := sectionNumbers[e.Number]
node := &TOCNode{
TOCEntry: e,
Children: nil,
Exists: exists,
}
parent.Children = append(parent.Children, node)
stack = append(stack, node)
}
return root.Children
}

type templateData struct {
spec.Document
LicenseHTML template.HTML
TOCTree []*TOCNode
Sections []Section
}

type Section struct {
spec.Section
Level int
BodyHTML template.HTML
}

func link(sb *strings.Builder, href, text string) {
sb.WriteString(`<a href="`)
sb.WriteString(template.HTMLEscapeString(href))
sb.WriteString(`">`)
sb.WriteString(template.HTMLEscapeString(text))
sb.WriteString(`</a>`)
}

func formatBody(text []spec.Text, sectionNumbers map[string]struct{}) template.HTML {
var sb strings.Builder
for _, t := range text {
switch {
case t.SectionRef != "":
if _, ok := sectionNumbers[t.SectionRef]; ok {
link(&sb, "#s"+t.SectionRef, t.Text)
} else {
sb.WriteString(template.HTMLEscapeString(t.Text))
}
case t.Href != "":
link(&sb, t.Href, t.Text)
default:
sb.WriteString(template.HTMLEscapeString(t.Text))
}
}
return template.HTML(sb.String())
}

func buildSections(in []spec.Section, sectionNumbers map[string]struct{}) []Section {
ret := make([]Section, len(in))
for i, s := range in {
ret[i].Section = in[i]
ret[i].Level = strings.Count(s.Number, ".") + 2
ret[i].BodyHTML = formatBody(in[i].Body, sectionNumbers)
}
return ret
}

func checkSectionLinks(d spec.Document, sectionNumbers, tocNumbers map[string]struct{}) {

for _, t := range d.TOC {
if _, ok := sectionNumbers[t.Number]; !ok {
fmt.Fprintf(os.Stderr, "section %q exists in TOC, but not in sections\n", t.Number)
}
}
for _, s := range d.Sections {
if _, ok := tocNumbers[s.Number]; !ok {
fmt.Fprintf(os.Stderr, "section %q exists in sections, but not in TOC\n", s.Number)
}
for _, tt := range s.Body {
if tt.SectionRef != "" {
if _, ok := sectionNumbers[tt.SectionRef]; !ok {
fmt.Fprintf(os.Stderr, "non-existing section %q is referenced from section %q\n",
tt.SectionRef, s.Number)
}
}
}
}
}

var tmpl = template.Must(template.ParseFS(templateFS, "template.gohtml"))

func processDocument(inputPath, outputPath string) (outErr error) {
data, err := os.ReadFile(inputPath)
if err != nil {
return err
}
doc, err := spec.Parse(string(data))
if err != nil {
return err
}
sectionNumbers := make(map[string]struct{})
for _, s := range doc.Sections {
sectionNumbers[s.Number] = struct{}{}
}
tocNumbers := make(map[string]struct{})
for _, t := range doc.TOC {
tocNumbers[t.Number] = struct{}{}
}
checkSectionLinks(doc, sectionNumbers, tocNumbers)
f, err := os.Create(outputPath)
if err != nil {
return err
}
defer func() {
closeErr := f.Close()
if closeErr != nil && outErr == nil {
outErr = closeErr
}
}()
return tmpl.Execute(f, templateData{
Document: doc,
LicenseHTML: formatBody(doc.License, sectionNumbers),
TOCTree: buildTOCTree(doc.TOC, sectionNumbers),
Sections: buildSections(doc.Sections, sectionNumbers),
})
}

func main() {
if len(os.Args) != 3 {
fmt.Fprintln(os.Stderr, "Usage: <cassandra_doc_dir> <output_dir>")
return
}
inputDir := os.Args[1]
outputDir := os.Args[2]
for _, name := range []string{"v5", "v4", "v3"} {
fmt.Fprintln(os.Stderr, name)
err := processDocument(filepath.Join(inputDir, fmt.Sprintf("native_protocol_%s.spec", name)),
filepath.Join(outputDir, fmt.Sprintf("native_protocol_%s.html", name)))
if err != nil {
fmt.Fprintf(os.Stderr, "%s: %v\n", name, err)
return
}
}
err := os.WriteFile(filepath.Join(outputDir, "NOTICE"), notice, 0666)
if err != nil {
fmt.Fprintf(os.Stderr, "NOTICE: %v\n", err)
return
}
}
Loading

0 comments on commit 2bb65c0

Please sign in to comment.