Fix RelPermalink() and Urls in menus vs canonifyUrls
authorbep <bjorn.erik.pedersen@gmail.com>
Fri, 12 Dec 2014 19:28:28 +0000 (20:28 +0100)
committerbep <bjorn.erik.pedersen@gmail.com>
Fri, 23 Jan 2015 13:13:00 +0000 (14:13 +0100)
canonifyUrls=true, RelPermalink and baseUrl with sub-path did not work.

This fixes that by adding a check for canonifyUrl=trues=true in RelPermalink().

So given

- baseUrl "http://somehost.com/sub/"
- the path "some-path/file.html"

For canonifyUrls=false RelPermalink() returns "/sub/some-path/file.html"
For canonifyUrls=true RelPermalink() returns "/some-path/file.html"

In the last case, the Url will be made absolute and clickable in a later step.

This commit also makes the menu urls defined in site config releative. To make them work with canonifying of urls, the context root is prepended if canonifying is turned off.

Fixes #519
Fixes #711

docs/content/extras/menus.md
helpers/url.go
helpers/url_test.go
hugolib/menu_test.go
hugolib/node.go
hugolib/page.go
hugolib/page_permalink_test.go
hugolib/site.go

index 9574325672d544b8ca15fbb2c4fe40fb6605cdd5..fcf90096f7e88cf509e7be36b2e7a6d15d389367 100644 (file)
@@ -96,10 +96,12 @@ Here’s an example `config.toml`:
         pre = "<i class='fa fa-heart'></i>"
         weight = -110
         identifier = "about"
+        url = "/about/"
     [[menu.main]]
         name = "getting started"
         pre = "<i class='fa fa-road'></i>"
         weight = -100
+        url = "/getting-started/"
 
 And the equivalent example `config.yaml`:
 
@@ -110,11 +112,16 @@ And the equivalent example `config.yaml`:
             Pre: "<i class='fa fa-heart'></i>"
             Weight: -110
             Identifier: "about"
+            Url: "/about/"
           - Name: "getting started"
             Pre: "<i class='fa fa-road'></i>"
             Weight: -100
+            Url: "/getting-started/"
     ---            
 
+
+**NOTE:** The urls must be relative to the context root. If the `BaseUrl` is `http://example.com/mysite/`, then the urls in the menu must not include the context root `mysite`. 
+  
 ## Nesting
 
 All nesting of content is done via the `parent` field.
index 46bc951f456b9cebab7ef28ecfcf19e1a996eb29..1b7608178f95f2ed70b5b1fa126faa0c5c109be1 100644 (file)
@@ -78,6 +78,25 @@ func MakePermalink(host, plink string) *url.URL {
        return base
 }
 
