| 
									
										
										
										
											2017-03-09 19:19:29 +01:00
										 |  |  | // Copyright 2017 The Hugo Authors. All rights reserved.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // Licensed under the Apache License, Version 2.0 (the "License");
 | 
					
						
							|  |  |  | // you may not use this file except in compliance with the License.
 | 
					
						
							|  |  |  | // You may obtain a copy of the License at
 | 
					
						
							|  |  |  | // http://www.apache.org/licenses/LICENSE-2.0
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // Unless required by applicable law or agreed to in writing, software
 | 
					
						
							|  |  |  | // distributed under the License is distributed on an "AS IS" BASIS,
 | 
					
						
							|  |  |  | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
					
						
							|  |  |  | // See the License for the specific language governing permissions and
 | 
					
						
							|  |  |  | // limitations under the License.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | package hugolib
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import (
 | 
					
						
							|  |  |  | 	"path/filepath"
 | 
					
						
							|  |  |  | 	"strings"
 | 
					
						
							|  |  |  | 	"testing"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-20 08:45:52 +02:00
										 |  |  | 	"github.com/gohugoio/hugo/media"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 19:19:29 +01:00
										 |  |  | 	"fmt"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-13 18:42:45 +02:00
										 |  |  | 	"github.com/gohugoio/hugo/output"
 | 
					
						
							| 
									
										
										
										
											2017-03-09 19:19:29 +01:00
										 |  |  | )
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func TestPageTargetPath(t *testing.T) {
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Add Hugo Piper with SCSS support and much more
Before this commit, you would have to use page bundles to do image processing etc. in Hugo.
This commit adds
* A new `/assets` top-level project or theme dir (configurable via `assetDir`)
* A new template func, `resources.Get` which can be used to "get a resource" that can be further processed.
This means that you can now do this in your templates (or shortcodes):
```bash
{{ $sunset := (resources.Get "images/sunset.jpg").Fill "300x200" }}
```
This also adds a new `extended` build tag that enables powerful SCSS/SASS support with source maps. To compile this from source, you will also need a C compiler installed:
```
HUGO_BUILD_TAGS=extended mage install
```
Note that you can use output of the SCSS processing later in a non-SCSSS-enabled Hugo.
The `SCSS` processor is a _Resource transformation step_ and it can be chained with the many others in a pipeline:
```bash
{{ $css := resources.Get "styles.scss" | resources.ToCSS | resources.PostCSS | resources.Minify | resources.Fingerprint }}
<link rel="stylesheet" href="{{ $styles.RelPermalink }}" integrity="{{ $styles.Data.Digest }}" media="screen">
```
The transformation funcs above have aliases, so it can be shortened to:
```bash
{{ $css := resources.Get "styles.scss" | toCSS | postCSS | minify | fingerprint }}
<link rel="stylesheet" href="{{ $styles.RelPermalink }}" integrity="{{ $styles.Data.Digest }}" media="screen">
```
A quick tip would be to avoid the fingerprinting part, and possibly also the not-superfast `postCSS` when you're doing development, as it allows Hugo to be smarter about the rebuilding.
Documentation will follow, but have a look at the demo repo in https://github.com/bep/hugo-sass-test
New functions to create `Resource` objects:
* `resources.Get` (see above)
* `resources.FromString`: Create a Resource from a string.
New `Resource` transformation funcs:
* `resources.ToCSS`: Compile `SCSS` or `SASS` into `CSS`.
* `resources.PostCSS`: Process your CSS with PostCSS. Config file support (project or theme or passed as an option).
* `resources.Minify`: Currently supports `css`, `js`, `json`, `html`, `svg`, `xml`.
* `resources.Fingerprint`: Creates a fingerprinted version of the given Resource with Subresource Integrity..
* `resources.Concat`: Concatenates a list of Resource objects. Think of this as a poor man's bundler.
* `resources.ExecuteAsTemplate`: Parses and executes the given Resource and data context (e.g. .Site) as a Go template.
Fixes #4381
Fixes #4903
Fixes #4858
											
										 
											2018-02-20 10:02:14 +01:00
										 |  |  | 	pathSpec := newTestDefaultPathSpec(t)
 | 
					
						
							| 
									
										
										
										
											2017-03-09 19:19:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-20 08:45:52 +02:00
										 |  |  | 	noExtNoDelimMediaType := media.TextType
 | 
					
						
							|  |  |  | 	noExtNoDelimMediaType.Suffix = ""
 | 
					
						
							|  |  |  | 	noExtNoDelimMediaType.Delimiter = ""
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Netlify style _redirects
 | 
					
						
							|  |  |  | 	noExtDelimFormat := output.Format{
 | 
					
						
							|  |  |  | 		Name:      "NER",
 | 
					
						
							|  |  |  | 		MediaType: noExtNoDelimMediaType,
 | 
					
						
							|  |  |  | 		BaseName:  "_redirects",
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-20 10:34:30 +01:00
										 |  |  | 	for _, multiHost := range []bool{false, true} {
 | 
					
						
							|  |  |  | 		for _, langPrefix := range []string{"", "no"} {
 | 
					
						
							|  |  |  | 			for _, uglyURLs := range []bool{false, true} {
 | 
					
						
							|  |  |  | 				t.Run(fmt.Sprintf("multihost=%t,langPrefix=%q,uglyURLs=%t", multiHost, langPrefix, uglyURLs),
 | 
					
						
							|  |  |  | 					func(t *testing.T) {
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						tests := []struct {
 | 
					
						
							|  |  |  | 							name     string
 | 
					
						
							|  |  |  | 							d        targetPathDescriptor
 | 
					
						
							|  |  |  | 							expected string
 | 
					
						
							|  |  |  | 						}{
 | 
					
						
							|  |  |  | 							{"JSON home", targetPathDescriptor{Kind: KindHome, Type: output.JSONFormat}, "/index.json"},
 | 
					
						
							|  |  |  | 							{"AMP home", targetPathDescriptor{Kind: KindHome, Type: output.AMPFormat}, "/amp/index.html"},
 | 
					
						
							|  |  |  | 							{"HTML home", targetPathDescriptor{Kind: KindHome, BaseName: "_index", Type: output.HTMLFormat}, "/index.html"},
 | 
					
						
							|  |  |  | 							{"Netlify redirects", targetPathDescriptor{Kind: KindHome, BaseName: "_index", Type: noExtDelimFormat}, "/_redirects"},
 | 
					
						
							|  |  |  | 							{"HTML section list", targetPathDescriptor{
 | 
					
						
							|  |  |  | 								Kind:     KindSection,
 | 
					
						
							|  |  |  | 								Sections: []string{"sect1"},
 | 
					
						
							| 
									
										
										
										
											2017-03-09 19:19:29 +01:00
										 |  |  | 								BaseName: "_index",
 | 
					
						
							| 
									
										
										
										
											2017-11-20 10:34:30 +01:00
										 |  |  | 								Type:     output.HTMLFormat}, "/sect1/index.html"},
 | 
					
						
							|  |  |  | 							{"HTML taxonomy list", targetPathDescriptor{
 | 
					
						
							| 
									
										
										
										
											2017-03-09 19:19:29 +01:00
										 |  |  | 								Kind:     KindTaxonomy,
 | 
					
						
							|  |  |  | 								Sections: []string{"tags", "hugo"},
 | 
					
						
							| 
									
										
										
										
											2017-11-20 10:34:30 +01:00
										 |  |  | 								BaseName: "_index",
 | 
					
						
							|  |  |  | 								Type:     output.HTMLFormat}, "/tags/hugo/index.html"},
 | 
					
						
							|  |  |  | 							{"HTML taxonomy term", targetPathDescriptor{
 | 
					
						
							|  |  |  | 								Kind:     KindTaxonomy,
 | 
					
						
							|  |  |  | 								Sections: []string{"tags"},
 | 
					
						
							|  |  |  | 								BaseName: "_index",
 | 
					
						
							|  |  |  | 								Type:     output.HTMLFormat}, "/tags/index.html"},
 | 
					
						
							|  |  |  | 							{
 | 
					
						
							|  |  |  | 								"HTML page", targetPathDescriptor{
 | 
					
						
							|  |  |  | 									Kind:     KindPage,
 | 
					
						
							|  |  |  | 									Dir:      "/a/b",
 | 
					
						
							|  |  |  | 									BaseName: "mypage",
 | 
					
						
							|  |  |  | 									Sections: []string{"a"},
 | 
					
						
							|  |  |  | 									Type:     output.HTMLFormat}, "/a/b/mypage/index.html"},
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 							{
 | 
					
						
							|  |  |  | 								"HTML page with index as base", targetPathDescriptor{
 | 
					
						
							|  |  |  | 									Kind:     KindPage,
 | 
					
						
							|  |  |  | 									Dir:      "/a/b",
 | 
					
						
							|  |  |  | 									BaseName: "index",
 | 
					
						
							|  |  |  | 									Sections: []string{"a"},
 | 
					
						
							|  |  |  | 									Type:     output.HTMLFormat}, "/a/b/index.html"},
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 							{
 | 
					
						
							|  |  |  | 								"HTML page with special chars", targetPathDescriptor{
 | 
					
						
							|  |  |  | 									Kind:     KindPage,
 | 
					
						
							|  |  |  | 									Dir:      "/a/b",
 | 
					
						
							|  |  |  | 									BaseName: "My Page!",
 | 
					
						
							|  |  |  | 									Type:     output.HTMLFormat}, "/a/b/My-Page/index.html"},
 | 
					
						
							|  |  |  | 							{"RSS home", targetPathDescriptor{Kind: kindRSS, Type: output.RSSFormat}, "/index.xml"},
 | 
					
						
							|  |  |  | 							{"RSS section list", targetPathDescriptor{
 | 
					
						
							|  |  |  | 								Kind:     kindRSS,
 | 
					
						
							|  |  |  | 								Sections: []string{"sect1"},
 | 
					
						
							|  |  |  | 								Type:     output.RSSFormat}, "/sect1/index.xml"},
 | 
					
						
							|  |  |  | 							{
 | 
					
						
							|  |  |  | 								"AMP page", targetPathDescriptor{
 | 
					
						
							|  |  |  | 									Kind:     KindPage,
 | 
					
						
							|  |  |  | 									Dir:      "/a/b/c",
 | 
					
						
							|  |  |  | 									BaseName: "myamp",
 | 
					
						
							|  |  |  | 									Type:     output.AMPFormat}, "/amp/a/b/c/myamp/index.html"},
 | 
					
						
							|  |  |  | 							{
 | 
					
						
							|  |  |  | 								"AMP page with URL with suffix", targetPathDescriptor{
 | 
					
						
							|  |  |  | 									Kind:     KindPage,
 | 
					
						
							|  |  |  | 									Dir:      "/sect/",
 | 
					
						
							|  |  |  | 									BaseName: "mypage",
 | 
					
						
							|  |  |  | 									URL:      "/some/other/url.xhtml",
 | 
					
						
							|  |  |  | 									Type:     output.HTMLFormat}, "/some/other/url.xhtml"},
 | 
					
						
							|  |  |  | 							{
 | 
					
						
							|  |  |  | 								"JSON page with URL without suffix", targetPathDescriptor{
 | 
					
						
							|  |  |  | 									Kind:     KindPage,
 | 
					
						
							|  |  |  | 									Dir:      "/sect/",
 | 
					
						
							|  |  |  | 									BaseName: "mypage",
 | 
					
						
							|  |  |  | 									URL:      "/some/other/path/",
 | 
					
						
							|  |  |  | 									Type:     output.JSONFormat}, "/some/other/path/index.json"},
 | 
					
						
							|  |  |  | 							{
 | 
					
						
							|  |  |  | 								"JSON page with URL without suffix and no trailing slash", targetPathDescriptor{
 | 
					
						
							|  |  |  | 									Kind:     KindPage,
 | 
					
						
							|  |  |  | 									Dir:      "/sect/",
 | 
					
						
							|  |  |  | 									BaseName: "mypage",
 | 
					
						
							|  |  |  | 									URL:      "/some/other/path",
 | 
					
						
							|  |  |  | 									Type:     output.JSONFormat}, "/some/other/path/index.json"},
 | 
					
						
							|  |  |  | 							{
 | 
					
						
							|  |  |  | 								"HTML page with expanded permalink", targetPathDescriptor{
 | 
					
						
							|  |  |  | 									Kind:              KindPage,
 | 
					
						
							|  |  |  | 									Dir:               "/a/b",
 | 
					
						
							|  |  |  | 									BaseName:          "mypage",
 | 
					
						
							|  |  |  | 									ExpandedPermalink: "/2017/10/my-title",
 | 
					
						
							|  |  |  | 									Type:              output.HTMLFormat}, "/2017/10/my-title/index.html"},
 | 
					
						
							|  |  |  | 							{
 | 
					
						
							|  |  |  | 								"Paginated HTML home", targetPathDescriptor{
 | 
					
						
							|  |  |  | 									Kind:     KindHome,
 | 
					
						
							|  |  |  | 									BaseName: "_index",
 | 
					
						
							|  |  |  | 									Type:     output.HTMLFormat,
 | 
					
						
							|  |  |  | 									Addends:  "page/3"}, "/page/3/index.html"},
 | 
					
						
							|  |  |  | 							{
 | 
					
						
							|  |  |  | 								"Paginated Taxonomy list", targetPathDescriptor{
 | 
					
						
							|  |  |  | 									Kind:     KindTaxonomy,
 | 
					
						
							|  |  |  | 									BaseName: "_index",
 | 
					
						
							|  |  |  | 									Sections: []string{"tags", "hugo"},
 | 
					
						
							|  |  |  | 									Type:     output.HTMLFormat,
 | 
					
						
							|  |  |  | 									Addends:  "page/3"}, "/tags/hugo/page/3/index.html"},
 | 
					
						
							|  |  |  | 							{
 | 
					
						
							|  |  |  | 								"Regular page with addend", targetPathDescriptor{
 | 
					
						
							|  |  |  | 									Kind:     KindPage,
 | 
					
						
							|  |  |  | 									Dir:      "/a/b",
 | 
					
						
							|  |  |  | 									BaseName: "mypage",
 | 
					
						
							|  |  |  | 									Addends:  "c/d/e",
 | 
					
						
							|  |  |  | 									Type:     output.HTMLFormat}, "/a/b/mypage/c/d/e/index.html"},
 | 
					
						
							| 
									
										
										
										
											2017-03-09 19:19:29 +01:00
										 |  |  | 						}
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-20 10:34:30 +01:00
										 |  |  | 						for i, test := range tests {
 | 
					
						
							|  |  |  | 							test.d.PathSpec = pathSpec
 | 
					
						
							|  |  |  | 							test.d.UglyURLs = uglyURLs
 | 
					
						
							|  |  |  | 							test.d.LangPrefix = langPrefix
 | 
					
						
							|  |  |  | 							test.d.IsMultihost = multiHost
 | 
					
						
							|  |  |  | 							test.d.Dir = filepath.FromSlash(test.d.Dir)
 | 
					
						
							|  |  |  | 							isUgly := uglyURLs && !test.d.Type.NoUgly
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 							expected := test.expected
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 							// TODO(bep) simplify
 | 
					
						
							|  |  |  | 							if test.d.Kind == KindPage && test.d.BaseName == test.d.Type.BaseName {
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 							} else if test.d.Kind == KindHome && test.d.Type.Path != "" {
 | 
					
						
							|  |  |  | 							} else if (!strings.HasPrefix(expected, "/index") || test.d.Addends != "") && test.d.URL == "" && isUgly {
 | 
					
						
							|  |  |  | 								expected = strings.Replace(expected,
 | 
					
						
							|  |  |  | 									"/"+test.d.Type.BaseName+"."+test.d.Type.MediaType.Suffix,
 | 
					
						
							|  |  |  | 									"."+test.d.Type.MediaType.Suffix, -1)
 | 
					
						
							|  |  |  | 							}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 							if test.d.LangPrefix != "" && !(test.d.Kind == KindPage && test.d.URL != "") {
 | 
					
						
							|  |  |  | 								expected = "/" + test.d.LangPrefix + expected
 | 
					
						
							|  |  |  | 							} else if multiHost && test.d.LangPrefix != "" && test.d.URL != "" {
 | 
					
						
							|  |  |  | 								expected = "/" + test.d.LangPrefix + expected
 | 
					
						
							|  |  |  | 							}
 | 
					
						
							| 
									
										
										
										
											2017-03-09 19:19:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-20 10:34:30 +01:00
										 |  |  | 							expected = filepath.FromSlash(expected)
 | 
					
						
							| 
									
										
										
										
											2017-03-09 19:19:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-20 10:34:30 +01:00
										 |  |  | 							pagePath := createTargetPath(test.d)
 | 
					
						
							| 
									
										
										
										
											2017-03-09 19:19:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-20 10:34:30 +01:00
										 |  |  | 							if pagePath != expected {
 | 
					
						
							|  |  |  | 								t.Fatalf("[%d] [%s] targetPath expected %q, got: %q", i, test.name, expected, pagePath)
 | 
					
						
							|  |  |  | 							}
 | 
					
						
							| 
									
										
										
										
											2017-03-09 19:19:29 +01:00
										 |  |  | 						}
 | 
					
						
							| 
									
										
										
										
											2017-11-20 10:34:30 +01:00
										 |  |  | 					})
 | 
					
						
							|  |  |  | 			}
 | 
					
						
							| 
									
										
										
										
											2017-06-06 08:43:33 +02:00
										 |  |  | 		}
 | 
					
						
							| 
									
										
										
										
											2017-03-09 19:19:29 +01:00
										 |  |  | 	}
 | 
					
						
							|  |  |  | }
 |