From e71389de0741a76fe8e0cb689eb41af918d6ff5b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bj=C3=B8rn=20Erik=20Pedersen?= Date: Fri, 31 Jan 2020 09:09:11 +0100 Subject: [PATCH] commands: Fix config environment handling Fixes #6503 Fixes #6824 --- commands/=/content/p1.md | 7 ++ commands/commands.go | 14 ++-- commands/commands_test.go | 135 ++++++++++++++++++++++++++++++++--- commands/config.go | 21 +++--- commands/convert.go | 20 +++--- commands/deploy.go | 23 +++--- commands/list.go | 14 ++-- commands/list_test.go | 16 ++--- commands/new.go | 4 +- commands/new_content_test.go | 105 --------------------------- commands/new_site.go | 14 ++-- commands/new_theme.go | 13 ++-- 12 files changed, 201 insertions(+), 185 deletions(-) create mode 100644 commands/=/content/p1.md diff --git a/commands/=/content/p1.md b/commands/=/content/p1.md new file mode 100644 index 00000000000..a9365ea97c6 --- /dev/null +++ b/commands/=/content/p1.md @@ -0,0 +1,7 @@ +{ + "title": "P1", + "weight": 1 +} + +Content + diff --git a/commands/commands.go b/commands/commands.go index 2187f7aabe2..66fd9caa43d 100644 --- a/commands/commands.go +++ b/commands/commands.go @@ -47,12 +47,12 @@ func (b *commandsBuilder) addAll() *commandsBuilder { b.newServerCmd(), newVersionCmd(), newEnvCmd(), - newConfigCmd(), + b.newConfigCmd(), newCheckCmd(), - newDeployCmd(), - newConvertCmd(), + b.newDeployCmd(), + b.newConvertCmd(), b.newNewCmd(), - newListCmd(), + b.newListCmd(), newImportCmd(), newGenCmd(), createReleaser(), @@ -111,6 +111,12 @@ func (b *commandsBuilder) newBuilderCmd(cmd *cobra.Command) *baseBuilderCmd { return bcmd } +func (b *commandsBuilder) newBuilderBasicCmd(cmd *cobra.Command) *baseBuilderCmd { + bcmd := &baseBuilderCmd{commandsBuilder: b, baseCmd: &baseCmd{cmd: cmd}} + bcmd.hugoBuilderCommon.handleCommonBuilderFlags(cmd) + return bcmd +} + func (c *baseCmd) flagsToConfig(cfg config.Provider) { initializeFlags(c.cmd, cfg) } diff --git a/commands/commands_test.go b/commands/commands_test.go index 565736793a6..58200be45e7 100644 --- a/commands/commands_test.go +++ b/commands/commands_test.go @@ -20,6 +20,10 @@ import ( "path/filepath" "testing" + "github.com/spf13/afero" + + "github.com/gohugoio/hugo/hugofs" + "github.com/gohugoio/hugo/common/types" "github.com/spf13/cobra" @@ -32,18 +36,119 @@ func TestExecute(t *testing.T) { c := qt.New(t) - dir, err := createSimpleTestSite(t, testSiteConfig{}) - c.Assert(err, qt.IsNil) + createSite := func(c *qt.C) (string, func()) { + dir, err := createSimpleTestSite(t, testSiteConfig{}) + c.Assert(err, qt.IsNil) + return dir, func() { + os.RemoveAll(dir) + } + } - defer func() { - os.RemoveAll(dir) - }() + c.Run("hugo", func(c *qt.C) { + dir, clean := createSite(c) + defer clean() + resp := Execute([]string{"-s=" + dir}) + c.Assert(resp.Err, qt.IsNil) + result := resp.Result + c.Assert(len(result.Sites) == 1, qt.Equals, true) + c.Assert(len(result.Sites[0].RegularPages()) == 1, qt.Equals, true) + c.Assert(result.Sites[0].Info.Params()["myparam"], qt.Equals, "paramproduction") + }) + + c.Run("hugo, set environment", func(c *qt.C) { + dir, clean := createSite(c) + defer clean() + resp := Execute([]string{"-s=" + dir, "-e=staging"}) + c.Assert(resp.Err, qt.IsNil) + result := resp.Result + c.Assert(result.Sites[0].Info.Params()["myparam"], qt.Equals, "paramstaging") + }) + + c.Run("convert toJSON", func(c *qt.C) { + dir, clean := createSite(c) + output := filepath.Join(dir, "myjson") + defer clean() + resp := Execute([]string{"convert", "toJSON", "-s=" + dir, "-e=staging", "-o=" + output}) + c.Assert(resp.Err, qt.IsNil) + converted := readFileFrom(c, filepath.Join(output, "content", "p1.md")) + c.Assert(converted, qt.Equals, "{\n \"title\": \"P1\",\n \"weight\": 1\n}\n\nContent\n\n", qt.Commentf(converted)) + }) + + c.Run("config, set environment", func(c *qt.C) { + dir, clean := createSite(c) + defer clean() + out, err := captureStdout(func() error { + resp := Execute([]string{"config", "-s=" + dir, "-e=staging"}) + return resp.Err + }) + c.Assert(err, qt.IsNil) + c.Assert(out, qt.Contains, "params = map[myparam:paramstaging]", qt.Commentf(out)) + }) + + c.Run("deploy, environment set", func(c *qt.C) { + dir, clean := createSite(c) + defer clean() + resp := Execute([]string{"deploy", "-s=" + dir, "-e=staging", "--target=mydeployment", "--dryRun"}) + c.Assert(resp.Err, qt.Not(qt.IsNil)) + c.Assert(resp.Err.Error(), qt.Contains, "open bucket gs://hugotestbucket: google: could not find default credentials") + }) + + c.Run("list", func(c *qt.C) { + dir, clean := createSite(c) + defer clean() + out, err := captureStdout(func() error { + resp := Execute([]string{"list", "all", "-s=" + dir, "-e=staging"}) + return resp.Err + }) + c.Assert(err, qt.IsNil) + c.Assert(out, qt.Contains, "p1.md") + }) + + c.Run("new theme", func(c *qt.C) { + dir, clean := createSite(c) + defer clean() + themesDir := filepath.Join(dir, "mythemes") + resp := Execute([]string{"new", "theme", "mytheme", "-s=" + dir, "-e=staging", "--themesDir=" + themesDir}) + c.Assert(resp.Err, qt.IsNil) + themeTOML := readFileFrom(c, filepath.Join(themesDir, "mytheme", "theme.toml")) + c.Assert(themeTOML, qt.Contains, "name = \"Mytheme\"") + }) + + c.Run("new site", func(c *qt.C) { + dir, clean := createSite(c) + defer clean() + siteDir := filepath.Join(dir, "mysite") + resp := Execute([]string{"new", "site", siteDir, "-e=staging"}) + c.Assert(resp.Err, qt.IsNil) + config := readFileFrom(c, filepath.Join(siteDir, "config.toml")) + c.Assert(config, qt.Contains, "baseURL = \"http://example.org/\"") + checkNewSiteInited(c, siteDir) + }) + +} + +func checkNewSiteInited(c *qt.C, basepath string) { + paths := []string{ + filepath.Join(basepath, "layouts"), + filepath.Join(basepath, "content"), + filepath.Join(basepath, "archetypes"), + filepath.Join(basepath, "static"), + filepath.Join(basepath, "data"), + filepath.Join(basepath, "config.toml"), + } + + for _, path := range paths { + _, err := os.Stat(path) + c.Assert(err, qt.IsNil) + } +} - resp := Execute([]string{"-s=" + dir}) - c.Assert(resp.Err, qt.IsNil) - result := resp.Result - c.Assert(len(result.Sites) == 1, qt.Equals, true) - c.Assert(len(result.Sites[0].RegularPages()) == 1, qt.Equals, true) +func readFileFrom(c *qt.C, filename string) string { + c.Helper() + filename = filepath.Clean(filename) + b, err := afero.ReadFile(hugofs.Os, filename) + c.Assert(err, qt.IsNil) + return string(b) } func TestCommandsPersistentFlags(t *testing.T) { @@ -233,6 +338,7 @@ func createSimpleTestSite(t *testing.T, cfg testSiteConfig) (string, error) { baseURL = "https://example.org" title = "Hugo Commands" + ` contentDir := "content" @@ -246,6 +352,15 @@ title = "Hugo Commands" // Just the basic. These are for CLI tests, not site testing. writeFile(t, filepath.Join(d, "config.toml"), cfgStr) + writeFile(t, filepath.Join(d, "config", "staging", "params.toml"), `myparam="paramstaging"`) + writeFile(t, filepath.Join(d, "config", "staging", "deployment.toml"), ` +[[targets]] +name = "mydeployment" +URL = "gs://hugotestbucket" +`) + + writeFile(t, filepath.Join(d, "config", "testing", "params.toml"), `myparam="paramtesting"`) + writeFile(t, filepath.Join(d, "config", "production", "params.toml"), `myparam="paramproduction"`) writeFile(t, filepath.Join(d, contentDir, "p1.md"), ` --- diff --git a/commands/config.go b/commands/config.go index 72c2a0d9738..37bf45e3cd0 100644 --- a/commands/config.go +++ b/commands/config.go @@ -15,6 +15,7 @@ package commands import ( "encoding/json" + "fmt" "os" "reflect" "regexp" @@ -27,27 +28,23 @@ import ( "github.com/gohugoio/hugo/modules" "github.com/spf13/cobra" - jww "github.com/spf13/jwalterweatherman" "github.com/spf13/viper" ) var _ cmder = (*configCmd)(nil) type configCmd struct { - hugoBuilderCommon - *baseCmd + *baseBuilderCmd } -func newConfigCmd() *configCmd { +func (b *commandsBuilder) newConfigCmd() *configCmd { cc := &configCmd{} - cc.baseCmd = newBaseCmd(&cobra.Command{ + cmd := &cobra.Command{ Use: "config", Short: "Print the site configuration", Long: `Print the site configuration, both default and custom settings.`, RunE: cc.printConfig, - }) - - cc.cmd.PersistentFlags().StringVarP(&cc.source, "source", "s", "", "filesystem path to read files relative from") + } printMountsCmd := &cobra.Command{ Use: "mounts", @@ -55,7 +52,9 @@ func newConfigCmd() *configCmd { RunE: cc.printMounts, } - cc.cmd.AddCommand(printMountsCmd) + cmd.AddCommand(printMountsCmd) + + cc.baseBuilderCmd = b.newBuilderBasicCmd(cmd) return cc } @@ -105,9 +104,9 @@ func (c *configCmd) printConfig(cmd *cobra.Command, args []string) error { for _, k := range keys { kv := reflect.ValueOf(allSettings[k]) if kv.Kind() == reflect.String { - jww.FEEDBACK.Printf("%s%s\"%+v\"\n", k, separator, allSettings[k]) + fmt.Printf("%s%s\"%+v\"\n", k, separator, allSettings[k]) } else { - jww.FEEDBACK.Printf("%s%s%+v\n", k, separator, allSettings[k]) + fmt.Printf("%s%s%+v\n", k, separator, allSettings[k]) } } diff --git a/commands/convert.go b/commands/convert.go index e4ff1ac61c5..b9129e594a9 100644 --- a/commands/convert.go +++ b/commands/convert.go @@ -44,27 +44,25 @@ var ( ) type convertCmd struct { - hugoBuilderCommon - outputDir string unsafe bool - *baseCmd + *baseBuilderCmd } -func newConvertCmd() *convertCmd { +func (b *commandsBuilder) newConvertCmd() *convertCmd { cc := &convertCmd{} - cc.baseCmd = newBaseCmd(&cobra.Command{ + cmd := &cobra.Command{ Use: "convert", Short: "Convert your content to different formats", Long: `Convert your content (e.g. front matter) to different formats. See convert's subcommands toJSON, toTOML and toYAML for more information.`, RunE: nil, - }) + } - cc.cmd.AddCommand( + cmd.AddCommand( &cobra.Command{ Use: "toJSON", Short: "Convert front matter to JSON", @@ -94,10 +92,10 @@ to use YAML for the front matter.`, }, ) - cc.cmd.PersistentFlags().StringVarP(&cc.outputDir, "output", "o", "", "filesystem path to write files to") - cc.cmd.PersistentFlags().StringVarP(&cc.source, "source", "s", "", "filesystem path to read files relative from") - cc.cmd.PersistentFlags().BoolVar(&cc.unsafe, "unsafe", false, "enable less safe operations, please backup first") - cc.cmd.PersistentFlags().SetAnnotation("source", cobra.BashCompSubdirsInDir, []string{}) + cmd.PersistentFlags().StringVarP(&cc.outputDir, "output", "o", "", "filesystem path to write files to") + cmd.PersistentFlags().BoolVar(&cc.unsafe, "unsafe", false, "enable less safe operations, please backup first") + + cc.baseBuilderCmd = b.newBuilderBasicCmd(cmd) return cc } diff --git a/commands/deploy.go b/commands/deploy.go index d74d93709c4..ab51c9eb6fa 100644 --- a/commands/deploy.go +++ b/commands/deploy.go @@ -24,8 +24,7 @@ var _ cmder = (*deployCmd)(nil) // deployCmd supports deploying sites to Cloud providers. type deployCmd struct { - hugoBuilderCommon - *baseCmd + *baseBuilderCmd } // TODO: In addition to the "deploy" command, consider adding a "--deploy" @@ -38,10 +37,10 @@ type deployCmd struct { // run "hugo && hugo deploy" again and again and upload new stuff every time. Is // this intended? -func newDeployCmd() *deployCmd { +func (b *commandsBuilder) newDeployCmd() *deployCmd { cc := &deployCmd{} - cc.baseCmd = newBaseCmd(&cobra.Command{ + cmd := &cobra.Command{ Use: "deploy", Short: "Deploy your site to a Cloud provider.", Long: `Deploy your site to a Cloud provider. @@ -64,14 +63,16 @@ documentation. } return deployer.Deploy(context.Background()) }, - }) + } - cc.cmd.Flags().String("target", "", "target deployment from deployments section in config file; defaults to the first one") - cc.cmd.Flags().Bool("confirm", false, "ask for confirmation before making changes to the target") - cc.cmd.Flags().Bool("dryRun", false, "dry run") - cc.cmd.Flags().Bool("force", false, "force upload of all files") - cc.cmd.Flags().Bool("invalidateCDN", true, "invalidate the CDN cache listed in the deployment target") - cc.cmd.Flags().Int("maxDeletes", 256, "maximum # of files to delete, or -1 to disable") + cmd.Flags().String("target", "", "target deployment from deployments section in config file; defaults to the first one") + cmd.Flags().Bool("confirm", false, "ask for confirmation before making changes to the target") + cmd.Flags().Bool("dryRun", false, "dry run") + cmd.Flags().Bool("force", false, "force upload of all files") + cmd.Flags().Bool("invalidateCDN", true, "invalidate the CDN cache listed in the deployment target") + cmd.Flags().Int("maxDeletes", 256, "maximum # of files to delete, or -1 to disable") + + cc.baseBuilderCmd = b.newBuilderBasicCmd(cmd) return cc } diff --git a/commands/list.go b/commands/list.go index f233ce62c56..0b7c187978d 100644 --- a/commands/list.go +++ b/commands/list.go @@ -29,8 +29,7 @@ import ( var _ cmder = (*listCmd)(nil) type listCmd struct { - hugoBuilderCommon - *baseCmd + *baseBuilderCmd } func (lc *listCmd) buildSites(config map[string]interface{}) (*hugolib.HugoSites, error) { @@ -59,19 +58,19 @@ func (lc *listCmd) buildSites(config map[string]interface{}) (*hugolib.HugoSites return sites, nil } -func newListCmd() *listCmd { +func (b *commandsBuilder) newListCmd() *listCmd { cc := &listCmd{} - cc.baseCmd = newBaseCmd(&cobra.Command{ + cmd := &cobra.Command{ Use: "list", Short: "Listing out various types of content", Long: `Listing out various types of content. List requires a subcommand, e.g. ` + "`hugo list drafts`.", RunE: nil, - }) + } - cc.cmd.AddCommand( + cmd.AddCommand( &cobra.Command{ Use: "drafts", Short: "List all drafts", @@ -202,8 +201,7 @@ List requires a subcommand, e.g. ` + "`hugo list drafts`.", }, ) - cc.cmd.PersistentFlags().StringVarP(&cc.source, "source", "s", "", "filesystem path to read files relative from") - cc.cmd.PersistentFlags().SetAnnotation("source", cobra.BashCompSubdirsInDir, []string{}) + cc.baseBuilderCmd = b.newBuilderBasicCmd(cmd) return cc } diff --git a/commands/list_test.go b/commands/list_test.go index bfc28067958..1216bd7daac 100644 --- a/commands/list_test.go +++ b/commands/list_test.go @@ -10,26 +10,21 @@ import ( "testing" qt "github.com/frankban/quicktest" - "github.com/spf13/cobra" ) -func captureStdout(f func() (*cobra.Command, error)) (string, error) { +func captureStdout(f func() error) (string, error) { old := os.Stdout r, w, _ := os.Pipe() os.Stdout = w - _, err := f() - - if err != nil { - return "", err - } + err := f() w.Close() os.Stdout = old var buf bytes.Buffer io.Copy(&buf, r) - return buf.String(), nil + return buf.String(), err } func TestListAll(t *testing.T) { @@ -47,7 +42,10 @@ func TestListAll(t *testing.T) { cmd.SetArgs([]string{"-s=" + dir, "list", "all"}) - out, err := captureStdout(cmd.ExecuteC) + out, err := captureStdout(func() error { + _, err := cmd.ExecuteC() + return err + }) c.Assert(err, qt.IsNil) r := csv.NewReader(strings.NewReader(out)) diff --git a/commands/new.go b/commands/new.go index 4fc0d4ed40a..576976e8eba 100644 --- a/commands/new.go +++ b/commands/new.go @@ -55,8 +55,8 @@ Ensure you run this within the root directory of your site.`, cmd.Flags().StringVarP(&cc.contentType, "kind", "k", "", "content type to create") cmd.Flags().StringVar(&cc.contentEditor, "editor", "", "edit new content with this editor, if provided") - cmd.AddCommand(newNewSiteCmd().getCommand()) - cmd.AddCommand(newNewThemeCmd().getCommand()) + cmd.AddCommand(b.newNewSiteCmd().getCommand()) + cmd.AddCommand(b.newNewThemeCmd().getCommand()) cmd.RunE = cc.newContent diff --git a/commands/new_content_test.go b/commands/new_content_test.go index 36726e37a0b..42a7c968c0e 100644 --- a/commands/new_content_test.go +++ b/commands/new_content_test.go @@ -17,9 +17,6 @@ import ( "path/filepath" "testing" - "github.com/gohugoio/hugo/hugofs" - "github.com/spf13/viper" - qt "github.com/frankban/quicktest" ) @@ -30,105 +27,3 @@ func TestNewContentPathSectionWithForwardSlashes(t *testing.T) { c.Assert(p, qt.Equals, filepath.FromSlash("/post/new.md")) c.Assert(s, qt.Equals, "post") } - -func checkNewSiteInited(fs *hugofs.Fs, basepath string, t *testing.T) { - c := qt.New(t) - paths := []string{ - filepath.Join(basepath, "layouts"), - filepath.Join(basepath, "content"), - filepath.Join(basepath, "archetypes"), - filepath.Join(basepath, "static"), - filepath.Join(basepath, "data"), - filepath.Join(basepath, "config.toml"), - } - - for _, path := range paths { - _, err := fs.Source.Stat(path) - c.Assert(err, qt.IsNil) - } -} - -func TestDoNewSite(t *testing.T) { - c := qt.New(t) - n := newNewSiteCmd() - basepath := filepath.Join("base", "blog") - _, fs := newTestCfg() - - c.Assert(n.doNewSite(fs, basepath, false), qt.IsNil) - - checkNewSiteInited(fs, basepath, t) -} - -func TestDoNewSite_noerror_base_exists_but_empty(t *testing.T) { - c := qt.New(t) - basepath := filepath.Join("base", "blog") - _, fs := newTestCfg() - n := newNewSiteCmd() - - c.Assert(fs.Source.MkdirAll(basepath, 0777), qt.IsNil) - - c.Assert(n.doNewSite(fs, basepath, false), qt.IsNil) -} - -func TestDoNewSite_error_base_exists(t *testing.T) { - c := qt.New(t) - basepath := filepath.Join("base", "blog") - _, fs := newTestCfg() - n := newNewSiteCmd() - - c.Assert(fs.Source.MkdirAll(basepath, 0777), qt.IsNil) - _, err := fs.Source.Create(filepath.Join(basepath, "foo")) - c.Assert(err, qt.IsNil) - // Since the directory already exists and isn't empty, expect an error - c.Assert(n.doNewSite(fs, basepath, false), qt.Not(qt.IsNil)) - -} - -func TestDoNewSite_force_empty_dir(t *testing.T) { - c := qt.New(t) - basepath := filepath.Join("base", "blog") - _, fs := newTestCfg() - n := newNewSiteCmd() - - c.Assert(fs.Source.MkdirAll(basepath, 0777), qt.IsNil) - c.Assert(n.doNewSite(fs, basepath, true), qt.IsNil) - - checkNewSiteInited(fs, basepath, t) -} - -func TestDoNewSite_error_force_dir_inside_exists(t *testing.T) { - c := qt.New(t) - basepath := filepath.Join("base", "blog") - _, fs := newTestCfg() - n := newNewSiteCmd() - - contentPath := filepath.Join(basepath, "content") - - c.Assert(fs.Source.MkdirAll(contentPath, 0777), qt.IsNil) - c.Assert(n.doNewSite(fs, basepath, true), qt.Not(qt.IsNil)) -} - -func TestDoNewSite_error_force_config_inside_exists(t *testing.T) { - c := qt.New(t) - basepath := filepath.Join("base", "blog") - _, fs := newTestCfg() - n := newNewSiteCmd() - - configPath := filepath.Join(basepath, "config.toml") - c.Assert(fs.Source.MkdirAll(basepath, 0777), qt.IsNil) - _, err := fs.Source.Create(configPath) - c.Assert(err, qt.IsNil) - - c.Assert(n.doNewSite(fs, basepath, true), qt.Not(qt.IsNil)) -} - -func newTestCfg() (*viper.Viper, *hugofs.Fs) { - - v := viper.New() - fs := hugofs.NewMem(v) - - v.SetFs(fs.Source) - - return v, fs - -} diff --git a/commands/new_site.go b/commands/new_site.go index f884a6433f1..9fb47096a8b 100644 --- a/commands/new_site.go +++ b/commands/new_site.go @@ -37,11 +37,11 @@ var _ cmder = (*newSiteCmd)(nil) type newSiteCmd struct { configFormat string - *baseCmd + *baseBuilderCmd } -func newNewSiteCmd() *newSiteCmd { - ccmd := &newSiteCmd{} +func (b *commandsBuilder) newNewSiteCmd() *newSiteCmd { + cc := &newSiteCmd{} cmd := &cobra.Command{ Use: "site [path]", @@ -49,15 +49,15 @@ func newNewSiteCmd() *newSiteCmd { Long: `Create a new site in the provided directory. The new site will have the correct structure, but no content or theme yet. Use ` + "`hugo new [contentPath]`" + ` to create new content.`, - RunE: ccmd.newSite, + RunE: cc.newSite, } - cmd.Flags().StringVarP(&ccmd.configFormat, "format", "f", "toml", "config & frontmatter format") + cmd.Flags().StringVarP(&cc.configFormat, "format", "f", "toml", "config & frontmatter format") cmd.Flags().Bool("force", false, "init inside non-empty directory") - ccmd.baseCmd = newBaseCmd(cmd) + cc.baseBuilderCmd = b.newBuilderBasicCmd(cmd) - return ccmd + return cc } diff --git a/commands/new_theme.go b/commands/new_theme.go index a0a4e89e327..ee3437360b0 100644 --- a/commands/new_theme.go +++ b/commands/new_theme.go @@ -29,12 +29,11 @@ import ( var _ cmder = (*newThemeCmd)(nil) type newThemeCmd struct { - *baseCmd - hugoBuilderCommon + *baseBuilderCmd } -func newNewThemeCmd() *newThemeCmd { - ccmd := &newThemeCmd{baseCmd: newBaseCmd(nil)} +func (b *commandsBuilder) newNewThemeCmd() *newThemeCmd { + cc := &newThemeCmd{} cmd := &cobra.Command{ Use: "theme [name]", @@ -43,12 +42,12 @@ func newNewThemeCmd() *newThemeCmd { New theme is a skeleton. Please add content to the touched files. Add your name to the copyright line in the license and adjust the theme.toml file as you see fit.`, - RunE: ccmd.newTheme, + RunE: cc.newTheme, } - ccmd.cmd = cmd + cc.baseBuilderCmd = b.newBuilderBasicCmd(cmd) - return ccmd + return cc } // newTheme creates a new Hugo theme template