Add minify config
authorSatowTakeshi <doublequotation@gmail.com>
Sat, 29 Feb 2020 09:44:05 +0000 (18:44 +0900)
committerBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Fri, 20 Mar 2020 19:35:57 +0000 (20:35 +0100)
Fixes #6750
Updates #6892

12 files changed:
docs/data/docs.json
docshelper/docs.go
hugolib/hugo_sites.go
markup/markup_config/config.go
minifiers/config.go [new file with mode: 0644]
minifiers/config_test.go [new file with mode: 0644]
minifiers/minifiers.go
minifiers/minifiers_test.go
publisher/publisher.go
resources/resource_transformers/minifier/minify.go
resources/resource_transformers/minifier/minify_test.go
tpl/resources/resources.go

index 942f04fb0509842f2a64d6b3ab975e61637b42d2..2af97631878a1720e5c7f4fac170df90bb2b81dd 100644 (file)
         "footnoteAnchorPrefix": "",
         "footnoteReturnLinkContents": ""
       }
+    },
+    "minifiers": {
+      "tdewolff": {
+        "enableHtml": true,
+        "enableCss": true,
+        "enableJs": true,
+        "enableJson": true,
+        "enableSvg": true,
+        "enableXml": true,
+        "html": {
+          "keepConditionalComments": true,
+          "keepDefaultAttrVals": true,
+          "keepDocumentTags": true,
+          "keepEndTags": true,
+          "keepWhitespace": false
+        },
+        "css": {
+          "decimals": -1,
+          "keepCSS2": true
+        },
+        "js": {},
+        "json": {},
+        "svg": {
+          "decimals": -1
+        },
+        "xml": {
+          "keepWhitespace": false
+        }
+      }
     }
   },
   "media": {
           "Aliases": null,
           "Examples": null
         },
+        "IsProduction": {
+          "Description": "",
+          "Args": null,
+          "Aliases": null,
+          "Examples": null
+        },
         "Version": {
           "Description": "",
           "Args": null,
             ]
           ]
         },
