Skip to content

Commit

Permalink
Add directory based archetypes
Browse files Browse the repository at this point in the history
  • Loading branch information
bep committed Sep 19, 2018
1 parent 058cc6c commit 4a60c4f
Show file tree
Hide file tree
Showing 8 changed files with 239 additions and 51 deletions.
31 changes: 25 additions & 6 deletions commands/new.go
Original file line number Diff line number Diff line change
Expand Up @@ -86,6 +86,26 @@ func (n *newCmd) newContent(cmd *cobra.Command, args []string) error {
var kind string

createPath, kind = newContentPathSection(createPath)
hasDot := strings.Contains(createPath, ".")

langMatch := func(s *hugolib.Site) bool {
lang := s.Language.Lang

if hasDot {
// Most likely a file.
if strings.Contains(createPath, "."+lang) {
return true
}
}

// Check for language content dir
contentDir := filepath.ToSlash(s.PathSpec.ContentDir)
if strings.HasPrefix(createPath, contentDir) {
return true
}

return false
}

if n.contentType != "" {
kind = n.contentType
Expand All @@ -100,21 +120,20 @@ func (n *newCmd) newContent(cmd *cobra.Command, args []string) error {

// If a site isn't in use in the archetype template, we can skip the build.
siteFactory := func(filename string, siteUsed bool) (*hugolib.Site, error) {
if !siteUsed {
return hugolib.NewSite(*cfg)
}
var s *hugolib.Site

if err := c.hugo.Build(hugolib.BuildCfg{SkipRender: true}); err != nil {
return nil, err
if siteUsed {
if err := c.hugo.Build(hugolib.BuildCfg{SkipRender: true}); err != nil {
return nil, err
}
}

s = c.hugo.Sites[0]

if len(c.hugo.Sites) > 1 {
// Find the best match.
for _, ss := range c.hugo.Sites {
if strings.Contains(createPath, "."+ss.Language.Lang) {
if langMatch(ss) {
s = ss
break
}
Expand Down
198 changes: 157 additions & 41 deletions create/content.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,12 +17,14 @@ package create
import (
"bytes"
"fmt"
"io"
"os"
"os/exec"
"path/filepath"

"github.com/gohugoio/hugo/helpers"
"github.com/gohugoio/hugo/hugolib"
"github.com/spf13/afero"
jww "github.com/spf13/jwalterweatherman"
)

Expand All @@ -32,56 +34,49 @@ func NewContent(
ps *helpers.PathSpec,
siteFactory func(filename string, siteUsed bool) (*hugolib.Site, error), kind, targetPath string) error {
ext := helpers.Ext(targetPath)
fs := ps.BaseFs.SourceFilesystems.Archetypes.Fs
archetypeFs := ps.BaseFs.SourceFilesystems.Archetypes.Fs

jww.INFO.Printf("attempting to create %q of %q of ext %q", targetPath, kind, ext)

archetypeFilename := findArchetype(ps, kind, ext)
archetypeFilename, isDir := findArchetype(ps, kind, ext)

if isDir {
cm, err := mapArcheTypeDir(ps, archetypeFs, archetypeFilename)
if err != nil {
return err
}
s, err := siteFactory(targetPath, cm.siteUsed)
if err != nil {
return err
}
contentPath := resolveContentPath(s, s.Fs.Source, targetPath)
return newContentFromDir(kind, s, archetypeFs, s.Fs.Source, cm, contentPath)
}

// Building the sites can be expensive, so only do it if really needed.
siteUsed := false

if archetypeFilename != "" {
f, err := fs.Open(archetypeFilename)
var err error
siteUsed, err = usesSiteVar(archetypeFs, archetypeFilename)
if err != nil {
return fmt.Errorf("failed to open archetype file: %s", err)
return err
}
defer f.Close()

if helpers.ReaderContains(f, []byte(".Site")) {
siteUsed = true
}
}

s, err := siteFactory(targetPath, siteUsed)
if err != nil {
return err
}

var content []byte
contentPath := resolveContentPath(s, archetypeFs, targetPath)

content, err = executeArcheTypeAsTemplate(s, kind, targetPath, archetypeFilename)
content, err := executeArcheTypeAsTemplate(s, kind, targetPath, archetypeFilename)
if err != nil {
return err
}

// The site may have multiple content dirs, and we currently do not know which contentDir the
// user wants to create this content in. We should improve on this, but we start by testing if the
// provided path points to an existing dir. If so, use it as is.
var contentPath string
var exists bool
targetDir := filepath.Dir(targetPath)

if targetDir != "" && targetDir != "." {
exists, _ = helpers.Exists(targetDir, fs)
}

if exists {
contentPath = targetPath
} else {
contentPath = s.PathSpec.AbsPathify(filepath.Join(s.Cfg.GetString("contentDir"), targetPath))
}

if err := helpers.SafeWriteToDisk(contentPath, bytes.NewReader(content), s.Fs.Source); err != nil {
return err
}
Expand All @@ -103,29 +98,150 @@ func NewContent(
return nil
}

func newContentFromDir(
kind string,
s *hugolib.Site,
sourceFs, targetFs afero.Fs,
cm archetypeMap, targetPath string) error {

for _, filename := range cm.otherFiles {
// Just copy the file to destination.
in, err := sourceFs.Open(filename)
if err != nil {
return err
}

targetFilename := filepath.Join(targetPath, filename)
targetDir := filepath.Dir(targetFilename)
if err := targetFs.MkdirAll(targetDir, 0777); err != nil && !os.IsExist(err) {
return fmt.Errorf("failed to create target directory for %s: %s", targetDir, err)
}

out, err := targetFs.Create(targetFilename)

_, err = io.Copy(out, in)
if err != nil {
return err
}

in.Close()
out.Close()
}

for _, filename := range cm.contentFiles {
targetFilename := filepath.Join(targetPath, filename)
content, err := executeArcheTypeAsTemplate(s, kind, targetFilename, filename)
if err != nil {
return err
}

if err := helpers.SafeWriteToDisk(targetFilename, bytes.NewReader(content), targetFs); err != nil {
return err
}
}

jww.FEEDBACK.Println(targetPath, "created")

return nil
}

type archetypeMap struct {
// These needs to be parsed and executed as Go templates.
contentFiles []string
// These are just copied to destination.
otherFiles []string
// If the templates needs a fully built site. This can potentially be
// expensive, so only do when needed.
siteUsed bool
}

func mapArcheTypeDir(
ps *helpers.PathSpec,
fs afero.Fs,
archetypeDir string) (archetypeMap, error) {

var m archetypeMap

walkFn := func(filename string, fi os.FileInfo, err error) error {
if err != nil {
return err
}

if fi.IsDir() {
return nil
}

if hugolib.IsContentFile(filename) {
m.contentFiles = append(m.contentFiles, filename)
if !m.siteUsed {
m.siteUsed, err = usesSiteVar(fs, filename)
if err != nil {
return err
}
}
return nil
}

m.otherFiles = append(m.otherFiles, filename)

return nil
}

if err := helpers.SymbolicWalk(fs, archetypeDir, walkFn); err != nil {
return m, err
}

return m, nil
}

func usesSiteVar(fs afero.Fs, filename string) (bool, error) {
f, err := fs.Open(filename)
if err != nil {
return false, fmt.Errorf("failed to open archetype file: %s", err)
}
defer f.Close()
return helpers.ReaderContains(f, []byte(".Site")), nil
}

func resolveContentPath(s *hugolib.Site, fs afero.Fs, targetPath string) string {
// The site may have multiple content dirs, and we currently do not know which contentDir the
// user wants to create this content in. We should improve on this, but we start by testing if the
// provided path points to an existing dir. If so, use it as is.
var contentPath string
var exists bool
targetDir := filepath.Dir(targetPath)

if targetDir != "" && targetDir != "." {
exists, _ = helpers.Exists(targetDir, fs)
}

if exists {
contentPath = targetPath
} else {
contentPath = s.PathSpec.AbsPathify(filepath.Join(s.Cfg.GetString("contentDir"), targetPath))
}

return contentPath
}

// FindArchetype takes a given kind/archetype of content and returns the path
// to the archetype in the archetype filesystem, blank if none found.
func findArchetype(ps *helpers.PathSpec, kind, ext string) (outpath string) {
func findArchetype(ps *helpers.PathSpec, kind, ext string) (outpath string, isDir bool) {
fs := ps.BaseFs.Archetypes.Fs

// If the new content isn't in a subdirectory, kind == "".
// Therefore it should be excluded otherwise `is a directory`
// error will occur. github.com/gohugoio/hugo/issues/411
var pathsToCheck = []string{"default"}
var pathsToCheck []string

if ext != "" {
if kind != "" {
pathsToCheck = append([]string{kind + ext, "default" + ext}, pathsToCheck...)
} else {
pathsToCheck = append([]string{"default" + ext}, pathsToCheck...)
}
if kind != "" {
pathsToCheck = append(pathsToCheck, kind+ext)
}
pathsToCheck = append(pathsToCheck, "default"+ext)

for _, p := range pathsToCheck {
if exists, _ := helpers.Exists(p, fs); exists {
return p
fi, err := fs.Stat(p)
if err == nil {
return p, fi.IsDir()
}
}

return ""
return "", false
}
40 changes: 40 additions & 0 deletions create/content_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -79,6 +79,40 @@ func TestNewContent(t *testing.T) {
}
}

func TestNewContentFromDir(t *testing.T) {
assert := require.New(t)
cfg, fs := newTestCfg()
assert.NoError(initFs(fs))

archetypeDir := filepath.Join("archetypes", "my-bundle")
assert.NoError(fs.Source.Mkdir(archetypeDir, 0755))

contentFile := `
File: %s
Site Lang: {{ .Site.Language.Lang }}
`

assert.NoError(afero.WriteFile(fs.Source, filepath.Join(archetypeDir, "index.md"), []byte(fmt.Sprintf(contentFile, "index.md")), 0755))
assert.NoError(afero.WriteFile(fs.Source, filepath.Join(archetypeDir, "pages", "bio.md"), []byte(fmt.Sprintf(contentFile, "bio.md")), 0755))
assert.NoError(afero.WriteFile(fs.Source, filepath.Join(archetypeDir, "resources", "hugo1.json"), []byte(`hugo1: {{ printf "no template handling in here" }}`), 0755))
assert.NoError(afero.WriteFile(fs.Source, filepath.Join(archetypeDir, "resources", "hugo2.xml"), []byte(`hugo2: {{ printf "no template handling in here" }}`), 0755))

h, err := hugolib.NewHugoSites(deps.DepsCfg{Cfg: cfg, Fs: fs})
assert.NoError(err)

siteFactory := func(filename string, siteUsed bool) (*hugolib.Site, error) {
return h.Sites[0], nil
}

assert.NoError(create.NewContent(h.PathSpec, siteFactory, "my-bundle", "post/my-post"))

assertContains(assert, readFileFromFs(t, fs.Source, filepath.Join("content", "post/my-post/my-bundle/resources/hugo1.json")), `hugo1: {{ printf "no template handling in here" }}`)
assertContains(assert, readFileFromFs(t, fs.Source, filepath.Join("content", "post/my-post/my-bundle/resources/hugo2.xml")), `hugo2: {{ printf "no template handling in here" }}`)
assertContains(assert, readFileFromFs(t, fs.Source, filepath.Join("content", "post/my-post/my-bundle/index.md")), `File: index.md`, `Site Lang: en`)
assertContains(assert, readFileFromFs(t, fs.Source, filepath.Join("content", "post/my-post/my-bundle/pages/bio.md")), `File: bio.md`, `Site Lang: en`)

}

func initViper(v *viper.Viper) {
v.Set("metaDataFormat", "toml")
v.Set("archetypeDir", "archetypes")
Expand Down Expand Up @@ -166,6 +200,12 @@ Some text.
return nil
}

func assertContains(assert *require.Assertions, v interface{}, matches ...string) {
for _, m := range matches {
assert.Contains(v, m)
}
}

// TODO(bep) extract common testing package with this and some others
func readFileFromFs(t *testing.T, fs afero.Fs, filename string) string {
filename = filepath.FromSlash(filename)
Expand Down
4 changes: 4 additions & 0 deletions go.mod
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ require (
github.com/chaseadamsio/goorgeous v1.1.0
github.com/cpuguy83/go-md2man v1.0.8 // indirect
github.com/danwakefield/fnmatch v0.0.0-20160403171240-cbb64ac3d964 // indirect
github.com/derekparker/delve v1.1.0 // indirect
github.com/disintegration/imaging v1.5.0
github.com/dlclark/regexp2 v1.1.6 // indirect
github.com/eknkc/amber v0.0.0-20171010120322-cdade1c07385
Expand Down Expand Up @@ -55,12 +56,15 @@ require (
github.com/tdewolff/minify v2.3.5+incompatible
github.com/tdewolff/parse v2.3.3+incompatible // indirect
github.com/tdewolff/test v0.0.0-20171106182207-265427085153 // indirect
github.com/visualfc/gocode v0.0.0-20180902020244-08a66b5525c8 // indirect
github.com/visualfc/gotools v0.0.0-20180902041808-77fd0f0b4437 // indirect
github.com/wellington/go-libsass v0.0.0-20180624165032-615eaa47ef79 // indirect
github.com/yosssi/ace v0.0.5
golang.org/x/image v0.0.0-20180708004352-c73c2afc3b81
golang.org/x/net v0.0.0-20180906233101-161cd47e91fd
golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f
golang.org/x/text v0.3.0
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e // indirect
gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127 // indirect
gopkg.in/yaml.v2 v2.2.1
)
Expand Down
Loading

0 comments on commit 4a60c4f

Please sign in to comment.