mirror of
https://github.com/danog/gojekyll.git
synced 2024-11-30 08:39:01 +01:00
116 lines
2.6 KiB
Go
116 lines
2.6 KiB
Go
package site
|
||
|
||
import (
|
||
"bytes"
|
||
"fmt"
|
||
"io"
|
||
"os"
|
||
"path/filepath"
|
||
|
||
"github.com/osteele/gojekyll/pages"
|
||
"github.com/osteele/gojekyll/plugins"
|
||
"github.com/osteele/gojekyll/utils"
|
||
)
|
||
|
||
// Write cleans the destination and writes files into it.
|
||
// It sets TZ from the site config.
|
||
func (s *Site) Write() (int, error) {
|
||
if err := s.setTimeZone(); err != nil {
|
||
return 0, err
|
||
}
|
||
if err := s.ensureRendered(); err != nil {
|
||
return 0, err
|
||
}
|
||
if err := s.Clean(); err != nil {
|
||
return 0, err
|
||
}
|
||
return s.WriteFiles()
|
||
}
|
||
|
||
// WriteFiles writes output files.
|
||
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.
|
||
func (s *Site) WriteDoc(d pages.Document) error {
|
||
from := d.SourcePath()
|
||
rel := d.Permalink()
|
||
if !d.Static() && filepath.Ext(rel) == "" {
|
||
rel = filepath.Join(rel, "index.html")
|
||
}
|
||
to := filepath.Join(s.DestDir(), rel)
|
||
if s.cfg.Verbose {
|
||
fmt.Println("create", to, "from", d.SourcePath())
|
||
}
|
||
if s.cfg.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 rendered document.
|
||
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 rendered page. It is called as part of site.Write,
|
||
// but also, in an incremental build, to write a single page – therefore it
|
||
// also ensures that all pages have been rendered before writing this one.
|
||
func (s *Site) WritePage(w io.Writer, p pages.Page) error {
|
||
if err := s.ensureRendered(); 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) (err error) {
|
||
b, err = p.PostRender(b)
|
||
return
|
||
})
|
||
if err != nil {
|
||
return err
|
||
}
|
||
_, err = w.Write(b)
|
||
return err
|
||
}
|