1
0
Fork 0
mirror of https://git.sr.ht/~adnano/kiln synced 2024-06-12 20:46:40 +02:00
kiln/kiln.go

431 lines
9.6 KiB
Go
Raw Normal View History

2020-09-23 01:46:30 +02:00
package main
import (
2020-09-29 16:57:15 +02:00
"bytes"
2020-09-29 20:10:49 +02:00
"fmt"
"html"
2020-09-23 01:46:30 +02:00
"io/ioutil"
"os"
"path/filepath"
"regexp"
2020-09-29 17:22:54 +02:00
"sort"
2020-09-23 01:46:30 +02:00
"strings"
"text/template"
"time"
2020-09-29 20:10:49 +02:00
"git.sr.ht/~adnano/gmi"
2020-09-23 01:46:30 +02:00
)
2020-09-23 19:50:25 +02:00
// Site represents a kiln site.
2020-09-23 01:46:30 +02:00
type Site struct {
2020-09-29 22:42:27 +02:00
URL string // Site URL
2020-09-23 03:11:56 +02:00
Directory *Directory // Site directory
Templates *template.Template // Templates
2020-09-23 01:46:30 +02:00
}
2020-09-29 22:42:27 +02:00
// Load loads the Site from the specified source directory.
func (s *Site) Load(srcDir string) error {
s.Directory = NewDirectory("")
2020-09-23 01:46:30 +02:00
tmpl, err := template.New("templates").ParseGlob("templates/*.gmi")
if err != nil {
2020-09-29 22:42:27 +02:00
return err
2020-09-23 01:46:30 +02:00
}
2020-09-29 22:42:27 +02:00
s.Templates = tmpl
2020-09-23 01:46:30 +02:00
2020-09-29 22:42:27 +02:00
if err := site.Directory.Read(""); err != nil {
return err
2020-09-23 01:46:30 +02:00
}
2020-09-29 22:42:27 +02:00
return nil
2020-09-23 01:46:30 +02:00
}
// Write writes the contents of the Index to the provided destination directory.
2020-09-29 17:22:54 +02:00
func (s *Site) Write(dstDir string, format OutputFormat) error {
2020-09-23 01:46:30 +02:00
// Empty the destination directory
if err := os.RemoveAll(dstDir); err != nil {
return err
}
2020-09-23 03:11:56 +02:00
// Create the destination directory
2020-09-23 01:46:30 +02:00
if err := os.MkdirAll(dstDir, 0755); err != nil {
return err
}
2020-09-23 03:11:56 +02:00
// Write the directory
2020-09-29 17:22:54 +02:00
return s.Directory.Write(dstDir, format)
2020-09-23 01:46:30 +02:00
}
// Manipulate processes and manipulates the site's content.
2020-09-23 03:11:56 +02:00
func (s *Site) Manipulate(dir *Directory) error {
// Write the directory index file, if it doesn't exist
if dir.Index == nil {
path := filepath.Join(dir.Permalink, "index.gmi")
2020-09-29 16:57:15 +02:00
var b bytes.Buffer
tmpl := s.Templates.Lookup("index.gmi")
2020-09-29 22:42:27 +02:00
if tmpl == nil {
tmpl = indexTmpl
}
if err := tmpl.Execute(&b, dir); err != nil {
return err
}
content := b.Bytes()
permalink := filepath.Dir(path)
if permalink == "." {
permalink = ""
2020-09-23 03:11:56 +02:00
}
2020-09-29 22:42:27 +02:00
page := &Page{
Permalink: "/" + permalink,
content: content,
}
dir.Index = page
2020-09-23 03:11:56 +02:00
}
2020-09-23 01:46:30 +02:00
2020-09-23 03:11:56 +02:00
// Manipulate pages
for i := range dir.Pages {
2020-09-29 16:57:15 +02:00
var b bytes.Buffer
2020-09-23 03:11:56 +02:00
tmpl := s.Templates.Lookup("page.gmi")
2020-09-29 22:42:27 +02:00
if tmpl == nil {
tmpl = pageTmpl
}
2020-09-29 16:57:15 +02:00
if err := tmpl.Execute(&b, dir.Pages[i]); err != nil {
2020-09-23 01:46:30 +02:00
return err
}
2020-09-29 16:57:15 +02:00
dir.Pages[i].content = b.Bytes()
2020-09-23 01:46:30 +02:00
}
return nil
}
2020-09-29 17:22:54 +02:00
// Sort sorts the site's pages by date.
func (s *Site) Sort() {
s.Directory.Sort()
}
2020-09-29 22:42:27 +02:00
type Feed struct {
Title string
Path string
SiteURL string
Updated time.Time
Entries []*Page
}
// CreateFeeds creates Atom feeds.
func (s *Site) CreateFeeds() error {
const atomPath = "atom.xml"
var b bytes.Buffer
tmpl := s.Templates.Lookup(atomPath)
if tmpl == nil {
tmpl = atomTmpl
}
feed := &Feed{
Title: "Example Feed",
Path: filepath.Join(s.Directory.Permalink, atomPath),
SiteURL: s.URL,
Updated: time.Now(),
Entries: s.Directory.Pages,
}
if err := tmpl.Execute(&b, feed); err != nil {
return err
}
path := filepath.Join(s.Directory.Permalink, atomPath)
s.Directory.Static[path] = b.Bytes()
return nil
}
2020-09-23 01:46:30 +02:00
// Page represents a page.
type Page struct {
// The permalink to this page.
Permalink string
// The title of this page, parsed from the Gemini contents.
Title string
// The date of the page. Dates are specified in the filename.
// Ex: 2020-09-22-hello-world.gmi
Date time.Time
// The content of this page.
2020-09-29 16:57:15 +02:00
content []byte
}
func (p *Page) Content() string {
return string(p.content)
2020-09-23 01:46:30 +02:00
}
2020-09-23 03:15:03 +02:00
var titleRE = regexp.MustCompile("^# ?([^#\r\n]+)\r?\n?\r?\n?")
2020-09-23 01:46:30 +02:00
2020-09-23 19:50:25 +02:00
// NewPage returns a new Page with the given path and content.
2020-09-29 16:57:15 +02:00
func NewPage(path string, content []byte) *Page {
2020-09-23 01:46:30 +02:00
// Try to parse the date from the page filename
2020-09-23 03:11:56 +02:00
var date time.Time
2020-09-23 01:46:30 +02:00
const layout = "2006-01-02"
base := filepath.Base(path)
if len(base) >= len(layout) {
dateStr := base[:len(layout)]
2020-09-23 03:11:56 +02:00
if time, err := time.Parse(layout, dateStr); err == nil {
date = time
}
// Remove the date from the path
base = base[len(layout):]
if len(base) > 0 {
// Remove a leading dash
if base[0] == '-' {
base = base[1:]
}
if len(base) > 0 {
dir := filepath.Dir(path)
if dir == "." {
dir = ""
}
path = filepath.Join(dir, base)
}
2020-09-23 01:46:30 +02:00
}
}
// Try to parse the title from the contents
2020-09-23 03:11:56 +02:00
var title string
2020-09-29 16:57:15 +02:00
if submatches := titleRE.FindSubmatch(content); submatches != nil {
title = string(submatches[1])
2020-09-23 03:11:56 +02:00
// Remove the title from the contents
content = content[len(submatches[0]):]
2020-09-23 01:46:30 +02:00
}
2020-09-29 16:57:15 +02:00
permalink := strings.TrimSuffix(path, ".gmi")
2020-09-23 03:11:56 +02:00
return &Page{
2020-09-29 16:57:15 +02:00
Permalink: "/" + permalink + "/",
2020-09-23 03:11:56 +02:00
Title: title,
Date: date,
2020-09-29 16:57:15 +02:00
content: content,
2020-09-23 03:11:56 +02:00
}
2020-09-23 01:46:30 +02:00
}
2020-09-23 03:11:56 +02:00
// Directory represents a directory of pages.
type Directory struct {
// The permalink to this directory.
2020-09-23 01:46:30 +02:00
Permalink string
2020-09-23 03:11:56 +02:00
// The pages in this directory.
2020-09-23 01:46:30 +02:00
Pages []*Page
2020-09-23 03:11:56 +02:00
// The subdirectories of this directory.
Directories []*Directory
// The index file (index.gmi).
Index *Page
2020-09-29 22:42:27 +02:00
// Static files
Static map[string][]byte
2020-09-23 01:46:30 +02:00
}
2020-09-23 19:50:25 +02:00
// NewDirectory returns a new Directory with the given path.
2020-09-23 03:11:56 +02:00
func NewDirectory(path string) *Directory {
2020-09-23 01:46:30 +02:00
var permalink string
if path == "" {
permalink = "/"
} else {
permalink = "/" + path + "/"
}
2020-09-23 03:11:56 +02:00
return &Directory{
2020-09-23 01:46:30 +02:00
Permalink: permalink,
2020-09-29 22:42:27 +02:00
Static: map[string][]byte{},
2020-09-23 01:46:30 +02:00
}
}
2020-09-23 03:11:56 +02:00
// Read reads from a directory and indexes the files and directories within it.
2020-09-29 22:42:27 +02:00
func (d *Directory) Read(path string) error {
entries, err := ioutil.ReadDir(filepath.Join("src", path))
2020-09-23 03:11:56 +02:00
if err != nil {
return err
}
for _, entry := range entries {
name := entry.Name()
if entry.IsDir() {
// Gather directory data
2020-09-29 22:42:27 +02:00
path := filepath.Join(path, name)
2020-09-23 03:11:56 +02:00
dir := NewDirectory(path)
2020-09-29 22:42:27 +02:00
if err := dir.Read(path); err != nil {
return err
}
2020-09-23 03:11:56 +02:00
d.Directories = append(d.Directories, dir)
} else {
2020-09-29 22:42:27 +02:00
srcPath := filepath.Join("src", path, name)
2020-09-23 03:11:56 +02:00
content, err := ioutil.ReadFile(srcPath)
if err != nil {
return err
}
switch filepath.Ext(name) {
case ".gmi", ".gemini":
2020-09-29 22:42:27 +02:00
path := filepath.Join(path, name)
2020-09-23 03:11:56 +02:00
// Gather page data
page := NewPage(path, content)
if name == "index.gmi" {
d.Index = page
} else {
d.Pages = append(d.Pages, page)
}
default:
// Static file
2020-09-29 22:42:27 +02:00
path := filepath.Join(path, name)
d.Static[path] = content
2020-09-23 03:11:56 +02:00
}
}
}
return nil
}
// Write writes the Directory to the provided destination path.
2020-09-29 16:57:15 +02:00
func (d *Directory) Write(dstDir string, format OutputFormat) error {
2020-09-23 03:11:56 +02:00
// Create the directory
2020-09-29 20:10:49 +02:00
dirPath := filepath.Join(dstDir, d.Permalink)
2020-09-23 03:11:56 +02:00
if err := os.MkdirAll(dirPath, 0755); err != nil {
return err
}
2020-09-29 22:42:27 +02:00
// Write static files
for path := range d.Static {
dstPath := filepath.Join(dstDir, path)
f, err := os.Create(dstPath)
if err != nil {
return err
}
data := d.Static[path]
if _, err := f.Write(data); err != nil {
return err
}
}
2020-09-23 03:11:56 +02:00
// Write the files
for _, page := range d.Pages {
2020-09-29 16:57:15 +02:00
path, content := format(page)
dstPath := filepath.Join(dstDir, path)
dir := filepath.Dir(dstPath)
os.MkdirAll(dir, 0755)
2020-09-23 03:11:56 +02:00
f, err := os.Create(dstPath)
if err != nil {
return err
}
2020-09-29 16:57:15 +02:00
if _, err := f.Write(content); err != nil {
2020-09-23 03:11:56 +02:00
return err
}
}
// Write the index file
if d.Index != nil {
2020-09-29 16:57:15 +02:00
path, content := format(d.Index)
dstPath := filepath.Join(dstDir, path)
2020-09-23 03:11:56 +02:00
f, err := os.Create(dstPath)
if err != nil {
return err
}
2020-09-29 16:57:15 +02:00
if _, err := f.Write(content); err != nil {
2020-09-23 03:11:56 +02:00
return err
}
}
// Write subdirectories
for _, dir := range d.Directories {
2020-09-29 16:57:15 +02:00
dir.Write(dstDir, format)
2020-09-23 03:11:56 +02:00
}
return nil
2020-09-23 01:46:30 +02:00
}
2020-09-29 16:57:15 +02:00
2020-09-29 17:22:54 +02:00
// Sort sorts the directory's pages by date.
func (d *Directory) Sort() {
sort.Slice(d.Pages, func(i, j int) bool {
return d.Pages[i].Date.After(d.Pages[j].Date)
})
// Sort subdirectories
for _, d := range d.Directories {
d.Sort()
}
}
2020-09-29 16:57:15 +02:00
// OutputFormat represents an output format.
type OutputFormat func(*Page) (path string, content []byte)
func OutputGemini(p *Page) (path string, content []byte) {
const indexPath = "index.gmi"
path = filepath.Join(p.Permalink, indexPath)
content = p.content
return
}
func OutputHTML(p *Page) (path string, content []byte) {
const indexPath = "index.html"
2020-09-29 20:10:49 +02:00
const meta = `<!DOCTYPE html>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width,initial-scale=1">
<link rel="stylesheet" href="/style.css">
<title>%s</title>
`
2020-09-29 16:57:15 +02:00
path = filepath.Join(p.Permalink, indexPath)
2020-09-29 20:10:49 +02:00
var b bytes.Buffer
2020-09-29 20:12:40 +02:00
fmt.Fprintf(&b, meta, html.EscapeString(p.Title))
2020-09-29 20:10:49 +02:00
var pre bool
var list bool
2020-09-29 16:57:15 +02:00
r := bytes.NewReader(p.content)
2020-09-29 20:10:49 +02:00
text := gmi.Parse(r)
for _, l := range text {
if _, ok := l.(gmi.LineListItem); ok {
if !list {
list = true
fmt.Fprint(&b, "<ul>\n")
}
} else if list {
list = false
fmt.Fprint(&b, "</ul>\n")
}
switch l.(type) {
case gmi.LineLink:
link := l.(gmi.LineLink)
url := html.EscapeString(link.URL)
name := html.EscapeString(link.Name)
if name == "" {
name = url
}
fmt.Fprintf(&b, "<p><a href='%s'>%s</a></p>\n", url, name)
case gmi.LinePreformattingToggle:
pre = !pre
if pre {
fmt.Fprint(&b, "<pre>\n")
} else {
fmt.Fprint(&b, "</pre>\n")
}
case gmi.LinePreformattedText:
text := string(l.(gmi.LinePreformattedText))
fmt.Fprintf(&b, "%s\n", html.EscapeString(text))
case gmi.LineHeading1:
text := string(l.(gmi.LineHeading1))
fmt.Fprintf(&b, "<h1>%s</h1>\n", html.EscapeString(text))
case gmi.LineHeading2:
text := string(l.(gmi.LineHeading2))
fmt.Fprintf(&b, "<h2>%s</h2>\n", html.EscapeString(text))
case gmi.LineHeading3:
text := string(l.(gmi.LineHeading3))
fmt.Fprintf(&b, "<h3>%s</h3>\n", html.EscapeString(text))
case gmi.LineListItem:
text := string(l.(gmi.LineListItem))
fmt.Fprintf(&b, "<li>%s</li>\n", html.EscapeString(text))
case gmi.LineQuote:
text := string(l.(gmi.LineQuote))
fmt.Fprintf(&b, "<blockquote>%s</blockquote>\n", html.EscapeString(text))
case gmi.LineText:
text := string(l.(gmi.LineText))
if text == "" {
fmt.Fprint(&b, "<br>\n")
} else {
fmt.Fprintf(&b, "<p>%s</p>\n", html.EscapeString(text))
}
}
}
if pre {
fmt.Fprint(&b, "</pre>\n")
}
if list {
fmt.Fprint(&b, "</ul>\n")
}
content = b.Bytes()
2020-09-29 16:57:15 +02:00
return
}