resource/scss: Add IncludePaths config option
authorBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Fri, 20 Jul 2018 13:02:35 +0000 (15:02 +0200)
committerBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Fri, 20 Jul 2018 15:50:44 +0000 (17:50 +0200)
Takes paths relative to the current working dir.

Fixes #4921

hugolib/filesystems/basefs.go
hugolib/page_bundler_capture_test.go
hugolib/page_bundler_test.go
hugolib/resource_chain_test.go
hugolib/testhelpers_test.go
resource/tocss/scss/client.go
resource/tocss/scss/tocss.go

index 731b5a195953c6a46519ff2ac8aa9cd98ff2976d..7398957a3b9de641106fe195078853dd094e5423 100644 (file)
@@ -244,8 +244,7 @@ func (d *SourceFilesystem) RealDirs(from string) []string {
        var dirnames []string
        for _, dir := range d.Dirnames {
                dirname := filepath.Join(dir, from)
-
-               if _, err := hugofs.Os.Stat(dirname); err == nil {
+               if _, err := d.SourceFs.Stat(dirname); err == nil {
                        dirnames = append(dirnames, dirname)
                }
        }
index 96d113bf746e0586c4bb2f3fe9abb5bdb8a3338d..ace96b633ad5a67e3b59bb7b35d594172a0afb10 100644 (file)
@@ -90,8 +90,9 @@ func TestPageBundlerCaptureSymlinks(t *testing.T) {
        }
 
        assert := require.New(t)
-       ps, workDir := newTestBundleSymbolicSources(t)
+       ps, clean, workDir := newTestBundleSymbolicSources(t)
        sourceSpec := source.NewSourceSpec(ps, ps.BaseFs.Content.Fs)
+       defer clean()
 
        fileStore := &storeFilenames{}
        logger := loggers.NewErrorLogger()
index 811dbf56fe807374c8ef3c623eda18d6d0329ffd..236672b650757bdc2f59bb3d4fc8962b9fc22421 100644 (file)
 package hugolib
 
 import (
-       "io/ioutil"
-
        "github.com/gohugoio/hugo/common/loggers"
 
        "os"
        "runtime"
-       "strings"
        "testing"
 
        "github.com/gohugoio/hugo/helpers"
@@ -325,7 +322,9 @@ func TestPageBundlerSiteWitSymbolicLinksInContent(t *testing.T) {
        }
 
        assert := require.New(t)
-       ps, workDir := newTestBundleSymbolicSources(t)
+       ps, clean, workDir := newTestBundleSymbolicSources(t)
+       defer clean()
+
        cfg := ps.Cfg
        fs := ps.Fs
 
@@ -667,7 +666,7 @@ TheContent.
        return fs, cfg
 }
 
-func newTestBundleSymbolicSources(t *testing.T) (*helpers.PathSpec, string) {
+func newTestBundleSymbolicSources(t *testing.T) (*helpers.PathSpec, func(), string) {
        assert := require.New(t)
        // We need to use the OS fs for this.
        cfg := viper.New()
@@ -675,13 +674,8 @@ func newTestBundleSymbolicSources(t *testing.T) (*helpers.PathSpec, string) {
        fs.Destination = &afero.MemMapFs{}
        loadDefaultSettingsFor(cfg)
 
-       workDir, err := ioutil.TempDir("", "hugosym")
-
-       if runtime.GOOS == "darwin" && !strings.HasPrefix(workDir, "/private") {
-               // To get the entry folder in line with the rest. This its a little bit
-               // mysterious, but so be it.
-               workDir = "/private" + workDir
-       }
+       workDir, clean, err := createTempDir("hugosym")
+       assert.NoError(err)
 
        contentDir := "base"
        cfg.Set("workingDir", workDir)
@@ -753,5 +747,5 @@ TheContent.
 
        ps, _ := helpers.NewPathSpec(fs, cfg)
 
-       return ps, workDir
+       return ps, clean, workDir
 }
index d504b7d757806daddb71b847fe224cd8510d2cc7..61ae7e611263d8710b505f900e60c9dbd7904590 100644 (file)
 package hugolib
 
 import (
+       "os"
        "path/filepath"
        "testing"
 
+       "github.com/spf13/viper"
+
+       "github.com/stretchr/testify/require"
+
+       "github.com/gohugoio/hugo/hugofs"
+
        "github.com/gohugoio/hugo/common/loggers"
        "github.com/gohugoio/hugo/resource/tocss/scss"
 )
 
+func TestSCSSWithIncludePaths(t *testing.T) {
+       if !scss.Supports() {
+               t.Skip("Skip SCSS")
+       }
+       assert := require.New(t)
+       workDir, clean, err := createTempDir("hugo-scss-include")
+       assert.NoError(err)
+       defer clean()
+
+       v := viper.New()
+       v.Set("workingDir", workDir)
+       b := newTestSitesBuilder(t).WithLogger(loggers.NewWarningLogger())
+       b.WithViper(v)
+       b.WithWorkingDir(workDir)
+       // Need to use OS fs for this.
+       b.Fs = hugofs.NewDefault(v)
+
+       fooDir := filepath.Join(workDir, "node_modules", "foo")
+       scssDir := filepath.Join(workDir, "assets", "scss")
+       assert.NoError(os.MkdirAll(fooDir, 0777))
+       assert.NoError(os.MkdirAll(filepath.Join(workDir, "content", "sect"), 0777))
+       assert.NoError(os.MkdirAll(filepath.Join(workDir, "data"), 0777))
+       assert.NoError(os.MkdirAll(filepath.Join(workDir, "i18n"), 0777))
+       assert.NoError(os.MkdirAll(filepath.Join(workDir, "layouts", "shortcodes"), 0777))
+       assert.NoError(os.MkdirAll(filepath.Join(workDir, "layouts", "_default"), 0777))
+       assert.NoError(os.MkdirAll(filepath.Join(scssDir), 0777))
+
+       b.WithSourceFile(filepath.Join(fooDir, "_moo.scss"), `
+$moolor: #fff;
+
+moo {
+  color: $moolor;
+}
+`)
+
+       b.WithSourceFile(filepath.Join(scssDir, "main.scss"), `
+@import "moo";
+
+`)
+
+       b.WithTemplatesAdded("index.html", `
+{{ $cssOpts := (dict "includePaths" (slice "node_modules/foo" ) ) }}
+{{ $r := resources.Get "scss/main.scss" |  toCSS $cssOpts  | minify  }}
+T1: {{ $r.Content }}
+`)
+       b.Build(BuildCfg{})
+
+       b.AssertFileContent(filepath.Join(workDir, "public/index.html"), `T1: moo{color:#fff}`)
+
+}
+
 func TestResourceChain(t *testing.T) {
        t.Parallel()
 
index 9a72bcbb89029d185d6a20df804c71ff9df27bbe..cee1f8c4c97e4bc3653216a4a85338298724adc2 100644 (file)
@@ -1,7 +1,9 @@
 package hugolib
 
 import (
+       "io/ioutil"
        "path/filepath"
+       "runtime"
        "testing"
 
        "bytes"
@@ -82,6 +84,20 @@ func newTestSitesBuilder(t testing.TB) *sitesBuilder {
        return &sitesBuilder{T: t, Fs: fs, configFormat: "toml", dumper: litterOptions}
 }
 
+func createTempDir(prefix string) (string, func(), error) {
+       workDir, err := ioutil.TempDir("", prefix)
+       if err != nil {
+               return "", nil, err
+       }
+
+       if runtime.GOOS == "darwin" && !strings.HasPrefix(workDir, "/private") {
+               // To get the entry folder in line with the rest. This its a little bit
+               // mysterious, but so be it.
+               workDir = "/private" + workDir
+       }
+       return workDir, func() { os.RemoveAll(workDir) }, nil
+}
+
 func (s *sitesBuilder) Running() *sitesBuilder {
        s.running = true
        return s
index 610ea3845d701795a4b40d03ff7977a7224650ad..126727e069aac4229741daf8100638d66184d856 100644 (file)
@@ -22,12 +22,13 @@ import (
 )
 
 type Client struct {
-       rs  *resource.Spec
-       sfs *filesystems.SourceFilesystem
+       rs     *resource.Spec
+       sfs    *filesystems.SourceFilesystem
+       workFs *filesystems.SourceFilesystem
 }
 
 func New(fs *filesystems.SourceFilesystem, rs *resource.Spec) (*Client, error) {
-       return &Client{sfs: fs, rs: rs}, nil
+       return &Client{sfs: fs, workFs: rs.BaseFs.Work, rs: rs}, nil
 }
 
 type Options struct {
@@ -38,6 +39,13 @@ type Options struct {
        // a Resource with that as a base for RelPermalink etc.
        TargetPath string
 
+       // Hugo automatically adds the entry directories (where the main.scss lives)
+       // for project and themes to the list of include paths sent to LibSASS.
+       // Any paths set in this setting will be appended. Note that these will be
+       // treated as relative to the working dir, i.e. no include paths outside the
+       // project/themes.
+       IncludePaths []string
+
        // Default is nested.
        // One of nested, expanded, compact, compressed.
        OutputStyle string
index ec4685d87ca603f545e3970cce44bba359712beb..715d5fd9f5795c62a9090cbe3c818e6d8d690d30 100644 (file)
@@ -49,10 +49,13 @@ func (t *toCSSTransformation) Transform(ctx *resource.ResourceTransformationCtx)
 
        options := t.options
 
-       // We may allow the end user to add IncludePaths later, if we find a use
-       // case for that.
        options.to.IncludePaths = t.c.sfs.RealDirs(path.Dir(ctx.SourcePath))
 
+       // Append any workDir relative include paths
+       for _, ip := range options.from.IncludePaths {
+               options.to.IncludePaths = append(options.to.IncludePaths, t.c.workFs.RealDirs(filepath.Clean(ip))...)
+       }
+
        if ctx.InMediaType.SubType == media.SASSType.SubType {
                options.to.SassSyntax = true
        }