Added missing files
This commit is contained in:
parent
20d5d26e20
commit
30f42db6a6
108
gofilereader.go
Normal file
108
gofilereader.go
Normal file
|
|
@ -0,0 +1,108 @@
|
|||
package progdoc
|
||||
|
||||
import (
|
||||
"bufio"
|
||||
"io"
|
||||
"strings"
|
||||
)
|
||||
|
||||
type goFileMeta struct {
|
||||
Name string
|
||||
Value string
|
||||
}
|
||||
|
||||
type goFilePart struct {
|
||||
Meta []goFileMeta
|
||||
Body string
|
||||
}
|
||||
|
||||
type goFileDocs struct {
|
||||
Parts []goFilePart
|
||||
}
|
||||
|
||||
type allGoFileDocs struct {
|
||||
Files []goFileDocs
|
||||
}
|
||||
|
||||
func parseGoFileDocs(r io.Reader) (goFileDocs, error) {
|
||||
scanner := bufio.NewScanner(r)
|
||||
var docs goFileDocs
|
||||
var currentPart *goFilePart
|
||||
var inDocBlock bool
|
||||
|
||||
for scanner.Scan() {
|
||||
line := scanner.Text()
|
||||
trimmed := strings.TrimSpace(line)
|
||||
|
||||
// Check if this is the start of a doc block (///)
|
||||
if strings.HasPrefix(trimmed, "///") {
|
||||
// Finalize previous part if exists
|
||||
if currentPart != nil {
|
||||
currentPart.Body = strings.TrimSpace(currentPart.Body)
|
||||
docs.Parts = append(docs.Parts, *currentPart)
|
||||
}
|
||||
// Start a new part
|
||||
currentPart = &goFilePart{}
|
||||
inDocBlock = true
|
||||
|
||||
// Process the first line content after ///
|
||||
content := strings.TrimPrefix(trimmed, "///")
|
||||
content = strings.TrimPrefix(content, " ")
|
||||
if content != "" {
|
||||
processPart(currentPart, content)
|
||||
}
|
||||
continue
|
||||
}
|
||||
|
||||
// Check if this continues a doc block (//)
|
||||
if inDocBlock && strings.HasPrefix(trimmed, "//") && !strings.HasPrefix(trimmed, "///") {
|
||||
// Extract content after //
|
||||
content := strings.TrimPrefix(trimmed, "//")
|
||||
content = strings.TrimPrefix(content, " ")
|
||||
processPart(currentPart, content)
|
||||
continue
|
||||
}
|
||||
|
||||
// If we hit a non-comment line, end the doc block
|
||||
if inDocBlock && !strings.HasPrefix(trimmed, "//") {
|
||||
inDocBlock = false
|
||||
if currentPart != nil {
|
||||
currentPart.Body = strings.TrimSpace(currentPart.Body)
|
||||
docs.Parts = append(docs.Parts, *currentPart)
|
||||
currentPart = nil
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Finalize any remaining part
|
||||
if currentPart != nil {
|
||||
currentPart.Body = strings.TrimSpace(currentPart.Body)
|
||||
docs.Parts = append(docs.Parts, *currentPart)
|
||||
}
|
||||
|
||||
if err := scanner.Err(); err != nil {
|
||||
return goFileDocs{}, err
|
||||
}
|
||||
|
||||
return docs, nil
|
||||
}
|
||||
|
||||
func processPart(part *goFilePart, line string) {
|
||||
// Check if this is a meta line (starts with :)
|
||||
if strings.HasPrefix(line, ":") {
|
||||
// Parse meta value
|
||||
rest := strings.TrimPrefix(line, ":")
|
||||
fields := strings.Fields(rest)
|
||||
if len(fields) > 0 {
|
||||
name := fields[0]
|
||||
value := strings.Join(fields[1:], " ")
|
||||
part.Meta = append(part.Meta, goFileMeta{Name: name, Value: value})
|
||||
}
|
||||
} else {
|
||||
// Add to body
|
||||
if part.Body != "" {
|
||||
part.Body += "\n"
|
||||
}
|
||||
part.Body += line
|
||||
}
|
||||
}
|
||||
60
gofilereader_test.go
Normal file
60
gofilereader_test.go
Normal file
|
|
@ -0,0 +1,60 @@
|
|||
package progdoc
|
||||
|
||||
import (
|
||||
"strings"
|
||||
"testing"
|
||||
|
||||
"github.com/stretchr/testify/assert"
|
||||
)
|
||||
|
||||
func TestParseGoFile(t *testing.T) {
|
||||
sampleGoFile := `package main
|
||||
|
||||
/// :meta this is some meta
|
||||
//
|
||||
// This is some content
|
||||
|
||||
import "fmt"
|
||||
|
||||
// Ignore me
|
||||
|
||||
/// :more meta
|
||||
//
|
||||
// :arg stuff
|
||||
// :arg more stuff
|
||||
// :foo bar
|
||||
//
|
||||
// This is some more content
|
||||
//
|
||||
// this is indented
|
||||
//
|
||||
// More content goes here and there
|
||||
|
||||
func main() {
|
||||
fmt.Println("Hello, World!")
|
||||
}`
|
||||
|
||||
want := goFileDocs{
|
||||
Parts: []goFilePart{
|
||||
{
|
||||
Meta: []goFileMeta{
|
||||
{Name: "meta", Value: "this is some meta"},
|
||||
},
|
||||
Body: "This is some content",
|
||||
},
|
||||
{
|
||||
Meta: []goFileMeta{
|
||||
{Name: "more", Value: "meta"},
|
||||
{Name: "arg", Value: "stuff"},
|
||||
{Name: "arg", Value: "more stuff"},
|
||||
{Name: "foo", Value: "bar"},
|
||||
},
|
||||
Body: "This is some more content\n\n this is indented\n\nMore content goes here and there",
|
||||
},
|
||||
},
|
||||
}
|
||||
|
||||
got, err := parseGoFileDocs(strings.NewReader(sampleGoFile))
|
||||
assert.NoError(t, err)
|
||||
assert.Equal(t, want, got)
|
||||
}
|
||||
123
pathbuilder.go
Normal file
123
pathbuilder.go
Normal file
|
|
@ -0,0 +1,123 @@
|
|||
package progdoc
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"html/template"
|
||||
"log"
|
||||
"os"
|
||||
"path/filepath"
|
||||
"strings"
|
||||
)
|
||||
|
||||
func Path(path string) PathBuilder {
|
||||
return PathBuilder{path: path}
|
||||
}
|
||||
|
||||
func (pb PathBuilder) File(file string) Option {
|
||||
return Option{
|
||||
configSitemap: func(sm *siteMap) error {
|
||||
_, err := os.Stat(file)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
// TODO: support things other than markdown
|
||||
src, err := inferSourceFromFilename(file)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
log.Printf("Page '%s' -> %s", pb.path, file)
|
||||
sm.Pages = append(sm.Pages, sitePage{
|
||||
Path: pb.path,
|
||||
Source: stdLayoutSource{MainSource: src},
|
||||
})
|
||||
|
||||
return nil
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
func (pb PathBuilder) GoFiles(dir, templateFile string) Option {
|
||||
return Option{
|
||||
configSitemap: func(sm *siteMap) error {
|
||||
var allDocs allGoFileDocs
|
||||
|
||||
tmplBytes, err := os.ReadFile(templateFile)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
tmpl, err := template.New("").Parse(string(tmplBytes))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if err := filepath.Walk(dir, func(path string, info os.FileInfo, err error) error {
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if !info.IsDir() && info.Mode().IsRegular() {
|
||||
absName, err := filepath.Abs(path)
|
||||
if err != nil {
|
||||
return err
|
||||
} else if !strings.HasSuffix(absName, ".go") {
|
||||
return nil
|
||||
}
|
||||
|
||||
goBts, err := os.ReadFile(path)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
goFileDoc, err := parseGoFileDocs(bytes.NewReader(goBts))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
allDocs.Files = append(allDocs.Files, goFileDoc)
|
||||
}
|
||||
return nil
|
||||
}); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
sm.Pages = append(sm.Pages, sitePage{
|
||||
Path: pb.path,
|
||||
Source: mdTemplateSource{
|
||||
Template: tmpl,
|
||||
Data: allDocs,
|
||||
},
|
||||
})
|
||||
return nil
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
func (pb PathBuilder) Dir(dir string) Option {
|
||||
return Option{
|
||||
configSitemap: func(sm *siteMap) error {
|
||||
return filepath.Walk(dir, func(path string, info os.FileInfo, err error) error {
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if !info.IsDir() && info.Mode().IsRegular() {
|
||||
src, err := inferSourceFromFilename(path)
|
||||
if err != nil {
|
||||
return nil
|
||||
}
|
||||
|
||||
relPath, err := filepath.Rel(dir, strings.TrimSuffix(path, filepath.Ext(path)))
|
||||
if err != nil {
|
||||
return nil
|
||||
}
|
||||
targetPath := filepath.Join(pb.path, relPath)
|
||||
|
||||
log.Printf("Page '%s' -> %s", targetPath, path)
|
||||
sm.Pages = append(sm.Pages, sitePage{
|
||||
Path: relPath,
|
||||
Source: stdLayoutSource{MainSource: src},
|
||||
})
|
||||
}
|
||||
return nil
|
||||
})
|
||||
},
|
||||
}
|
||||
}
|
||||
Loading…
Reference in a new issue