+        "Sqrt": {
+          "Description": "Sqrt returns the square root of a number.\nNOTE: will return for NaN for negative values of a",
+          "Args": [
+            "a"
+          ],
+          "Aliases": null,
+          "Examples": [
+            [
+              "{{math.Sqrt 81}}",
+              "9"
+            ]
+          ]
+        },
         "Sub": {
           "Description": "Sub subtracts two numbers.",
           "Args": [
index 94cb70dec97a29fc8c16893328f3cae5af334ecc..17e0ccd91ba2900da3518e9dc4a9aa74b62678ec 100644 (file)
@@ -24,7 +24,11 @@ var DocProviders = make(map[string]DocProvider)
 
 // AddDocProvider adds or updates the DocProvider for a given name.
 func AddDocProvider(name string, provider DocProvider) {
-       DocProviders[name] = provider
+       if prev, ok := DocProviders[name]; !ok {
+               DocProviders[name] = provider
+       } else {
+               DocProviders[name] = merge(prev, provider)
+       }
 }
 
 // DocProvider is used to save arbitrary JSON data
@@ -35,3 +39,13 @@ type DocProvider func() map[string]interface{}
 func (d DocProvider) MarshalJSON() ([]byte, error) {
        return json.MarshalIndent(d(), "", "  ")
 }
+
+func merge(a, b DocProvider) DocProvider {
+       next := a()
+       for k, v := range b() {
+               next[k] = v
+       }
+       return func() map[string]interface{} {
+               return next
+       }
+}
index 04a231fdf29264793ca6d6bd59fbbc200fc12082..dca9e49680d83c479ed3242ef231ce07434958c7 100644 (file)
@@ -408,7 +408,12 @@ func applyDeps(cfg deps.DepsCfg, sites ...*Site) error {
                        s.Deps = d
 
                        // Set up the main publishing chain.
-                       s.publisher = publisher.NewDestinationPublisher(d.PathSpec.BaseFs.PublishFs, s.outputFormatsConfig, s.mediaTypesConfig, cfg.Cfg.GetBool("minify"))
+                       pub, err := publisher.NewDestinationPublisher(d.PathSpec.BaseFs.PublishFs, s.outputFormatsConfig, s.mediaTypesConfig, cfg.Cfg)
+
+                       if err != nil {
+                               return err
+                       }
+                       s.publisher = pub
 
                        if err := s.initializeSiteInfo(); err != nil {
                                return err
index 529553cb5a2f728242a6c763ff09ab42f70be3bd..34c8807acf34664e0a6f82ed15062f236c4a49e5 100644 (file)
@@ -100,6 +100,5 @@ func init() {
                return docs
 
        }
-       // TODO(bep) merge maps
        docshelper.AddDocProvider("config", docsProvider)
 }
diff --git a/minifiers/config.go b/minifiers/config.go
new file mode 100644 (file)
index 0000000..20d122e
--- /dev/null
@@ -0,0 +1,111 @@
+// Copyright 2019 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 minifiers
+
+import (
+       "github.com/gohugoio/hugo/config"
+       "github.com/gohugoio/hugo/docshelper"
+       "github.com/gohugoio/hugo/parser"
+
+       "github.com/mitchellh/mapstructure"
+       "github.com/tdewolff/minify/v2/css"
+       "github.com/tdewolff/minify/v2/html"
+       "github.com/tdewolff/minify/v2/js"
+       "github.com/tdewolff/minify/v2/json"
+       "github.com/tdewolff/minify/v2/svg"
+       "github.com/tdewolff/minify/v2/xml"
+)
+
+var defaultTdewolffConfig = tdewolffConfig{
+       HTML: html.Minifier{
+               KeepDocumentTags:        true,
+               KeepConditionalComments: true,
+               KeepEndTags:             true,
+               KeepDefaultAttrVals:     true,
+               KeepWhitespace:          false,
+               // KeepQuotes:              false, >= v2.6.2
+       },
+       CSS: css.Minifier{
+               Decimals: -1, // will be deprecated
+               // Precision: 0,  // use Precision with >= v2.7.0
+               KeepCSS2: true,
+       },
+       JS:   js.Minifier{},
+       JSON: json.Minifier{},
+       SVG: svg.Minifier{
+               Decimals: -1, // will be deprecated
+               // Precision: 0,  // use Precision with >= v2.7.0
+       },
+       XML: xml.Minifier{
+               KeepWhitespace: false,
+       },
+}
+
+type tdewolffConfig struct {
+       HTML html.Minifier
+       CSS  css.Minifier
+       JS   js.Minifier
+       JSON json.Minifier
+       SVG  svg.Minifier
+       XML  xml.Minifier
+}
+
+type minifiersConfig struct {
+       EnableHTML bool
+       EnableCSS  bool
+       EnableJS   bool
+       EnableJSON bool
+       EnableSVG  bool
+       EnableXML  bool
+
+       Tdewolff tdewolffConfig
+}
+
+var defaultConfig = minifiersConfig{
+       EnableHTML: true,
+       EnableCSS:  true,
+       EnableJS:   true,
+       EnableJSON: true,
+       EnableSVG:  true,
+       EnableXML:  true,
+
+       Tdewolff: defaultTdewolffConfig,
+}
+
+func decodeConfig(cfg config.Provider) (conf minifiersConfig, err error) {
+       conf = defaultConfig
+
+       m := cfg.GetStringMap("minifiers")
+       if m == nil {
+               return
+       }
+
+       err = mapstructure.WeakDecode(m, &conf)
+
+       if err != nil {
+               return
+       }
+
+       return
+}
+
+func init() {
+       docsProvider := func() map[string]interface{} {
+               docs := make(map[string]interface{})
+               docs["minifiers"] = parser.LowerCaseCamelJSONMarshaller{Value: defaultConfig}
+               return docs
+
+       }
+       docshelper.AddDocProvider("config", docsProvider)
+}
diff --git a/minifiers/config_test.go b/minifiers/config_test.go
new file mode 100644 (file)
index 0000000..b4f68f8
--- /dev/null
@@ -0,0 +1,52 @@
+// Copyright 2019 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 minifiers
+
+import (
+       "fmt"
+       "testing"
+
+       "github.com/spf13/viper"
+
+       qt "github.com/frankban/quicktest"
+)
+
+func TestConfig(t *testing.T) {
+       c := qt.New(t)
+       v := viper.New()
+
+       v.Set("minifiers", map[string]interface{}{
+               "enablexml": false,
+               "tdewolff": map[string]interface{}{
+                       "html": map[string]interface{}{
+                               "keepwhitespace": false,
+                       },
+               },
+       })
+
+       conf, err := decodeConfig(v)
+       fmt.Println(conf)
+
+       c.Assert(err, qt.IsNil)
+
+       // explicitly set value
+       c.Assert(conf.Tdewolff.HTML.KeepWhitespace, qt.Equals, false)
+       // default value
+       c.Assert(conf.Tdewolff.HTML.KeepEndTags, qt.Equals, true)
+       c.Assert(conf.Tdewolff.CSS.KeepCSS2, qt.Equals, true)
+
+       // `enable` flags
+       c.Assert(conf.EnableHTML, qt.Equals, true)
+       c.Assert(conf.EnableXML, qt.Equals, false)
+}
index 9533ebb6917ef168414022a92600619ae670f8fb..3feb816011e96d6f365a81702ff1b9d1e56eb54f 100644 (file)
@@ -20,17 +20,12 @@ import (
        "io"
        "regexp"
 
+       "github.com/gohugoio/hugo/config"
        "github.com/gohugoio/hugo/output"
        "github.com/gohugoio/hugo/transform"
 
        "github.com/gohugoio/hugo/media"
        "github.com/tdewolff/minify/v2"
-       "github.com/tdewolff/minify/v2/css"
-       "github.com/tdewolff/minify/v2/html"
-       "github.com/tdewolff/minify/v2/js"
-       "github.com/tdewolff/minify/v2/json"
-       "github.com/tdewolff/minify/v2/svg"
-       "github.com/tdewolff/minify/v2/xml"
 )
 
 // Client wraps a minifier.
@@ -62,39 +57,44 @@ func (m Client) Minify(mediatype media.Type, dst io.Writer, src io.Reader) error
 // New creates a new Client with the provided MIME types as the mapping foundation.
 // The HTML minifier is also registered for additional HTML types (AMP etc.) in the
 // provided list of output formats.
-func New(mediaTypes media.Types, outputFormats output.Formats) Client {
-       m := minify.New()
-       htmlMin := &html.Minifier{
-               KeepDocumentTags:        true,
-               KeepConditionalComments: true,
-               KeepEndTags:             true,
-               KeepDefaultAttrVals:     true,
-       }
+func New(mediaTypes media.Types, outputFormats output.Formats, cfg config.Provider) (Client, error) {
+       conf, err := decodeConfig(cfg)
 
-       cssMin := &css.Minifier{
-               Decimals: -1,
-               KeepCSS2: true,
+       m := minify.New()
+       if err != nil {
+               return Client{m: m}, err
        }
 
        // We use the Type definition of the media types defined in the site if found.
-       addMinifier(m, mediaTypes, "css", cssMin)
-       addMinifierFunc(m, mediaTypes, "js", js.Minify)
-       m.AddFuncRegexp(regexp.MustCompile("^(application|text)/(x-)?(java|ecma)script$"), js.Minify)
-       m.AddFuncRegexp(regexp.MustCompile(`^(application|text)/(x-|ld\+)?json$`), json.Minify)
-       addMinifierFunc(m, mediaTypes, "json", json.Minify)
-       addMinifierFunc(m, mediaTypes, "svg", svg.Minify)
-       addMinifierFunc(m, mediaTypes, "xml", xml.Minify)
+       if conf.EnableCSS {
+               addMinifier(m, mediaTypes, "css", &conf.Tdewolff.CSS)
+       }
+       if conf.EnableJS {
+               addMinifier(m, mediaTypes, "js", &conf.Tdewolff.JS)
+               m.AddRegexp(regexp.MustCompile("^(application|text)/(x-)?(java|ecma)script$"), &conf.Tdewolff.JS)
+       }
+       if conf.EnableJSON {
+               addMinifier(m, mediaTypes, "json", &conf.Tdewolff.JSON)
+               m.AddRegexp(regexp.MustCompile(`^(application|text)/(x-|ld\+)?json$`), &conf.Tdewolff.JSON)
+       }
+       if conf.EnableSVG {
+               addMinifier(m, mediaTypes, "svg", &conf.Tdewolff.SVG)
+       }
+       if conf.EnableXML {
+               addMinifier(m, mediaTypes, "xml", &conf.Tdewolff.XML)
+       }
 
        // HTML
-       addMinifier(m, mediaTypes, "html", htmlMin)
-       for _, of := range outputFormats {
-               if of.IsHTML {
-                       m.Add(of.MediaType.Type(), htmlMin)
+       if conf.EnableHTML {
+               addMinifier(m, mediaTypes, "html", &conf.Tdewolff.HTML)
+               for _, of := range outputFormats {
+                       if of.IsHTML {
+                               m.Add(of.MediaType.Type(), &conf.Tdewolff.HTML)
+                       }
                }
        }
 
-       return Client{m: m}
-
+       return Client{m: m}, nil
 }
 
 func addMinifier(m *minify.M, mt media.Types, suffix string, min minify.Minifier) {
index 87e706320f85834798c9a05e9343f8099914b466..64588f83c361c2bad6a041e22f7f593e47f802e5 100644 (file)
@@ -23,11 +23,13 @@ import (
 
        qt "github.com/frankban/quicktest"
        "github.com/gohugoio/hugo/output"
+       "github.com/spf13/viper"
 )
 
 func TestNew(t *testing.T) {
        c := qt.New(t)
-       m := New(media.DefaultTypes, output.DefaultFormats)
+       v := viper.New()
+       m, _ := New(media.DefaultTypes, output.DefaultFormats, v)
 
        var rawJS string
        var minJS string
@@ -73,9 +75,44 @@ func TestNew(t *testing.T) {
 
 }
 
+func TestConfiguredMinify(t *testing.T) {
+       c := qt.New(t)
+       v := viper.New()
+       v.Set("minifiers", map[string]interface{}{
+               "enablexml": false,
+               "tdewolff": map[string]interface{}{
+                       "html": map[string]interface{}{
+                               "keepwhitespace": true,
+                       },
+               },
+       })
+       m, _ := New(media.DefaultTypes, output.DefaultFormats, v)
+
+       for _, test := range []struct {
+               tp                media.Type
+               rawString         string
+               expectedMinString string
+               errorExpected     bool
+       }{
+               {media.HTMLType, "<hello> Hugo! </hello>", "<hello> Hugo! </hello>", false}, // configured minifier
+               {media.CSSType, " body { color: blue; }  ", "body{color:blue}", false},      // default minifier
+               {media.XMLType, " <hello>  Hugo!   </hello>  ", "", true},                   // disable Xml minificatin
+       } {
+               var b bytes.Buffer
+               if !test.errorExpected {
+                       c.Assert(m.Minify(test.tp, &b, strings.NewReader(test.rawString)), qt.IsNil)
+                       c.Assert(b.String(), qt.Equals, test.expectedMinString)
+               } else {
+                       err := m.Minify(test.tp, &b, strings.NewReader(test.rawString))
+                       c.Assert(err, qt.ErrorMatches, "minifier does not exist for mimetype")
+               }
+       }
+}
+
 func TestJSONRoundTrip(t *testing.T) {
        c := qt.New(t)
-       m := New(media.DefaultTypes, output.DefaultFormats)
+       v := viper.New()
+       m, _ := New(media.DefaultTypes, output.DefaultFormats, v)
 
        for _, test := range []string{`{
     "glossary": {
@@ -113,7 +150,8 @@ func TestJSONRoundTrip(t *testing.T) {
 
 func TestBugs(t *testing.T) {
        c := qt.New(t)
-       m := New(media.DefaultTypes, output.DefaultFormats)
+       v := viper.New()
+       m, _ := New(media.DefaultTypes, output.DefaultFormats, v)
 
        for _, test := range []struct {
                tp                media.Type
index 119be356ba1b0b1953153344497d2feb2e8c94f1..e1179572b2195856fad8427c1c52653f9317d3ac 100644 (file)
@@ -18,6 +18,7 @@ import (
        "io"
        "sync/atomic"
 
+       "github.com/gohugoio/hugo/config"
        "github.com/gohugoio/hugo/media"
 
        "github.com/gohugoio/hugo/minifiers"
@@ -73,13 +74,17 @@ type DestinationPublisher struct {
 }
 
 // NewDestinationPublisher creates a new DestinationPublisher.
-func NewDestinationPublisher(fs afero.Fs, outputFormats output.Formats, mediaTypes media.Types, minify bool) DestinationPublisher {
-       pub := DestinationPublisher{fs: fs}
+func NewDestinationPublisher(fs afero.Fs, outputFormats output.Formats, mediaTypes media.Types, cfg config.Provider) (pub DestinationPublisher, err error) {
+       pub = DestinationPublisher{fs: fs}
+       minify := cfg.GetBool("minify")
        if minify {
-               pub.min = minifiers.New(mediaTypes, outputFormats)
+               pub.min, err = minifiers.New(mediaTypes, outputFormats, cfg)
+               if err != nil {
+                       return
+               }
                pub.minify = true
        }
-       return pub
+       return
 }
 
 // Publish applies any relevant transformations and writes the file
index 38e3fc93aa7e180abd552c4767fbfe9a0a9c9b22..060485e80c53fd3fcf5ca63751404d6de059507a 100644 (file)
@@ -29,8 +29,12 @@ type Client struct {
 
 // New creates a new Client given a specification. Note that it is the media types
 // configured for the site that is used to match files to the correct minifier.
-func New(rs *resources.Spec) *Client {
-       return &Client{rs: rs, m: minifiers.New(rs.MediaTypes, rs.OutputFormats)}
+func New(rs *resources.Spec) (*Client, error) {
+       m, err := minifiers.New(rs.MediaTypes, rs.OutputFormats, rs.Cfg)
+       if err != nil {
+               return nil, err
+       }
+       return &Client{rs: rs, m: m}, nil
 }
 
 type minifyTransformation struct {
@@ -43,9 +47,7 @@ func (t *minifyTransformation) Key() internal.ResourceTransformationKey {
 }
 
 func (t *minifyTransformation) Transform(ctx *resources.ResourceTransformationCtx) error {
-       if err := t.m.Minify(ctx.InMediaType, ctx.To, ctx.From); err != nil {
-               return err
-       }
+       _ = t.m.Minify(ctx.InMediaType, ctx.To, ctx.From)
        ctx.AddOutPathIdentifier(".min")
        return nil
 }
index 3f8853520de57a05745431e7d669bd3579212096..a1b22f1098f5ef74b9d54c3bee51287efce7b9e7 100644 (file)
@@ -27,7 +27,7 @@ func TestTransform(t *testing.T) {
 
        spec, err := htesting.NewTestResourceSpec()
        c.Assert(err, qt.IsNil)
-       client := New(spec)
+       client, _ := New(spec)
 
        r, err := htesting.NewResourceTransformerForSpec(spec, "hugo.html", "<h1>   Hugo Rocks!   </h1>")
        c.Assert(err, qt.IsNil)
@@ -41,3 +41,23 @@ func TestTransform(t *testing.T) {
        c.Assert(content, qt.Equals, "<h1>Hugo Rocks!</h1>")
 
 }
+
+func TestNoMinifier(t *testing.T) {
+       c := qt.New(t)
+
+       spec, _ := htesting.NewTestResourceSpec()
+       spec.Cfg.Set("minifiers.enableXML", false)
+       client, _ := New(spec)
+
+       original := "<title>   Hugo Rocks!   </title>"
+       r, err := htesting.NewResourceTransformerForSpec(spec, "hugo.xml", original)
+       c.Assert(err, qt.IsNil)
+
+       transformed, err := client.Minify(r)
+       c.Assert(err, qt.IsNil)
+
+       content, err := transformed.(resource.ContentProvider).Content()
+       // error should be ignored because general users cannot control codes under `theme`s
+       c.Assert(err, qt.IsNil)
+       c.Assert(content, qt.Equals, original)
+}
index be10510f2b45bc19f3c8f112b6c1909ef4d29ec1..a1055632c084344103c4284cebd2873bd7fa4a47 100644 (file)
@@ -46,13 +46,18 @@ func New(deps *deps.Deps) (*Namespace, error) {
                return nil, err
        }
 
+       minifyClient, err := minifier.New(deps.ResourceSpec)
+       if err != nil {
+               return nil, err
+       }
+
        return &Namespace{
                deps:            deps,
                scssClient:      scssClient,
                createClient:    create.New(deps.ResourceSpec),
                bundlerClient:   bundler.New(deps.ResourceSpec),
                integrityClient: integrity.New(deps.ResourceSpec),
-               minifyClient:    minifier.New(deps.ResourceSpec),
+               minifyClient:    minifyClient,
                postcssClient:   postcss.New(deps.ResourceSpec),
                templatesClient: templates.New(deps.ResourceSpec, deps),
        }, nil