2023-04-10 15:36:59 +00:00
|
|
|
// Copyright (c) HashiCorp, Inc.
|
|
|
|
// SPDX-License-Identifier: MPL-2.0
|
|
|
|
|
2015-09-30 21:21:50 +00:00
|
|
|
package command
|
|
|
|
|
|
|
|
import (
|
2023-02-02 15:37:40 +00:00
|
|
|
"fmt"
|
2015-09-30 21:21:50 +00:00
|
|
|
"os"
|
|
|
|
"strings"
|
|
|
|
"testing"
|
|
|
|
|
2022-03-15 12:42:43 +00:00
|
|
|
"github.com/hashicorp/nomad/ci"
|
2023-02-10 17:02:29 +00:00
|
|
|
"github.com/hashicorp/nomad/command/asset"
|
2015-09-30 21:21:50 +00:00
|
|
|
"github.com/mitchellh/cli"
|
2023-02-02 15:37:40 +00:00
|
|
|
"github.com/shoenig/test/must"
|
2018-05-01 16:51:13 +00:00
|
|
|
"github.com/stretchr/testify/require"
|
2015-09-30 21:21:50 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestInitCommand_Implements(t *testing.T) {
|
2022-03-15 12:42:43 +00:00
|
|
|
ci.Parallel(t)
|
2018-03-21 00:37:28 +00:00
|
|
|
var _ cli.Command = &JobInitCommand{}
|
2015-09-30 21:21:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestInitCommand_Run(t *testing.T) {
|
2022-03-15 12:42:43 +00:00
|
|
|
ci.Parallel(t)
|
2020-10-05 14:07:41 +00:00
|
|
|
ui := cli.NewMockUi()
|
2018-03-21 00:37:28 +00:00
|
|
|
cmd := &JobInitCommand{Meta: Meta{Ui: ui}}
|
2015-09-30 21:21:50 +00:00
|
|
|
|
|
|
|
// Fails on misuse
|
|
|
|
if code := cmd.Run([]string{"some", "bad", "args"}); code != 1 {
|
|
|
|
t.Fatalf("expect exit code 1, got: %d", code)
|
|
|
|
}
|
2018-04-18 17:51:17 +00:00
|
|
|
if out := ui.ErrorWriter.String(); !strings.Contains(out, commandErrorText(cmd)) {
|
2015-09-30 21:21:50 +00:00
|
|
|
t.Fatalf("expect help output, got: %s", out)
|
|
|
|
}
|
|
|
|
ui.ErrorWriter.Reset()
|
|
|
|
|
|
|
|
// Ensure we change the cwd back
|
|
|
|
origDir, err := os.Getwd()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
defer os.Chdir(origDir)
|
|
|
|
|
|
|
|
// Create a temp dir and change into it
|
2022-05-12 15:42:40 +00:00
|
|
|
dir := t.TempDir()
|
2015-09-30 21:21:50 +00:00
|
|
|
if err := os.Chdir(dir); err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Works if the file doesn't exist
|
|
|
|
if code := cmd.Run([]string{}); code != 0 {
|
|
|
|
t.Fatalf("expect exit code 0, got: %d", code)
|
|
|
|
}
|
2023-03-08 15:20:04 +00:00
|
|
|
content, err := os.ReadFile(DefaultInitName)
|
2015-09-30 21:21:50 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
2023-02-10 17:02:29 +00:00
|
|
|
defaultJob := asset.JobExample
|
2019-08-14 14:23:33 +00:00
|
|
|
if string(content) != string(defaultJob) {
|
2015-09-30 21:21:50 +00:00
|
|
|
t.Fatalf("unexpected file content\n\n%s", string(content))
|
|
|
|
}
|
|
|
|
|
2018-05-01 16:51:13 +00:00
|
|
|
// Works with -short flag
|
|
|
|
os.Remove(DefaultInitName)
|
|
|
|
if code := cmd.Run([]string{"-short"}); code != 0 {
|
|
|
|
require.Zero(t, code, "unexpected exit code: %d", code)
|
|
|
|
}
|
2023-03-08 15:20:04 +00:00
|
|
|
content, err = os.ReadFile(DefaultInitName)
|
2018-05-01 16:51:13 +00:00
|
|
|
require.NoError(t, err)
|
2023-02-10 17:02:29 +00:00
|
|
|
shortJob := asset.JobExampleShort
|
2019-08-14 14:23:33 +00:00
|
|
|
require.Equal(t, string(content), string(shortJob))
|
2018-05-01 16:51:13 +00:00
|
|
|
|
2015-09-30 21:21:50 +00:00
|
|
|
// Fails if the file exists
|
|
|
|
if code := cmd.Run([]string{}); code != 1 {
|
|
|
|
t.Fatalf("expect exit code 1, got: %d", code)
|
|
|
|
}
|
|
|
|
if out := ui.ErrorWriter.String(); !strings.Contains(out, "exists") {
|
|
|
|
t.Fatalf("expect file exists error, got: %s", out)
|
|
|
|
}
|
|
|
|
}
|
2016-09-24 18:42:25 +00:00
|
|
|
|
|
|
|
func TestInitCommand_defaultJob(t *testing.T) {
|
2022-03-15 12:42:43 +00:00
|
|
|
ci.Parallel(t)
|
2016-09-24 18:42:25 +00:00
|
|
|
// Ensure the job file is always written with spaces instead of tabs. Since
|
|
|
|
// the default job file is embedded in the go file, it's easy for tabs to
|
|
|
|
// slip in.
|
2023-02-10 17:02:29 +00:00
|
|
|
defaultJob := asset.JobExample
|
2019-08-14 14:23:33 +00:00
|
|
|
if strings.Contains(string(defaultJob), "\t") {
|
2016-09-24 18:42:25 +00:00
|
|
|
t.Error("default job contains tab character - please convert to spaces")
|
|
|
|
}
|
|
|
|
}
|
2019-12-19 19:59:12 +00:00
|
|
|
|
2023-02-02 15:37:40 +00:00
|
|
|
func TestInitCommand_listTemplates(t *testing.T) {
|
|
|
|
ci.Parallel(t)
|
|
|
|
srv, _, url := testServer(t, true, nil)
|
|
|
|
defer srv.Shutdown()
|
|
|
|
|
|
|
|
ui := cli.NewMockUi()
|
|
|
|
|
|
|
|
jobCmd := &JobInitCommand{Meta: Meta{Ui: ui}}
|
|
|
|
jobCmd.Run([]string{"-address=" + url, "-list-templates"})
|
|
|
|
expectedOutput := "No variables in nomad/job-templates\n"
|
|
|
|
must.StrContains(t, ui.ErrorWriter.String(), expectedOutput)
|
|
|
|
|
|
|
|
varCmd := &VarPutCommand{Meta: Meta{Ui: ui}}
|
|
|
|
// Set up 3 job template variables
|
|
|
|
for i := 1; i <= 3; i++ {
|
|
|
|
templateName := fmt.Sprintf("template-%d", i)
|
|
|
|
must.Eq(t, 0, varCmd.Run([]string{"-address=" + url, "-out=json", "nomad/job-templates/" + templateName, "k1=v1", "k2=v2", "k3=v3"}))
|
|
|
|
}
|
|
|
|
ui.ErrorWriter.Reset()
|
|
|
|
ui.OutputWriter.Reset()
|
|
|
|
|
|
|
|
jobCmd = &JobInitCommand{Meta: Meta{Ui: ui}}
|
|
|
|
must.Eq(t, 0, jobCmd.Run([]string{"-address=" + url, "-list-templates"}))
|
|
|
|
expectedOutput = "Use nomad job init -template=<template> with any of the following:\n template-1\n template-2\n template-3\n"
|
|
|
|
must.StrContains(t, ui.OutputWriter.String(), expectedOutput)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestInitCommand_fromJobTemplate(t *testing.T) {
|
|
|
|
ci.Parallel(t)
|
|
|
|
srv, _, url := testServer(t, true, nil)
|
|
|
|
defer srv.Shutdown()
|
|
|
|
|
|
|
|
ui := cli.NewMockUi()
|
|
|
|
varCmd := &VarPutCommand{Meta: Meta{Ui: ui}}
|
|
|
|
|
|
|
|
tinyJob := `job "tiny" {
|
|
|
|
group "foo" {
|
|
|
|
task "bar" {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}`
|
|
|
|
|
|
|
|
// Set up job template variables
|
|
|
|
varCmd.Run([]string{"-address=" + url, "-out=json", "nomad/job-templates/invalid-template", "k1=v1"})
|
|
|
|
varCmd.Run([]string{"-address=" + url, "-out=json", "nomad/job-templates/valid-template", "template=" + tinyJob})
|
|
|
|
ui.ErrorWriter.Reset()
|
|
|
|
ui.OutputWriter.Reset()
|
|
|
|
|
|
|
|
// Ensure we change the cwd back
|
|
|
|
origDir, err := os.Getwd()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
defer os.Chdir(origDir)
|
|
|
|
|
|
|
|
// Create a temp dir and change into it
|
|
|
|
dir := t.TempDir()
|
|
|
|
if err := os.Chdir(dir); err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
jobCmd := &JobInitCommand{Meta: Meta{Ui: ui}}
|
|
|
|
|
2023-02-10 17:02:29 +00:00
|
|
|
// Doesn't work if our var lacks a template key
|
2023-02-02 15:37:40 +00:00
|
|
|
must.Eq(t, 1, jobCmd.Run([]string{"-address=" + url, "-template=invalid-template"}))
|
|
|
|
|
|
|
|
// Works if the file doesn't exist
|
|
|
|
must.Eq(t, 0, jobCmd.Run([]string{"-address=" + url, "-template=valid-template"}))
|
|
|
|
|
2023-03-08 15:20:04 +00:00
|
|
|
content, err := os.ReadFile(DefaultInitName)
|
2023-02-02 15:37:40 +00:00
|
|
|
must.NoError(t, err)
|
|
|
|
must.Eq(t, string(content), string(tinyJob))
|
|
|
|
|
|
|
|
ui.ErrorWriter.Reset()
|
2023-02-02 17:47:47 +00:00
|
|
|
expectedOutput := "Initializing a job template from valid-template\nExample job file written to example.nomad.hcl\n"
|
2023-02-02 15:37:40 +00:00
|
|
|
must.StrContains(t, ui.OutputWriter.String(), expectedOutput)
|
|
|
|
}
|
|
|
|
|
2019-12-19 19:59:12 +00:00
|
|
|
func TestInitCommand_customFilename(t *testing.T) {
|
2022-03-15 12:42:43 +00:00
|
|
|
ci.Parallel(t)
|
2020-10-05 14:07:41 +00:00
|
|
|
ui := cli.NewMockUi()
|
2019-12-19 19:59:12 +00:00
|
|
|
cmd := &JobInitCommand{Meta: Meta{Ui: ui}}
|
|
|
|
filename := "custom.nomad"
|
|
|
|
|
|
|
|
// Ensure we change the cwd back
|
|
|
|
origDir, err := os.Getwd()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
defer os.Chdir(origDir)
|
|
|
|
|
|
|
|
// Create a temp dir and change into it
|
2022-05-12 15:42:40 +00:00
|
|
|
dir := t.TempDir()
|
2019-12-19 19:59:12 +00:00
|
|
|
if err := os.Chdir(dir); err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Works if the file doesn't exist
|
|
|
|
if code := cmd.Run([]string{filename}); code != 0 {
|
|
|
|
t.Fatalf("expect exit code 0, got: %d", code)
|
|
|
|
}
|
2023-03-08 15:20:04 +00:00
|
|
|
content, err := os.ReadFile(filename)
|
2019-12-19 19:59:12 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err: %s", err)
|
|
|
|
}
|
2023-02-10 17:02:29 +00:00
|
|
|
defaultJob := asset.JobExample
|
2019-12-19 19:59:12 +00:00
|
|
|
if string(content) != string(defaultJob) {
|
|
|
|
t.Fatalf("unexpected file content\n\n%s", string(content))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Works with -short flag
|
|
|
|
os.Remove(filename)
|
|
|
|
if code := cmd.Run([]string{"-short", filename}); code != 0 {
|
|
|
|
require.Zero(t, code, "unexpected exit code: %d", code)
|
|
|
|
}
|
2023-03-08 15:20:04 +00:00
|
|
|
content, err = os.ReadFile(filename)
|
2019-12-19 19:59:12 +00:00
|
|
|
require.NoError(t, err)
|
2023-02-10 17:02:29 +00:00
|
|
|
shortJob := asset.JobExampleShort
|
2019-12-19 19:59:12 +00:00
|
|
|
require.Equal(t, string(content), string(shortJob))
|
|
|
|
|
|
|
|
// Fails if the file exists
|
|
|
|
if code := cmd.Run([]string{filename}); code != 1 {
|
|
|
|
t.Fatalf("expect exit code 1, got: %d", code)
|
|
|
|
}
|
|
|
|
if out := ui.ErrorWriter.String(); !strings.Contains(out, "exists") {
|
|
|
|
t.Fatalf("expect file exists error, got: %s", out)
|
|
|
|
}
|
|
|
|
}
|