+// AddContextRoot adds the context root to an URL if it's not already set.
+// For relative URL entries on sites with a base url with a context root set (i.e. http://example.com/mysite),
+// relative URLs must not include the context root if canonifyUrls is enabled. But if it's disabled, it must be set.
+func AddContextRoot(baseUrl, relativePath string) string {
+
+       url, err := url.Parse(baseUrl)
+       if err != nil {
+               panic(err)
+       }
+
+       newPath := path.Join(url.Path, relativePath)
+
+       // path strips traling slash
+       if strings.HasSuffix(relativePath, "/") {
+               newPath += "/"
+       }
+       return newPath
+}
+
 func UrlPrep(ugly bool, in string) string {
        if ugly {
                x := Uglify(SanitizeUrl(in))
index 1d5c770ead8eb1eafee22cadef24b2e43b75d5e5..3df1a05c2b8f3a02e6496dfcf17ea5ad57f1b57e 100644 (file)
@@ -68,6 +68,29 @@ func TestUrlPrep(t *testing.T) {
 
 }
 
+func TestAddContextRoot(t *testing.T) {
+       tests := []struct {
+               baseUrl  string
+               url      string
+               expected string
+       }{
+               {"http://example.com/sub/", "/foo", "/sub/foo"},
+               {"http://example.com/sub/", "/foo/index.html", "/sub/foo/index.html"},
+               {"http://example.com/sub1/sub2", "/foo", "/sub1/sub2/foo"},
+               {"http://example.com", "/foo", "/foo"},
+               // cannot guess that the context root is already added int the example below
+               {"http://example.com/sub/", "/sub/foo", "/sub/sub/foo"},
+               {"http://example.com/тря", "/трям/", "/тря/трям/"},
+       }
+
+       for _, test := range tests {
+               output := AddContextRoot(test.baseUrl, test.url)
+               if output != test.expected {
+                       t.Errorf("Expected %#v, got %#v\n", test.expected, output)
+               }
+       }
+}
+
 func TestPretty(t *testing.T) {
        assert.Equal(t, PrettifyUrlPath("/section/name.html"), "/section/name/index.html")
        assert.Equal(t, PrettifyUrlPath("/section/sub/name.html"), "/section/sub/name/index.html")
index afe881a94438cf129fd51b1ec43c94ba3d652cb5..667105f506891d8c1ae3e0af87941b2243c58e35 100644 (file)
@@ -265,18 +265,27 @@ func TestPageMenu(t *testing.T) {
 // issue #719
 func TestMenuWithUnicodeUrls(t *testing.T) {
        for _, uglyUrls := range []bool{true, false} {
-               doTestMenuWithUnicodeUrls(t, uglyUrls)
+               for _, canonifyUrls := range []bool{true, false} {
+                       doTestMenuWithUnicodeUrls(t, canonifyUrls, uglyUrls)
+               }
        }
 }
 
-func doTestMenuWithUnicodeUrls(t *testing.T, uglyUrls bool) {
+func doTestMenuWithUnicodeUrls(t *testing.T, canonifyUrls, uglyUrls bool) {
+       viper.Set("CanonifyUrls", canonifyUrls)
        viper.Set("UglyUrls", uglyUrls)
+
        ts := setupMenuTests(t, MENU_PAGE_SOURCES)
        defer resetMenuTestState(ts)
 
        unicodeRussian := ts.findTestMenuEntryById("unicode", "unicode-russian")
 
-       expectedBase := "http://foo.local/zoo/%D0%BD%D0%BE%D0%B2%D0%BE%D1%81%D1%82%D0%B8-%D0%BF%D1%80%D0%BE%D0%B5%D0%BA%D1%82%D0%B0"
+       expectedBase := "/%D0%BD%D0%BE%D0%B2%D0%BE%D1%81%D1%82%D0%B8-%D0%BF%D1%80%D0%BE%D0%B5%D0%BA%D1%82%D0%B0"
+
+       if !canonifyUrls {
+               expectedBase = "/zoo" + expectedBase
+       }
+
        var expected string
        if uglyUrls {
                expected = expectedBase + ".html"
@@ -288,6 +297,7 @@ func doTestMenuWithUnicodeUrls(t *testing.T, uglyUrls bool) {
 }
 
 func TestTaxonomyNodeMenu(t *testing.T) {
+       viper.Set("CanonifyUrls", true)
        ts := setupMenuTests(t, MENU_PAGE_SOURCES)
        defer resetMenuTestState(ts)
 
@@ -333,7 +343,7 @@ func TestHomeNodeMenu(t *testing.T) {
        defer resetMenuTestState(ts)
 
        home := ts.site.newHomeNode()
-       homeMenuEntry := &MenuEntry{Name: home.Title, Url: string(home.Permalink)}
+       homeMenuEntry := &MenuEntry{Name: home.Title, Url: home.Url}
 
        for i, this := range []struct {
                menu           string
index 0c3ed9ce634e28d791068a1860c6c812df42a35f..ccf3e88228c3d259e066a8015c1dc09b1c02d2ba 100644 (file)
@@ -38,7 +38,7 @@ func (n *Node) Now() time.Time {
 
 func (n *Node) HasMenuCurrent(menuId string, inme *MenuEntry) bool {
        if inme.HasChildren() {
-               me := MenuEntry{Name: n.Title, Url: string(n.Permalink)}
+               me := MenuEntry{Name: n.Title, Url: n.Url}
 
                for _, child := range inme.Children {
                        if me.IsSameResource(child) {
@@ -52,8 +52,7 @@ func (n *Node) HasMenuCurrent(menuId string, inme *MenuEntry) bool {
 
 func (n *Node) IsMenuCurrent(menuId string, inme *MenuEntry) bool {
 
-       me := MenuEntry{Name: n.Title, Url: string(n.Permalink)}
-
+       me := MenuEntry{Name: n.Title, Url: n.Url}
        if !me.IsSameResource(inme) {
                return false
        }
index c30728c0378c71b76c450bd85b59ed2e6da44e3a..395462d84296b3056b3e744f9c2020cfc741b47c 100644 (file)
@@ -397,6 +397,16 @@ func (p *Page) RelPermalink() (string, error) {
                return "", err
        }
 
+       if viper.GetBool("CanonifyUrls") {
+               // replacements for relpermalink with baseUrl on the form http://myhost.com/sub/ will fail later on
+               // have to return the Url relative from baseUrl
+               relpath, err := helpers.GetRelativePath(link.String(), string(p.Site.BaseUrl))
+               if err != nil {
+                       return "", err
+               }
+               return "/" + filepath.ToSlash(relpath), nil
+       }
+
        link.Scheme = ""
        link.Host = ""
        link.User = nil
@@ -549,7 +559,7 @@ func (page *Page) Menus() PageMenus {
        ret := PageMenus{}
 
        if ms, ok := page.Params["menu"]; ok {
-               link, _ := page.Permalink()
+               link, _ := page.RelPermalink()
 
                me := MenuEntry{Name: page.LinkTitle(), Weight: page.Weight, Url: link}
 
index 97b3d24c6ed95b1d3b7bfa5cc74fe879d3382b2b..b73e08721b8c5eb8aebdc0458fdba8c94e9c6853 100644 (file)
@@ -11,34 +11,39 @@ import (
 
 func TestPermalink(t *testing.T) {
        tests := []struct {
-               file        string
-               dir         string
-               base        template.URL
-               slug        string
-               url         string
-               uglyurls    bool
-               expectedAbs string
-               expectedRel string
+               file         string
+               dir          string
+               base         template.URL
+               slug         string
+               url          string
+               uglyUrls     bool
+               canonifyUrls bool
+               expectedAbs  string
+               expectedRel  string
        }{
-               {"x/y/z/boofar.md", "x/y/z", "", "", "", false, "/x/y/z/boofar/", "/x/y/z/boofar/"},
-               {"x/y/z/boofar.md", "x/y/z/", "", "", "", false, "/x/y/z/boofar/", "/x/y/z/boofar/"},
-               {"x/y/z/boofar.md", "x/y/z/", "", "boofar", "", false, "/x/y/z/boofar/", "/x/y/z/boofar/"},
-               {"x/y/z/boofar.md", "x/y/z", "http://barnew/", "", "", false, "http://barnew/x/y/z/boofar/", "/x/y/z/boofar/"},
-               {"x/y/z/boofar.md", "x/y/z/", "http://barnew/", "boofar", "", false, "http://barnew/x/y/z/boofar/", "/x/y/z/boofar/"},
-               {"x/y/z/boofar.md", "x/y/z", "", "", "", true, "/x/y/z/boofar.html", "/x/y/z/boofar.html"},
-               {"x/y/z/boofar.md", "x/y/z/", "", "", "", true, "/x/y/z/boofar.html", "/x/y/z/boofar.html"},
-               {"x/y/z/boofar.md", "x/y/z/", "", "boofar", "", true, "/x/y/z/boofar.html", "/x/y/z/boofar.html"},
-               {"x/y/z/boofar.md", "x/y/z", "http://barnew/", "", "", true, "http://barnew/x/y/z/boofar.html", "/x/y/z/boofar.html"},
-               {"x/y/z/boofar.md", "x/y/z/", "http://barnew/", "boofar", "", true, "http://barnew/x/y/z/boofar.html", "/x/y/z/boofar.html"},
+               {"x/y/z/boofar.md", "x/y/z", "", "", "", false, false, "/x/y/z/boofar/", "/x/y/z/boofar/"},
+               {"x/y/z/boofar.md", "x/y/z/", "", "", "", false, false, "/x/y/z/boofar/", "/x/y/z/boofar/"},
+               {"x/y/z/boofar.md", "x/y/z/", "", "boofar", "", false, false, "/x/y/z/boofar/", "/x/y/z/boofar/"},
+               {"x/y/z/boofar.md", "x/y/z", "http://barnew/", "", "", false, false, "http://barnew/x/y/z/boofar/", "/x/y/z/boofar/"},
+               {"x/y/z/boofar.md", "x/y/z/", "http://barnew/", "boofar", "", false, false, "http://barnew/x/y/z/boofar/", "/x/y/z/boofar/"},
+               {"x/y/z/boofar.md", "x/y/z", "", "", "", true, false, "/x/y/z/boofar.html", "/x/y/z/boofar.html"},
+               {"x/y/z/boofar.md", "x/y/z/", "", "", "", true, false, "/x/y/z/boofar.html", "/x/y/z/boofar.html"},
+               {"x/y/z/boofar.md", "x/y/z/", "", "boofar", "", true, false, "/x/y/z/boofar.html", "/x/y/z/boofar.html"},
+               {"x/y/z/boofar.md", "x/y/z", "http://barnew/", "", "", true, false, "http://barnew/x/y/z/boofar.html", "/x/y/z/boofar.html"},
+               {"x/y/z/boofar.md", "x/y/z/", "http://barnew/", "boofar", "", true, false, "http://barnew/x/y/z/boofar.html", "/x/y/z/boofar.html"},
+               {"x/y/z/boofar.md", "x/y/z/", "http://barnew/boo/", "boofar", "", true, false, "http://barnew/boo/x/y/z/boofar.html", "/boo/x/y/z/boofar.html"},
+               {"x/y/z/boofar.md", "x/y/z/", "http://barnew/boo/", "boofar", "", true, true, "http://barnew/boo/x/y/z/boofar.html", "/x/y/z/boofar.html"},
+               {"x/y/z/boofar.md", "x/y/z/", "http://barnew/boo", "boofar", "", true, true, "http://barnew/boo/x/y/z/boofar.html", "/x/y/z/boofar.html"},
 
                // test url overrides
-               {"x/y/z/boofar.md", "x/y/z", "", "", "/z/y/q/", false, "/z/y/q/", "/z/y/q/"},
+               {"x/y/z/boofar.md", "x/y/z", "", "", "/z/y/q/", false, false, "/z/y/q/", "/z/y/q/"},
        }
 
        viper.Set("DefaultExtension", "html")
 
        for i, test := range tests {
-               viper.Set("uglyurls", test.uglyurls)
+               viper.Set("uglyurls", test.uglyUrls)
+               viper.Set("canonifyurls", test.canonifyUrls)
                p := &Page{
                        Node: Node{
                                UrlPath: UrlPath{
@@ -75,7 +80,7 @@ func TestPermalink(t *testing.T) {
 
                expected = test.expectedRel
                if u != expected {
-                       t.Errorf("Test %d: Expected abs url: %s, got: %s", i, expected, u)
+                       t.Errorf("Test %d: Expected rel url: %s, got: %s", i, expected, u)
                }
        }
 }
index 413e9f4e2b8c133e8a2547e6ecd95dfd9b00a8d8..95a978f93640e04660ae1c624790a648b7a90dc2 100644 (file)
@@ -106,6 +106,7 @@ type SiteInfo struct {
        Permalinks      PermalinkOverrides
        Params          map[string]interface{}
        BuildDrafts     bool
+       canonifyUrls    bool
 }
 
 // SiteSocial is a place to put social details on a site level. These are the
@@ -362,6 +363,7 @@ func (s *Site) initializeSiteInfo() {
                Copyright:       viper.GetString("copyright"),
                DisqusShortname: viper.GetString("DisqusShortname"),
                BuildDrafts:     viper.GetBool("BuildDrafts"),
+               canonifyUrls:    viper.GetBool("CanonifyUrls"),
                Pages:           &s.Pages,
                Recent:          &s.Pages,
                Menus:           &s.Menus,
@@ -608,10 +610,16 @@ func (s *Site) getMenusFromConfig() Menus {
                                        }
 
                                        menuEntry.MarshallMap(ime)
+
                                        if strings.HasPrefix(menuEntry.Url, "/") {
-                                               // make it absolute so it matches the nodes
-                                               menuEntry.Url = s.permalinkStr(menuEntry.Url)
+                                               // make it match the nodes
+                                               menuEntryUrl := menuEntry.Url
+                                               if !s.Info.canonifyUrls {
+                                                       menuEntryUrl = helpers.AddContextRoot(string(s.Info.BaseUrl), menuEntryUrl)
+                                               }
+                                               menuEntry.Url = s.prepUrl(menuEntryUrl)
                                        }
+
                                        if ret[name] == nil {
                                                ret[name] = &Menu{}
                                        }