1
0
mirror of https://github.com/danog/gojekyll.git synced 2024-11-27 00:34:42 +01:00
gojekyll/site/write.go
2017-08-03 16:24:40 +02:00

129 lines
2.6 KiB
Go

package site
import (
"bytes"
"fmt"
"io"
"os"
"path/filepath"
"strings"
"github.com/osteele/gojekyll/pages"
"github.com/osteele/gojekyll/plugins"
"github.com/osteele/gojekyll/utils"
)
// WriteFiles writes output files.
// It attends to options.dry_run.
func (s *Site) WriteFiles() (count int, err error) {
errs := make(chan error)
// without this, large sites run out of file descriptors
sem := make(chan bool, 20)
for i, n := 0, cap(sem); i < n; i++ {
sem <- true
}
for _, d := range s.OutputDocs() {
count++
go func(d pages.Document) {
<-sem
errs <- s.WriteDoc(d)
sem <- true
}(d)
}
var errList []error
for i := 0; i < count; i++ {
if e := <-errs; e != nil {
errList = append(errList, e)
}
}
return count, combineErrors(errList)
}
// WriteDoc writes a document to the destination directory.
// It attends to options.dry_run.
func (s *Site) WriteDoc(d pages.Document) error {
from := d.SourcePath()
to := filepath.Join(s.DestDir(), d.Permalink())
if !d.Static() && filepath.Ext(to) == "" {
to = filepath.Join(to, "index.html")
}
if s.config.Verbose {
fmt.Println("create", to, "from", d.SourcePath())
}
if s.config.DryRun {
// FIXME render the page, just don't write it
return nil
}
// nolint: gas
if err := os.MkdirAll(filepath.Dir(to), 0755); err != nil {
return err
}
switch {
case d.Static():
return utils.CopyFileContents(to, from, 0644)
default:
return utils.VisitCreatedFile(to, func(w io.Writer) error {
return s.WriteDocument(w, d)
})
}
}
// WriteDocument writes the document to a writer.
func (s *Site) WriteDocument(w io.Writer, d pages.Document) error {
switch p := d.(type) {
case pages.Page:
return s.WritePage(w, p)
default:
return d.Write(w)
}
}
// WritePage writes the page to a writer.
func (s *Site) WritePage(w io.Writer, p pages.Page) error {
if err := s.prepareRendering(); err != nil {
return err
}
buf := new(bytes.Buffer)
if err := p.Write(buf); err != nil {
return err
}
b := buf.Bytes()
err := s.runHooks(func(p plugins.Plugin) error {
b = p.PostRender(b)
return nil
})
if err != nil {
return err
}
_, err = w.Write(b)
return err
}
func combineErrors(errs []error) error {
switch len(errs) {
case 0:
return nil
case 1:
return errs[0]
default:
messages := make([]string, len(errs))
for i, e := range errs {
messages[i] = e.Error()
}
return fmt.Errorf(strings.Join(messages, "\n"))
}
}
func (s *Site) prepareRendering() error {
if !s.preparedToRender {
if err := s.initializeRenderingPipeline(); err != nil {
return err
}
if err := s.setPageContent(); err != nil {
return err
}
s.preparedToRender = true
}
return nil
}