mirror of
https://github.com/gohugoio/hugo.git
synced 2025-03-22 13:33:50 +00:00
Move alias rendering to target
This commit is contained in:
parent
74b55fc7c8
commit
2f10da1570
6 changed files with 155 additions and 26 deletions
|
@ -22,9 +22,19 @@ func (s *Site) ShowPlan(out io.Writer) (err error) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
fmt.Fprintf(out, "%s\n", trns)
|
fmt.Fprintf(out, "%s\n", trns)
|
||||||
|
|
||||||
|
if s.Alias == nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, alias := range p.Aliases {
|
||||||
|
aliasTrans, err := s.Alias.Translate(alias)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
fmt.Fprintf(out, " %s => %s\n", alias, aliasTrans)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
|
@ -76,6 +76,7 @@ type Site struct {
|
||||||
Shortcodes map[string]ShortcodeFunc
|
Shortcodes map[string]ShortcodeFunc
|
||||||
timer *nitro.B
|
timer *nitro.B
|
||||||
Target target.Output
|
Target target.Output
|
||||||
|
Alias target.Translator
|
||||||
}
|
}
|
||||||
|
|
||||||
type SiteInfo struct {
|
type SiteInfo struct {
|
||||||
|
@ -192,14 +193,18 @@ func (s *Site) initialize() {
|
||||||
Base: s.absContentDir(),
|
Base: s.absContentDir(),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
s.initializeSiteInfo()
|
||||||
|
|
||||||
|
s.Shortcodes = make(map[string]ShortcodeFunc)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *Site) initializeSiteInfo() {
|
||||||
s.Info = SiteInfo{
|
s.Info = SiteInfo{
|
||||||
BaseUrl: template.URL(s.Config.BaseUrl),
|
BaseUrl: template.URL(s.Config.BaseUrl),
|
||||||
Title: s.Config.Title,
|
Title: s.Config.Title,
|
||||||
Recent: &s.Pages,
|
Recent: &s.Pages,
|
||||||
Config: &s.Config,
|
Config: &s.Config,
|
||||||
}
|
}
|
||||||
|
|
||||||
s.Shortcodes = make(map[string]ShortcodeFunc)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check if File / Directory Exists
|
// Check if File / Directory Exists
|
||||||
|
@ -405,14 +410,10 @@ func (s *Site) RenderAliases() error {
|
||||||
t = "alias-xhtml"
|
t = "alias-xhtml"
|
||||||
}
|
}
|
||||||
content, err := s.RenderThing(p, t)
|
content, err := s.RenderThing(p, t)
|
||||||
if strings.HasSuffix(a, "/") {
|
|
||||||
a = a + "index.html"
|
|
||||||
}
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
err = s.WritePublic(a, content.Bytes())
|
if err = s.WriteAlias(a, content.Bytes()); err != nil {
|
||||||
if err != nil {
|
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -655,3 +656,19 @@ func (s *Site) WritePublic(path string, content []byte) (err error) {
|
||||||
|
|
||||||
return s.Target.Publish(path, bytes.NewReader(content))
|
return s.Target.Publish(path, bytes.NewReader(content))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *Site) WriteAlias(path string, content []byte) (err error) {
|
||||||
|
if s.Alias == nil {
|
||||||
|
s.initTarget()
|
||||||
|
s.Alias = new(target.HTMLRedirectAlias)
|
||||||
|
}
|
||||||
|
|
||||||
|
if s.Config.Verbose {
|
||||||
|
fmt.Println(path)
|
||||||
|
}
|
||||||
|
|
||||||
|
if path, err = s.Alias.Translate(path); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return s.Target.Publish(path, bytes.NewReader(content))
|
||||||
|
}
|
||||||
|
|
|
@ -7,19 +7,26 @@ import (
|
||||||
"testing"
|
"testing"
|
||||||
)
|
)
|
||||||
|
|
||||||
var fakeSource = []struct {
|
const ALIAS_DOC_1 = "---\ntitle: alias doc\naliases:\n - \"alias1/\"\n - \"alias-2/\"\n---\naliases\n"
|
||||||
|
|
||||||
|
type byteSource struct {
|
||||||
name string
|
name string
|
||||||
content []byte
|
content []byte
|
||||||
}{
|
}
|
||||||
|
|
||||||
|
var fakeSource = []byteSource{
|
||||||
{"foo/bar/file.md", []byte(SIMPLE_PAGE)},
|
{"foo/bar/file.md", []byte(SIMPLE_PAGE)},
|
||||||
|
{"alias/test/file1.md", []byte(ALIAS_DOC_1)},
|
||||||
|
//{"slug/test/file1.md", []byte(SLUG_DOC_1)},
|
||||||
}
|
}
|
||||||
|
|
||||||
type inMemorySource struct {
|
type inMemorySource struct {
|
||||||
|
byteSource []byteSource
|
||||||
}
|
}
|
||||||
|
|
||||||
func (i inMemorySource) Files() (files []*source.File) {
|
func (i *inMemorySource) Files() (files []*source.File) {
|
||||||
files = make([]*source.File, len(fakeSource))
|
files = make([]*source.File, len(i.byteSource))
|
||||||
for i, fake := range fakeSource {
|
for i, fake := range i.byteSource {
|
||||||
files[i] = &source.File{
|
files[i] = &source.File{
|
||||||
Name: fake.name,
|
Name: fake.name,
|
||||||
Contents: bytes.NewReader(fake.content),
|
Contents: bytes.NewReader(fake.content),
|
||||||
|
@ -44,38 +51,57 @@ func TestDegenerateNoFiles(t *testing.T) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestDegenerateNoTarget(t *testing.T) {
|
func TestDegenerateNoTarget(t *testing.T) {
|
||||||
s := &Site{Source: new(inMemorySource)}
|
s := &Site{
|
||||||
|
Source: &inMemorySource{fakeSource},
|
||||||
|
}
|
||||||
must(s.CreatePages())
|
must(s.CreatePages())
|
||||||
expected := "foo/bar/file.md\n canonical => !no target specified!\n"
|
expected := "foo/bar/file.md\n canonical => !no target specified!\n" +
|
||||||
|
"alias/test/file1.md\n canonical => !no target specified!\n"
|
||||||
checkShowPlanExpected(t, s, expected)
|
checkShowPlanExpected(t, s, expected)
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestFileTarget(t *testing.T) {
|
func TestFileTarget(t *testing.T) {
|
||||||
s := &Site{
|
s := &Site{
|
||||||
Source: new(inMemorySource),
|
Source: &inMemorySource{fakeSource},
|
||||||
Target: new(target.Filesystem),
|
Target: new(target.Filesystem),
|
||||||
|
Alias: new(target.HTMLRedirectAlias),
|
||||||
}
|
}
|
||||||
must(s.CreatePages())
|
must(s.CreatePages())
|
||||||
checkShowPlanExpected(t, s, "foo/bar/file.md\n canonical => foo/bar/file/index.html\n")
|
expected := "foo/bar/file.md\n canonical => foo/bar/file/index.html\n" +
|
||||||
|
"alias/test/file1.md\n" +
|
||||||
|
" canonical => alias/test/file1/index.html\n" +
|
||||||
|
" alias1/ => alias1/index.html\n" +
|
||||||
|
" alias-2/ => alias-2/index.html\n"
|
||||||
|
checkShowPlanExpected(t, s, expected)
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestFileTargetUgly(t *testing.T) {
|
func TestFileTargetUgly(t *testing.T) {
|
||||||
s := &Site{
|
s := &Site{
|
||||||
Target: &target.Filesystem{UglyUrls: true},
|
Target: &target.Filesystem{UglyUrls: true},
|
||||||
Source: new(inMemorySource),
|
Source: &inMemorySource{fakeSource},
|
||||||
|
Alias: new(target.HTMLRedirectAlias),
|
||||||
}
|
}
|
||||||
s.CreatePages()
|
s.CreatePages()
|
||||||
expected := "foo/bar/file.md\n canonical => foo/bar/file.html\n"
|
expected := "foo/bar/file.md\n canonical => foo/bar/file.html\n" +
|
||||||
|
"alias/test/file1.md\n" +
|
||||||
|
" canonical => alias/test/file1.html\n" +
|
||||||
|
" alias1/ => alias1/index.html\n" +
|
||||||
|
" alias-2/ => alias-2/index.html\n"
|
||||||
checkShowPlanExpected(t, s, expected)
|
checkShowPlanExpected(t, s, expected)
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestFileTargetPublishDir(t *testing.T) {
|
func TestFileTargetPublishDir(t *testing.T) {
|
||||||
s := &Site{
|
s := &Site{
|
||||||
Target: &target.Filesystem{PublishDir: "../public"},
|
Target: &target.Filesystem{PublishDir: "../public"},
|
||||||
Source: new(inMemorySource),
|
Source: &inMemorySource{fakeSource},
|
||||||
|
Alias: &target.HTMLRedirectAlias{PublishDir: "../public"},
|
||||||
}
|
}
|
||||||
|
|
||||||
must(s.CreatePages())
|
must(s.CreatePages())
|
||||||
expected := "foo/bar/file.md\n canonical => ../public/foo/bar/file/index.html\n"
|
expected := "foo/bar/file.md\n canonical => ../public/foo/bar/file/index.html\n" +
|
||||||
|
"alias/test/file1.md\n" +
|
||||||
|
" canonical => ../public/alias/test/file1/index.html\n" +
|
||||||
|
" alias1/ => ../public/alias1/index.html\n" +
|
||||||
|
" alias-2/ => ../public/alias-2/index.html\n"
|
||||||
checkShowPlanExpected(t, s, expected)
|
checkShowPlanExpected(t, s, expected)
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,11 +3,12 @@ package hugolib
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"io"
|
"io"
|
||||||
"strings"
|
|
||||||
"testing"
|
"testing"
|
||||||
)
|
)
|
||||||
|
|
||||||
const SLUG_DOC_1 = "---\ntitle: slug doc 1\nslug: slug-doc-1\n---\nslug doc 1 content"
|
const SLUG_DOC_1 = "---\ntitle: slug doc 1\nslug: slug-doc-1\naliases:\n - sd1/foo/\n - sd2\n - sd3/\n - sd4.php\n---\nslug doc 1 content"
|
||||||
|
|
||||||
|
//const SLUG_DOC_1 = "---\ntitle: slug doc 1\nslug: slug-doc-1\n---\nslug doc 1 content"
|
||||||
const SLUG_DOC_2 = "---\ntitle: slug doc 2\nslug: slug-doc-2\n---\nslug doc 2 content"
|
const SLUG_DOC_2 = "---\ntitle: slug doc 2\nslug: slug-doc-2\n---\nslug doc 2 content"
|
||||||
|
|
||||||
const INDEX_TEMPLATE = "{{ range .Data.Pages }}.{{ end }}"
|
const INDEX_TEMPLATE = "{{ range .Data.Pages }}.{{ end }}"
|
||||||
|
@ -43,14 +44,25 @@ func (t *InMemoryTarget) Translate(label string) (dest string, err error) {
|
||||||
return label, nil
|
return label, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var urlFakeSource = []byteSource{
|
||||||
|
{"content/blue/doc1.md", []byte(SLUG_DOC_1)},
|
||||||
|
{"content/blue/doc2.md", []byte(SLUG_DOC_2)},
|
||||||
|
}
|
||||||
|
|
||||||
func TestPageCount(t *testing.T) {
|
func TestPageCount(t *testing.T) {
|
||||||
target := new(InMemoryTarget)
|
target := new(InMemoryTarget)
|
||||||
s := &Site{Target: target}
|
s := &Site{
|
||||||
|
Target: target,
|
||||||
|
Config: Config{UglyUrls: false},
|
||||||
|
Source: &inMemorySource{urlFakeSource},
|
||||||
|
}
|
||||||
|
s.initializeSiteInfo()
|
||||||
s.prepTemplates()
|
s.prepTemplates()
|
||||||
must(s.addTemplate("indexes/blue.html", INDEX_TEMPLATE))
|
must(s.addTemplate("indexes/blue.html", INDEX_TEMPLATE))
|
||||||
s.Pages = append(s.Pages, mustReturn(ReadFrom(strings.NewReader(SLUG_DOC_1), "content/blue/doc1.md")))
|
|
||||||
s.Pages = append(s.Pages, mustReturn(ReadFrom(strings.NewReader(SLUG_DOC_2), "content/blue/doc2.md")))
|
|
||||||
|
|
||||||
|
if err := s.CreatePages(); err != nil {
|
||||||
|
t.Errorf("Unable to create pages: %s", err)
|
||||||
|
}
|
||||||
if err := s.BuildSiteMeta(); err != nil {
|
if err := s.BuildSiteMeta(); err != nil {
|
||||||
t.Errorf("Unable to build site metadata: %s", err)
|
t.Errorf("Unable to build site metadata: %s", err)
|
||||||
}
|
}
|
||||||
|
@ -59,6 +71,10 @@ func TestPageCount(t *testing.T) {
|
||||||
t.Errorf("Unable to render site lists: %s", err)
|
t.Errorf("Unable to render site lists: %s", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if err := s.RenderAliases(); err != nil {
|
||||||
|
t.Errorf("Unable to render site lists: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
blueIndex := target.files["blue"]
|
blueIndex := target.files["blue"]
|
||||||
if blueIndex == nil {
|
if blueIndex == nil {
|
||||||
t.Errorf("No indexed rendered. %v", target.files)
|
t.Errorf("No indexed rendered. %v", target.files)
|
||||||
|
@ -67,4 +83,15 @@ func TestPageCount(t *testing.T) {
|
||||||
if len(blueIndex) != 2 {
|
if len(blueIndex) != 2 {
|
||||||
t.Errorf("Number of pages does not equal 2, got %d. %q", len(blueIndex), blueIndex)
|
t.Errorf("Number of pages does not equal 2, got %d. %q", len(blueIndex), blueIndex)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for _, s := range []string{
|
||||||
|
"sd1/foo/index.html",
|
||||||
|
"sd2",
|
||||||
|
"sd3/index.html",
|
||||||
|
"sd4.php",
|
||||||
|
} {
|
||||||
|
if _, ok := target.files[s]; !ok {
|
||||||
|
t.Errorf("No alias rendered: %s", s)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
31
target/alias_test.go
Normal file
31
target/alias_test.go
Normal file
|
@ -0,0 +1,31 @@
|
||||||
|
package target
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestHTMLRedirectAlias(t *testing.T) {
|
||||||
|
var o Translator
|
||||||
|
o = new(HTMLRedirectAlias)
|
||||||
|
|
||||||
|
tests := []struct {
|
||||||
|
value string
|
||||||
|
expected string
|
||||||
|
}{
|
||||||
|
{"", ""},
|
||||||
|
{"alias 1", "alias-1"},
|
||||||
|
{"alias 2/", "alias-2/index.html"},
|
||||||
|
{"alias 3.html", "alias-3.html"},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, test := range tests {
|
||||||
|
path, err := o.Translate(test.value)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Translate returned an error: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if path != test.expected {
|
||||||
|
t.Errorf("Expected: %s, got: %s", test.expected, path)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
18
target/htmlredirect.go
Normal file
18
target/htmlredirect.go
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
package target
|
||||||
|
|
||||||
|
import (
|
||||||
|
helpers "github.com/spf13/hugo/template"
|
||||||
|
"path"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
type HTMLRedirectAlias struct {
|
||||||
|
PublishDir string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *HTMLRedirectAlias) Translate(alias string) (aliasPath string, err error) {
|
||||||
|
if strings.HasSuffix(alias, "/") {
|
||||||
|
alias = alias + "index.html"
|
||||||
|
}
|
||||||
|
return path.Join(h.PublishDir, helpers.Urlize(alias)), nil
|
||||||
|
}
|
Loading…
Reference in a new issue