package main
import (
"fmt"
"golang.org/x/net/html"
"golang.org/x/net/html/atom"
"io/fs"
"log"
"os"
"path"
"strings"
)
func fixNode(s fs.FS, n *html.Node, fixed *bool) {
if n.Type != html.ElementNode || n.DataAtom != atom.A {
goto recurse
}
for ix := range n.Attr {
a := n.Attr[ix]
if a.Key != "href" {
continue
}
if strings.HasPrefix(a.Val, "/") || strings.Contains(a.Val, "://") {
break
}
info, err := fs.Stat(s, strings.TrimSuffix(a.Val, "/"))
if err != nil {
break
}
if !info.IsDir() {
break
}
*fixed = true
if strings.HasSuffix(a.Val, "/") {
n.Attr[ix].Val = fmt.Sprintf("%vindex.html", a.Val)
} else {
n.Attr[ix].Val = fmt.Sprintf("%v/index.html", a.Val)
}
break
}
recurse:
for c := n.FirstChild; c != nil; c = c.NextSibling {
fixNode(s, c, fixed)
}
}
func main() {
start := "."
here := os.DirFS(start)
err := fs.WalkDir(here, ".", func(p string, d fs.DirEntry, err error) error {
if d.IsDir() && d.Name() == ".git" {
return fs.SkipDir
}
if d.IsDir() {
return nil
}
if !strings.HasSuffix(d.Name(), ".html") && !strings.HasSuffix(d.Name(), ".htm") {
return nil
}
parent := here
if p != d.Name() {
pname := strings.TrimSuffix(p, "/" + d.Name())
parent, err = fs.Sub(here, pname)
if err != nil {
return err
}
}
realPath := path.Join(start, p)
file, err := os.Open(realPath)
if err != nil {
return err
}
defer file.Close()
node, err := html.Parse(file)
if err != nil {
log.Printf("warning: failed to parse %v\n", realPath)
return nil
}
fixed := false
fixNode(parent, node, &fixed)
if fixed {
log.Printf("fixed %v", p)
}
// This code is mainly intended for build process, so no write/rename dance.
fileW, err := os.OpenFile(realPath, os.O_RDWR|os.O_CREATE, 0755)
if err != nil {
return nil
}
defer fileW.Close()
html.Render(fileW, node)
return nil
})
if err != nil {
log.Fatalf("failed to walk directory: %v", err)
}
}