modules: Make ignoreVendor a glob pattern

Fixes #7642
This commit is contained in:
Bjørn Erik Pedersen 2020-09-09 16:51:13 +02:00
parent 84adecf97b
commit 9a1e6d15a3
9 changed files with 124 additions and 83 deletions

View file

@ -273,6 +273,7 @@ func (cc *hugoBuilderCommon) handleCommonBuilderFlags(cmd *cobra.Command) {
cmd.PersistentFlags().StringVarP(&cc.environment, "environment", "e", "", "build environment") cmd.PersistentFlags().StringVarP(&cc.environment, "environment", "e", "", "build environment")
cmd.PersistentFlags().StringP("themesDir", "", "", "filesystem path to themes directory") cmd.PersistentFlags().StringP("themesDir", "", "", "filesystem path to themes directory")
cmd.PersistentFlags().BoolP("ignoreVendor", "", false, "ignores any _vendor directory") cmd.PersistentFlags().BoolP("ignoreVendor", "", false, "ignores any _vendor directory")
cmd.PersistentFlags().StringP("ignoreVendorPaths", "", "", "ignores any _vendor for module paths matching the given Glob pattern")
} }
func (cc *hugoBuilderCommon) handleFlags(cmd *cobra.Command) { func (cc *hugoBuilderCommon) handleFlags(cmd *cobra.Command) {

View file

@ -151,7 +151,7 @@ func readFileFrom(c *qt.C, filename string) string {
return string(b) return string(b)
} }
func TestCommandsPersistentFlags(t *testing.T) { func TestFlags(t *testing.T) {
c := qt.New(t) c := qt.New(t)
noOpRunE := func(cmd *cobra.Command, args []string) error { noOpRunE := func(cmd *cobra.Command, args []string) error {
@ -159,90 +159,103 @@ func TestCommandsPersistentFlags(t *testing.T) {
} }
tests := []struct { tests := []struct {
name string
args []string args []string
check func(command []cmder) check func(c *qt.C, cmd *serverCmd)
}{{[]string{"server", }{
"--config=myconfig.toml", {
"--configDir=myconfigdir", // https://github.com/gohugoio/hugo/issues/7642
"--contentDir=mycontent", name: "ignoreVendor as bool",
"--disableKinds=page,home", args: []string{"server", "--ignoreVendor"},
"--environment=testing", check: func(c *qt.C, cmd *serverCmd) {
"--configDir=myconfigdir", cfg := viper.New()
"--layoutDir=mylayouts", cmd.flagsToConfig(cfg)
"--theme=mytheme", c.Assert(cfg.Get("ignoreVendor"), qt.Equals, true)
"--gc", },
"--themesDir=mythemes", },
"--cleanDestinationDir", {
"--navigateToChanged", // https://github.com/gohugoio/hugo/issues/7642
"--disableLiveReload", name: "ignoreVendorPaths",
"--noHTTPCache", args: []string{"server", "--ignoreVendorPaths=github.com/**"},
"--i18n-warnings", check: func(c *qt.C, cmd *serverCmd) {
"--destination=/tmp/mydestination", cfg := viper.New()
"-b=https://example.com/b/", cmd.flagsToConfig(cfg)
"--port=1366", c.Assert(cfg.Get("ignoreVendorPaths"), qt.Equals, "github.com/**")
"--renderToDisk", },
"--source=mysource", },
"--path-warnings", {
}, func(commands []cmder) { name: "Persistent flags",
var sc *serverCmd args: []string{"server",
for _, command := range commands { "--config=myconfig.toml",
if b, ok := command.(commandsBuilderGetter); ok { "--configDir=myconfigdir",
v := b.getCommandsBuilder().hugoBuilderCommon "--contentDir=mycontent",
c.Assert(v.cfgFile, qt.Equals, "myconfig.toml") "--disableKinds=page,home",
c.Assert(v.cfgDir, qt.Equals, "myconfigdir") "--environment=testing",
c.Assert(v.source, qt.Equals, "mysource") "--configDir=myconfigdir",
c.Assert(v.baseURL, qt.Equals, "https://example.com/b/") "--layoutDir=mylayouts",
} "--theme=mytheme",
"--gc",
"--themesDir=mythemes",
"--cleanDestinationDir",
"--navigateToChanged",
"--disableLiveReload",
"--noHTTPCache",
"--i18n-warnings",
"--destination=/tmp/mydestination",
"-b=https://example.com/b/",
"--port=1366",
"--renderToDisk",
"--source=mysource",
"--path-warnings",
},
check: func(c *qt.C, sc *serverCmd) {
c.Assert(sc, qt.Not(qt.IsNil))
c.Assert(sc.navigateToChanged, qt.Equals, true)
c.Assert(sc.disableLiveReload, qt.Equals, true)
c.Assert(sc.noHTTPCache, qt.Equals, true)
c.Assert(sc.renderToDisk, qt.Equals, true)
c.Assert(sc.serverPort, qt.Equals, 1366)
c.Assert(sc.environment, qt.Equals, "testing")
if srvCmd, ok := command.(*serverCmd); ok { cfg := viper.New()
sc = srvCmd sc.flagsToConfig(cfg)
} c.Assert(cfg.GetString("publishDir"), qt.Equals, "/tmp/mydestination")
} c.Assert(cfg.GetString("contentDir"), qt.Equals, "mycontent")
c.Assert(cfg.GetString("layoutDir"), qt.Equals, "mylayouts")
c.Assert(cfg.GetStringSlice("theme"), qt.DeepEquals, []string{"mytheme"})
c.Assert(cfg.GetString("themesDir"), qt.Equals, "mythemes")
c.Assert(cfg.GetString("baseURL"), qt.Equals, "https://example.com/b/")
c.Assert(sc, qt.Not(qt.IsNil)) c.Assert(cfg.Get("disableKinds"), qt.DeepEquals, []string{"page", "home"})
c.Assert(sc.navigateToChanged, qt.Equals, true)
c.Assert(sc.disableLiveReload, qt.Equals, true)
c.Assert(sc.noHTTPCache, qt.Equals, true)
c.Assert(sc.renderToDisk, qt.Equals, true)
c.Assert(sc.serverPort, qt.Equals, 1366)
c.Assert(sc.environment, qt.Equals, "testing")
cfg := viper.New() c.Assert(cfg.GetBool("gc"), qt.Equals, true)
sc.flagsToConfig(cfg)
c.Assert(cfg.GetString("publishDir"), qt.Equals, "/tmp/mydestination")
c.Assert(cfg.GetString("contentDir"), qt.Equals, "mycontent")
c.Assert(cfg.GetString("layoutDir"), qt.Equals, "mylayouts")
c.Assert(cfg.GetStringSlice("theme"), qt.DeepEquals, []string{"mytheme"})
c.Assert(cfg.GetString("themesDir"), qt.Equals, "mythemes")
c.Assert(cfg.GetString("baseURL"), qt.Equals, "https://example.com/b/")
c.Assert(cfg.Get("disableKinds"), qt.DeepEquals, []string{"page", "home"}) // The flag is named path-warnings
c.Assert(cfg.GetBool("logPathWarnings"), qt.Equals, true)
c.Assert(cfg.GetBool("gc"), qt.Equals, true) // The flag is named i18n-warnings
c.Assert(cfg.GetBool("logI18nWarnings"), qt.Equals, true)
// The flag is named path-warnings }}}
c.Assert(cfg.GetBool("logPathWarnings"), qt.Equals, true)
// The flag is named i18n-warnings
c.Assert(cfg.GetBool("logI18nWarnings"), qt.Equals, true)
}}}
for _, test := range tests { for _, test := range tests {
b := newCommandsBuilder() c.Run(test.name, func(c *qt.C) {
root := b.addAll().build()
for _, c := range b.commands { b := newCommandsBuilder()
if c.getCommand() == nil { root := b.addAll().build()
continue
for _, cmd := range b.commands {
if cmd.getCommand() == nil {
continue
}
// We are only intereseted in the flag handling here.
cmd.getCommand().RunE = noOpRunE
} }
// We are only intereseted in the flag handling here. rootCmd := root.getCommand()
c.getCommand().RunE = noOpRunE rootCmd.SetArgs(test.args)
} c.Assert(rootCmd.Execute(), qt.IsNil)
rootCmd := root.getCommand() test.check(c, b.commands[0].(*serverCmd))
rootCmd.SetArgs(test.args) })
c.Assert(rootCmd.Execute(), qt.IsNil)
test.check(b.commands)
} }
} }

View file

@ -200,6 +200,7 @@ func initializeFlags(cmd *cobra.Command, cfg config.Provider) {
"noTimes", "noTimes",
"noChmod", "noChmod",
"ignoreVendor", "ignoreVendor",
"ignoreVendorPaths",
"templateMetrics", "templateMetrics",
"templateMetricsHints", "templateMetricsHints",

View file

@ -120,7 +120,7 @@ Note that:
* You can run `hugo mod vendor` on any level in the module tree. * You can run `hugo mod vendor` on any level in the module tree.
* Vendoring will not store modules stored in your `themes` folder. * Vendoring will not store modules stored in your `themes` folder.
* Most commands accept a `--ignoreVendor` flag, which will then run as if the none of the `_vendor` folders in the module tree existed. * Most commands accept a `--ignoreVendorPaths` flag, which will then not use the vendored modules in `_vendor` for the module paths matching the [Glob](https://github.com/gobwas/glob) pattern given. Note that before Hugo 0.75 this flag was named `--ignoreVendor` and was a "all or nothing". {{< new-in "0.75.0" >}}
Also see the [CLI Doc](/commands/hugo_mod_vendor/). Also see the [CLI Doc](/commands/hugo_mod_vendor/).

View file

@ -18,6 +18,9 @@ import (
"path/filepath" "path/filepath"
"strings" "strings"
"github.com/gobwas/glob"
hglob "github.com/gohugoio/hugo/hugofs/glob"
"github.com/gohugoio/hugo/common/loggers" "github.com/gohugoio/hugo/common/loggers"
"github.com/gohugoio/hugo/cache/filecache" "github.com/gohugoio/hugo/cache/filecache"
@ -202,6 +205,12 @@ func LoadConfig(d ConfigSourceDescriptor, doWithConfig ...func(cfg config.Provid
} }
} }
// We made this a Glob pattern in Hugo 0.75, we don't need both.
if v.GetBool("ignoreVendor") {
helpers.Deprecated("--ignoreVendor", "--ignoreVendorPaths **", false)
v.Set("ignoreVendorPaths", "**")
}
modulesConfig, err := l.loadModulesConfig(v) modulesConfig, err := l.loadModulesConfig(v)
if err != nil { if err != nil {
return v, configFiles, err return v, configFiles, err
@ -417,7 +426,10 @@ func (l configLoader) collectModules(modConfig modules.Config, v1 *viper.Viper,
themesDir := paths.AbsPathify(l.WorkingDir, v1.GetString("themesDir")) themesDir := paths.AbsPathify(l.WorkingDir, v1.GetString("themesDir"))
ignoreVendor := v1.GetBool("ignoreVendor") var ignoreVendor glob.Glob
if s := v1.GetString("ignoreVendorPaths"); s != "" {
ignoreVendor, _ = hglob.GetGlob(hglob.NormalizePath(s))
}
filecacheConfigs, err := filecache.DecodeConfig(l.Fs, v1) filecacheConfigs, err := filecache.DecodeConfig(l.Fs, v1)
if err != nil { if err != nil {

View file

@ -126,11 +126,15 @@ baseURL = "https://example.com"
title = "My Modular Site" title = "My Modular Site"
workingDir = %q workingDir = %q
theme = %q theme = %q
ignoreVendor = %t ignoreVendorPaths = %q
` `
config := fmt.Sprintf(configTemplate, workingDir, m.Path(), ignoreVendor) ignoreVendorPaths := ""
if ignoreVendor {
ignoreVendorPaths = "github.com/**"
}
config := fmt.Sprintf(configTemplate, workingDir, m.Path(), ignoreVendorPaths)
b := newTestSitesBuilder(t) b := newTestSitesBuilder(t)

View file

@ -605,8 +605,9 @@ type ClientConfig struct {
// etc. // etc.
HookBeforeFinalize func(m *ModulesConfig) error HookBeforeFinalize func(m *ModulesConfig) error
// Ignore any _vendor directory. // Ignore any _vendor directory for module paths matching the given pattern.
IgnoreVendor bool // This can be nil.
IgnoreVendor glob.Glob
// Absolute path to the project dir. // Absolute path to the project dir.
WorkingDir string WorkingDir string
@ -618,6 +619,10 @@ type ClientConfig struct {
ModuleConfig Config ModuleConfig Config
} }
func (c ClientConfig) shouldIgnoreVendor(path string) bool {
return c.IgnoreVendor != nil && c.IgnoreVendor.Match(path)
}
type goBinaryStatus int type goBinaryStatus int
type goModule struct { type goModule struct {

View file

@ -17,6 +17,8 @@ import (
"bytes" "bytes"
"testing" "testing"
"github.com/gohugoio/hugo/hugofs/glob"
"github.com/gohugoio/hugo/common/hugo" "github.com/gohugoio/hugo/common/hugo"
"github.com/gohugoio/hugo/htesting" "github.com/gohugoio/hugo/htesting"
@ -89,7 +91,7 @@ project github.com/gohugoio/hugoTestModules1_darwin/modh2_2_2@v1.3.0+vendor
Fs: hugofs.Os, Fs: hugofs.Os,
WorkingDir: workingDir, WorkingDir: workingDir,
ModuleConfig: modConfig, ModuleConfig: modConfig,
IgnoreVendor: true, IgnoreVendor: globAll,
}) })
graphb.Reset() graphb.Reset()
@ -101,6 +103,8 @@ project github.com/gohugoio/hugoTestModules1_darwin/modh2_2_2@v1.3.0+vendor
} }
var globAll, _ = glob.GetGlob("**")
func TestGetModlineSplitter(t *testing.T) { func TestGetModlineSplitter(t *testing.T) {
c := qt.New(t) c := qt.New(t)

View file

@ -196,7 +196,8 @@ func (c *collector) initModules() error {
gomods: goModules{}, gomods: goModules{},
} }
if !c.ccfg.IgnoreVendor && c.isVendored(c.ccfg.WorkingDir) { // If both these are true, we don't even need Go installed to build.
if c.ccfg.IgnoreVendor == nil && c.isVendored(c.ccfg.WorkingDir) {
return nil return nil
} }
@ -229,7 +230,7 @@ func (c *collector) add(owner *moduleAdapter, moduleImport Import, disabled bool
modulePath := moduleImport.Path modulePath := moduleImport.Path
var realOwner Module = owner var realOwner Module = owner
if !c.ccfg.IgnoreVendor { if !c.ccfg.shouldIgnoreVendor(modulePath) {
if err := c.collectModulesTXT(owner); err != nil { if err := c.collectModulesTXT(owner); err != nil {
return nil, err return nil, err
} }