From 3d6beda9fc53091145b830a45fba52bb5c036d8d Mon Sep 17 00:00:00 2001 From: Luiz Aoqui Date: Tue, 26 Oct 2021 19:19:28 -0400 Subject: [PATCH 1/3] allow using specific object id on diff --- nomad/structs/diff.go | 34 ++++++++++++++++++++++++++-------- nomad/structs/diff_test.go | 28 ++++++++++++++++++++++++++-- nomad/structs/structs.go | 10 ++++++++++ 3 files changed, 62 insertions(+), 10 deletions(-) diff --git a/nomad/structs/diff.go b/nomad/structs/diff.go index 845c37c5a52..d3b2406da52 100644 --- a/nomad/structs/diff.go +++ b/nomad/structs/diff.go @@ -10,6 +10,14 @@ import ( "github.com/mitchellh/hashstructure" ) +// DiffableWithID defines an object that have an unique and stable field that +// can be used as an identifier when generating a diff. +type DiffableWithID interface { + // DiffID returns the value to use to match entities between the old and + // the new input. + DiffID() string +} + // DiffType denotes the type of a diff object. type DiffType string @@ -2419,11 +2427,20 @@ func primitiveObjectSetDiff(old, new []interface{}, filter []string, name string makeSet := func(objects []interface{}) map[string]interface{} { objMap := make(map[string]interface{}, len(objects)) for _, obj := range objects { - hash, err := hashstructure.Hash(obj, nil) - if err != nil { - panic(err) + var key string + + if diffable, ok := obj.(DiffableWithID); ok { + key = diffable.DiffID() } - objMap[fmt.Sprintf("%d", hash)] = obj + + if key == "" { + hash, err := hashstructure.Hash(obj, nil) + if err != nil { + panic(err) + } + key = fmt.Sprintf("%d", hash) + } + objMap[key] = obj } return objMap @@ -2433,10 +2450,11 @@ func primitiveObjectSetDiff(old, new []interface{}, filter []string, name string newSet := makeSet(new) var diffs []*ObjectDiff - for k, v := range oldSet { - // Deleted - if _, ok := newSet[k]; !ok { - diffs = append(diffs, primitiveObjectDiff(v, nil, filter, name, contextual)) + for k, oldObj := range oldSet { + newObj := newSet[k] + diff := primitiveObjectDiff(oldObj, newObj, filter, name, contextual) + if diff != nil { + diffs = append(diffs, diff) } } for k, v := range newSet { diff --git a/nomad/structs/diff_test.go b/nomad/structs/diff_test.go index 6410ce4af1d..bfb2f960b22 100644 --- a/nomad/structs/diff_test.go +++ b/nomad/structs/diff_test.go @@ -4459,7 +4459,7 @@ func TestTaskDiff(t *testing.T) { New: &Task{ Artifacts: []*TaskArtifact{ { - GetterSource: "foo", + GetterSource: "foo/bar", GetterOptions: map[string]string{ "foo": "bar", }, @@ -4482,6 +4482,18 @@ func TestTaskDiff(t *testing.T) { Expected: &TaskDiff{ Type: DiffTypeEdited, Objects: []*ObjectDiff{ + { + Type: DiffTypeEdited, + Name: "Artifact", + Fields: []*FieldDiff{ + { + Type: DiffTypeEdited, + Name: "GetterSource", + Old: "foo", + New: "foo/bar", + }, + }, + }, { Type: DiffTypeAdded, Name: "Artifact", @@ -6914,7 +6926,7 @@ func TestTaskDiff(t *testing.T) { { SourcePath: "foo", DestPath: "bar", - EmbeddedTmpl: "baz", + EmbeddedTmpl: "baz new", ChangeMode: "bam", ChangeSignal: "SIGHUP", Splay: 1, @@ -6934,6 +6946,18 @@ func TestTaskDiff(t *testing.T) { Expected: &TaskDiff{ Type: DiffTypeEdited, Objects: []*ObjectDiff{ + { + Type: DiffTypeEdited, + Name: "Template", + Fields: []*FieldDiff{ + { + Type: DiffTypeEdited, + Name: "EmbeddedTmpl", + Old: "baz", + New: "baz new", + }, + }, + }, { Type: DiffTypeAdded, Name: "Template", diff --git a/nomad/structs/structs.go b/nomad/structs/structs.go index a773ba1071c..94b31e1eec9 100644 --- a/nomad/structs/structs.go +++ b/nomad/structs/structs.go @@ -7479,6 +7479,11 @@ func (t *Template) Warnings() error { return mErr.ErrorOrNil() } +// DiffID fulfills the DiffableWithID interface. +func (t *Template) DiffID() string { + return t.DestPath +} + // AllocState records a single event that changes the state of the whole allocation type AllocStateField uint8 @@ -8120,6 +8125,11 @@ func (ta *TaskArtifact) GoString() string { return fmt.Sprintf("%+v", ta) } +// DiffID fulfills the DiffableWithID interface. +func (ta *TaskArtifact) DiffID() string { + return ta.RelativeDest +} + // hashStringMap appends a deterministic hash of m onto h. func hashStringMap(h hash.Hash, m map[string]string) { keys := make([]string, 0, len(m)) From f6a22779ba400ce63b71054c019ab5b01000dac5 Mon Sep 17 00:00:00 2001 From: Luiz Aoqui Date: Wed, 27 Oct 2021 19:18:22 -0400 Subject: [PATCH 2/3] changelog: add entry for #11400 --- .changelog/11400.txt | 3 +++ 1 file changed, 3 insertions(+) create mode 100644 .changelog/11400.txt diff --git a/.changelog/11400.txt b/.changelog/11400.txt new file mode 100644 index 00000000000..15557ab411e --- /dev/null +++ b/.changelog/11400.txt @@ -0,0 +1,3 @@ +```release-note:improvement +cli: Improve `nomad job plan` output for `artifact` and `template` changes +``` From e0c7bb585ecc179228a1a6e81d6866c5482fa489 Mon Sep 17 00:00:00 2001 From: Luiz Aoqui Date: Thu, 28 Oct 2021 11:08:08 -0400 Subject: [PATCH 3/3] improve godocs for DiffableWithID interface --- nomad/structs/diff.go | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/nomad/structs/diff.go b/nomad/structs/diff.go index d3b2406da52..d0f9e5bb227 100644 --- a/nomad/structs/diff.go +++ b/nomad/structs/diff.go @@ -10,8 +10,8 @@ import ( "github.com/mitchellh/hashstructure" ) -// DiffableWithID defines an object that have an unique and stable field that -// can be used as an identifier when generating a diff. +// DiffableWithID defines an object that has a unique and stable value that can +// be used as an identifier when generating a diff. type DiffableWithID interface { // DiffID returns the value to use to match entities between the old and // the new input.