kubes 0.3.2 → 0.4.1
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/CHANGELOG.md +29 -0
- data/README.md +6 -5
- data/docs/_config.yml +1 -1
- data/docs/_docs/ci/cloudbuild.md +2 -0
- data/docs/_docs/config/args.md +10 -0
- data/docs/_docs/config/args/docker.md +19 -0
- data/docs/_docs/config/args/kubectl.md +19 -0
- data/docs/_docs/config/docker.md +4 -40
- data/docs/_docs/config/env.md +1 -1
- data/docs/_docs/config/hooks.md +10 -0
- data/docs/_docs/config/hooks/docker.md +70 -0
- data/docs/_docs/config/hooks/kubectl.md +83 -0
- data/docs/_docs/config/hooks/kubes.md +67 -0
- data/docs/_docs/config/hooks/ruby.md +74 -0
- data/docs/_docs/config/kubectl.md +3 -54
- data/docs/_docs/config/reference.md +20 -0
- data/docs/_docs/config/skip.md +58 -0
- data/docs/_docs/dsl/resources.md +1 -1
- data/docs/_docs/dsl/resources/backend_config.md +1 -1
- data/docs/_docs/intro.md +6 -3
- data/docs/_docs/learn/dsl/review-project.md +4 -2
- data/docs/_docs/learn/yaml/review-project.md +4 -2
- data/docs/_docs/{auto-context.md → misc/auto-context.md} +0 -0
- data/docs/_docs/{kustomize.md → misc/kustomize.md} +0 -0
- data/docs/_docs/misc/separate-steps.md +21 -0
- data/docs/_docs/patterns/migrations.md +121 -0
- data/docs/_includes/commands.html +2 -0
- data/docs/_includes/config/hooks/options.md +20 -0
- data/docs/_includes/sidebar.html +32 -10
- data/docs/_reference/kubes-exec.md +14 -6
- data/docs/_reference/kubes-init.md +1 -0
- data/docs/_reference/kubes-logs.md +1 -0
- data/docs/_sass/theme.scss +25 -1
- data/kubes.gemspec +3 -0
- data/lib/kubes.rb +3 -0
- data/lib/kubes/cli.rb +25 -6
- data/lib/kubes/cli/apply.rb +2 -1
- data/lib/kubes/cli/base.rb +11 -0
- data/lib/kubes/cli/compile.rb +8 -0
- data/lib/kubes/cli/delete.rb +1 -1
- data/lib/kubes/cli/exec.rb +37 -6
- data/lib/kubes/cli/get.rb +3 -2
- data/lib/kubes/cli/init.rb +39 -2
- data/lib/kubes/cli/logs.rb +50 -3
- data/lib/kubes/cli/prune.rb +95 -0
- data/lib/kubes/compiler.rb +11 -7
- data/lib/kubes/compiler/decorator/base.rb +7 -1
- data/lib/kubes/compiler/decorator/{resources/secret.rb → hashable.rb} +5 -4
- data/lib/kubes/compiler/decorator/hashable/field.rb +53 -0
- data/lib/kubes/compiler/decorator/hashable/storage.rb +19 -0
- data/lib/kubes/compiler/decorator/post.rb +77 -0
- data/lib/kubes/compiler/decorator/pre.rb +12 -0
- data/lib/kubes/compiler/strategy.rb +2 -2
- data/lib/kubes/compiler/strategy/base.rb +1 -1
- data/lib/kubes/compiler/strategy/result.rb +4 -6
- data/lib/kubes/config.rb +16 -11
- data/lib/kubes/docker/strategy/build/docker.rb +1 -1
- data/lib/kubes/docker/strategy/build/gcloud.rb +1 -1
- data/lib/kubes/docker/strategy/image_name.rb +1 -1
- data/lib/kubes/docker/strategy/push/docker.rb +1 -1
- data/lib/kubes/docker/strategy/push/gcloud.rb +1 -1
- data/lib/kubes/docker/strategy/utils.rb +1 -1
- data/lib/kubes/hooks/builder.rb +29 -15
- data/lib/kubes/hooks/concern.rb +10 -0
- data/lib/kubes/hooks/dsl.rb +2 -1
- data/lib/kubes/hooks/runner.rb +22 -0
- data/lib/kubes/kubectl.rb +21 -18
- data/lib/kubes/kubectl/batch.rb +8 -5
- data/lib/kubes/kubectl/{decider.rb → dispatcher.rb} +1 -1
- data/lib/kubes/kubectl/fetch/base.rb +13 -10
- data/lib/kubes/kubectl/fetch/deployment.rb +21 -11
- data/lib/kubes/kubectl/fetch/pods.rb +4 -15
- data/lib/kubes/kubectl/kustomize.rb +1 -1
- data/lib/kubes/kubectl/ordering.rb +12 -0
- data/lib/kubes/util/consider.rb +2 -1
- data/lib/kubes/util/sh.rb +1 -1
- data/lib/kubes/version.rb +1 -1
- data/lib/templates/dsl/.kubes/resources/base/all.rb.tt +6 -1
- data/lib/templates/dsl/.kubes/resources/shared/namespace.rb.tt +1 -1
- data/lib/templates/dsl/.kubes/resources/web/deployment.rb +1 -1
- data/lib/templates/yaml/.kubes/resources/base/all.yaml.tt +1 -1
- data/lib/templates/yaml/.kubes/resources/shared/namespace.yaml.tt +1 -1
- data/lib/templates/yaml/.kubes/resources/web/deployment.yaml.tt +1 -1
- data/spec/fixtures/decorators/deployment/both/envFrom.yaml +31 -0
- data/spec/fixtures/decorators/deployment/both/valueFrom.yaml +33 -0
- data/spec/fixtures/decorators/deployment/both/volumes.yaml +40 -0
- data/spec/fixtures/prune/capture.yaml +57 -0
- data/spec/fixtures/prune/fetch_items.yaml +268 -0
- data/spec/kubes/cli/prune_spec.rb +38 -0
- data/spec/kubes/compiler/decorator/{resources → post}/deployment_spec.rb +52 -6
- data/spec/kubes/compiler/decorator/{resources → post}/pod_spec.rb +2 -11
- metadata +56 -19
- data/lib/kubes/compiler/decorator.rb +0 -17
- data/lib/kubes/compiler/decorator/compile.rb +0 -12
- data/lib/kubes/compiler/decorator/resources/base.rb +0 -13
- data/lib/kubes/compiler/decorator/resources/container.rb +0 -76
- data/lib/kubes/compiler/decorator/resources/container/mapping.rb +0 -28
- data/lib/kubes/compiler/decorator/resources/deployment.rb +0 -10
- data/lib/kubes/compiler/decorator/resources/pod.rb +0 -10
- data/lib/kubes/compiler/decorator/write.rb +0 -14
- data/lib/kubes/docker/strategy/hooks.rb +0 -9
data/lib/kubes/compiler.rb
CHANGED
@@ -1,5 +1,6 @@
|
|
1
1
|
module Kubes
|
2
2
|
class Compiler
|
3
|
+
include Kubes::Hooks::Concern
|
3
4
|
include Kubes::Logging
|
4
5
|
include Kubes::Util::Consider
|
5
6
|
|
@@ -8,13 +9,16 @@ module Kubes
|
|
8
9
|
end
|
9
10
|
|
10
11
|
def run
|
11
|
-
|
12
|
-
|
13
|
-
|
14
|
-
|
12
|
+
Kubes.config # trigger config load. So can set ENV['VAR'] in config/envs/dev.rb etc
|
13
|
+
run_hooks("kubes.rb", name: "compile") do
|
14
|
+
results = resources.map do |path|
|
15
|
+
strategy = Strategy.new(@options.merge(path: path))
|
16
|
+
strategy.compile
|
17
|
+
end.compact
|
15
18
|
|
16
|
-
|
17
|
-
|
19
|
+
results.each do |result|
|
20
|
+
write(result)
|
21
|
+
end
|
18
22
|
end
|
19
23
|
|
20
24
|
puts "Compiled .kubes/resources files to .kubes/output" if show_compiled_message?
|
@@ -49,7 +53,7 @@ module Kubes
|
|
49
53
|
end
|
50
54
|
|
51
55
|
def write(result)
|
52
|
-
result.
|
56
|
+
result.decorate!(:post)
|
53
57
|
filename, content = result.filename, result.content
|
54
58
|
dest = "#{Kubes.root}/.kubes/output/#{filename}"
|
55
59
|
|
@@ -1,14 +1,20 @@
|
|
1
1
|
module Kubes::Compiler::Decorator
|
2
2
|
class Base
|
3
|
+
attr_reader :data
|
3
4
|
def initialize(data)
|
4
5
|
@data = data
|
5
6
|
end
|
6
7
|
|
8
|
+
def run
|
9
|
+
return @data unless Kubes.config.suffix_hash
|
10
|
+
process
|
11
|
+
end
|
12
|
+
|
7
13
|
def result
|
8
14
|
if @data.is_a?(Kubes::Compiler::Dsl::Core::Blocks)
|
9
15
|
@data.results.each { |k,v| process(v) } # returns nil
|
10
16
|
else
|
11
|
-
process
|
17
|
+
process # processes and returns @data
|
12
18
|
end
|
13
19
|
@data # important to return @data so we keep the original @data structure: Blocks or Hash
|
14
20
|
end
|
@@ -1,17 +1,18 @@
|
|
1
1
|
require 'digest'
|
2
2
|
|
3
|
-
module Kubes::Compiler::Decorator
|
4
|
-
class
|
3
|
+
module Kubes::Compiler::Decorator
|
4
|
+
class Hashable < Base
|
5
5
|
include Kubes::Compiler::Util::YamlDump
|
6
6
|
|
7
|
-
def
|
7
|
+
def store
|
8
8
|
# even though name is required, will allow logic to get the kubectl apply and kubectl to surface the required name error
|
9
9
|
name = @data.dig('metadata','name')
|
10
10
|
return @data unless name
|
11
11
|
|
12
|
+
# puts "name #{name}" # TODO: scope Kind so Secret and ConfigMap can have the same name...
|
12
13
|
md5 = md5(@data)
|
13
14
|
@data['metadata']['name'] = "#{name}-#{md5}"
|
14
|
-
|
15
|
+
Storage.store(@data['kind'], name, md5)
|
15
16
|
@data
|
16
17
|
end
|
17
18
|
|
@@ -0,0 +1,53 @@
|
|
1
|
+
class Kubes::Compiler::Decorator::Hashable
|
2
|
+
class Field
|
3
|
+
# item is full wrapper structure
|
4
|
+
#
|
5
|
+
# secretRef: <--- wrapper
|
6
|
+
# name: demo-secret
|
7
|
+
#
|
8
|
+
def initialize(item)
|
9
|
+
@item = item
|
10
|
+
end
|
11
|
+
|
12
|
+
def hashable?
|
13
|
+
x = @item.keys & map.keys
|
14
|
+
!x.empty?
|
15
|
+
end
|
16
|
+
|
17
|
+
def kind
|
18
|
+
wrapper =~ /configMap/ ? "ConfigMap" : "Secret"
|
19
|
+
end
|
20
|
+
|
21
|
+
# The key of the hashable value.
|
22
|
+
#
|
23
|
+
# envFrom:
|
24
|
+
# - secretRef:
|
25
|
+
# name: demo-secret <--- wrapper is 'name'
|
26
|
+
#
|
27
|
+
def key
|
28
|
+
map[wrapper]
|
29
|
+
end
|
30
|
+
|
31
|
+
# The wrapper field is nested right above the item with the hashable value.
|
32
|
+
#
|
33
|
+
# envFrom:
|
34
|
+
# - secretRef: <--- wrapper
|
35
|
+
# name: demo-secret
|
36
|
+
#
|
37
|
+
def wrapper
|
38
|
+
@item.keys.first
|
39
|
+
end
|
40
|
+
|
41
|
+
# wrapper element to key that stores the hashable value
|
42
|
+
def map
|
43
|
+
{
|
44
|
+
'configMapRef' => 'name',
|
45
|
+
'configMapKeyRef' => 'name',
|
46
|
+
'configMap' => 'name',
|
47
|
+
'secretRef' => 'name',
|
48
|
+
'secretKeyRef' => 'name',
|
49
|
+
'secret' => 'secretName',
|
50
|
+
}
|
51
|
+
end
|
52
|
+
end
|
53
|
+
end
|
@@ -0,0 +1,19 @@
|
|
1
|
+
class Kubes::Compiler::Decorator::Hashable
|
2
|
+
module Storage
|
3
|
+
@@md5s = {}
|
4
|
+
def store(kind, name, md5)
|
5
|
+
@@md5s[kind] ||= {}
|
6
|
+
@@md5s[kind][name] = md5
|
7
|
+
end
|
8
|
+
|
9
|
+
def fetch(kind, name)
|
10
|
+
@@md5s[kind] ||= {}
|
11
|
+
@@md5s[kind][name]
|
12
|
+
end
|
13
|
+
|
14
|
+
def md5s
|
15
|
+
@@md5s
|
16
|
+
end
|
17
|
+
extend self
|
18
|
+
end
|
19
|
+
end
|
@@ -0,0 +1,77 @@
|
|
1
|
+
module Kubes::Compiler::Decorator
|
2
|
+
class Post < Base
|
3
|
+
def process
|
4
|
+
add_hash(@data)
|
5
|
+
clean_namespace
|
6
|
+
@data
|
7
|
+
end
|
8
|
+
|
9
|
+
def clean_namespace
|
10
|
+
return unless @data['kind'] == 'Namespace'
|
11
|
+
@data['metadata'].delete('namespace')
|
12
|
+
@data
|
13
|
+
end
|
14
|
+
|
15
|
+
def add_hash(item, options={})
|
16
|
+
# hashable set from previous stack call
|
17
|
+
if options[:hashable_field] && item.is_a?(Hash)
|
18
|
+
field = options[:hashable_field]
|
19
|
+
value_without_md5 = item[field.key]
|
20
|
+
@reset_hashable_field = true unless value_without_md5
|
21
|
+
if field.hashable? && value_without_md5
|
22
|
+
md5 = Hashable::Storage.fetch(field.kind, value_without_md5)
|
23
|
+
v = [value_without_md5, md5].compact.join('-')
|
24
|
+
item[field.key] = v
|
25
|
+
end
|
26
|
+
end
|
27
|
+
|
28
|
+
options[:hashable_field] ||= hashable_field(item) # set for next stack call
|
29
|
+
# Pretty tricky case. Given:
|
30
|
+
#
|
31
|
+
# envFrom:
|
32
|
+
# - secretRef:
|
33
|
+
# name: demo-secret
|
34
|
+
# - configMapRef:
|
35
|
+
# name: demo-config-map
|
36
|
+
#
|
37
|
+
# Need to reset the stored hashable_field in the call stack.
|
38
|
+
# Else the field.kind is cached and the md5 look is incorrect
|
39
|
+
# The spec/fixtures/decorators/deployment/both/envFrom.yaml fixture covers this.
|
40
|
+
if @reset_hashable_field
|
41
|
+
options[:hashable_field] = hashable_field(item)
|
42
|
+
@reset_hashable_field = false
|
43
|
+
end
|
44
|
+
case item
|
45
|
+
when Array, Hash
|
46
|
+
item.each { |i| add_hash(i, options) }
|
47
|
+
end
|
48
|
+
item
|
49
|
+
end
|
50
|
+
|
51
|
+
# Returns the nested key name that will be hashable. Examples:
|
52
|
+
#
|
53
|
+
# 1. envFrom example
|
54
|
+
# envFrom:
|
55
|
+
# - secretRef:
|
56
|
+
# name: demo-secret
|
57
|
+
#
|
58
|
+
# 2. valueFrom example
|
59
|
+
# valueFrom:
|
60
|
+
# secretKeyRef:
|
61
|
+
# name: demo-secret
|
62
|
+
# key: password
|
63
|
+
#
|
64
|
+
# 3. volumes example
|
65
|
+
# volumes:
|
66
|
+
# - secret:
|
67
|
+
# secretName: demo-secret
|
68
|
+
#
|
69
|
+
# This is useful to capture for the next level of the stack call
|
70
|
+
#
|
71
|
+
def hashable_field(item)
|
72
|
+
return false unless item.is_a?(Hash)
|
73
|
+
field = Hashable::Field.new(item)
|
74
|
+
field if field.hashable?
|
75
|
+
end
|
76
|
+
end
|
77
|
+
end
|
@@ -11,7 +11,7 @@ class Kubes::Compiler
|
|
11
11
|
|
12
12
|
strategy = klass.new(@options.merge(path: @path)) # Dsl or Erb
|
13
13
|
result = strategy.run
|
14
|
-
result.
|
14
|
+
result.decorate!(:pre) # compile pre phase decoration
|
15
15
|
result
|
16
16
|
end
|
17
17
|
|
@@ -19,7 +19,7 @@ class Kubes::Compiler
|
|
19
19
|
ext = File.extname(@path)
|
20
20
|
case ext
|
21
21
|
when '.rb' then Dsl
|
22
|
-
when '.yaml' then Erb
|
22
|
+
when '.yaml','.yml' then Erb
|
23
23
|
else Pass
|
24
24
|
end
|
25
25
|
end
|
@@ -7,7 +7,7 @@ class Kubes::Compiler::Strategy
|
|
7
7
|
@path = options[:path]
|
8
8
|
|
9
9
|
@filename = @path.sub(%r{.*\.kubes/resources/},'') # IE: web/deployment.rb or web/deployment.yaml
|
10
|
-
@save_file = @filename.sub('.rb','.yaml')
|
10
|
+
@save_file = @filename.sub('.yml','.yaml').sub('.rb','.yaml')
|
11
11
|
end
|
12
12
|
end
|
13
13
|
end
|
@@ -11,12 +11,10 @@ class Kubes::Compiler::Strategy
|
|
11
11
|
@data.respond_to?(:read)
|
12
12
|
end
|
13
13
|
|
14
|
-
|
15
|
-
|
16
|
-
|
17
|
-
|
18
|
-
def write_decorate!
|
19
|
-
@data = Kubes::Compiler::Decorator::Write.new(@data).result
|
14
|
+
# decorate(:pre) or decorate(:post)
|
15
|
+
def decorate!(phase)
|
16
|
+
klass = "Kubes::Compiler::Decorator::#{phase.to_s.camelize}".constantize
|
17
|
+
@data = klass.new(@data).result
|
20
18
|
end
|
21
19
|
|
22
20
|
def content
|
data/lib/kubes/config.rb
CHANGED
@@ -11,31 +11,36 @@ module Kubes
|
|
11
11
|
def defaults
|
12
12
|
config = ActiveSupport::OrderedOptions.new
|
13
13
|
|
14
|
-
config.
|
15
|
-
config.state.docker_image_path = "#{Kubes.root}/.kubes/state/docker_image.txt"
|
14
|
+
config.auto_prune = true
|
16
15
|
|
17
|
-
config.
|
18
|
-
config.logger.level = ENV['KUBES_LOG_LEVEL'] || :info
|
16
|
+
config.builder = "docker" # IE: docker or gcloud
|
19
17
|
|
20
18
|
# Auto-switching options
|
21
19
|
config.kubectl = ActiveSupport::OrderedOptions.new
|
22
20
|
config.kubectl.context = nil
|
23
21
|
config.kubectl.context_keep = true # after switching context keep it
|
24
|
-
config.kubectl.exit_on_fail = nil # whether or not continue if the kubectl command fails
|
25
22
|
|
26
|
-
|
27
|
-
config.kubectl.
|
28
|
-
|
23
|
+
# whether or not continue if the kubectl command fails
|
24
|
+
config.kubectl.exit_on_fail = ActiveSupport::OrderedOptions.new
|
25
|
+
config.kubectl.exit_on_fail.apply = true # whether or not continue if the kubectl apply command fails
|
26
|
+
config.kubectl.exit_on_fail.delete = false # whether or not continue if the kubectl delete command fails
|
27
|
+
# Note: delete is a internal method to ActiveSupport::OrderedOptions so will have to access it with ['...']
|
29
28
|
|
30
29
|
config.kubectl.order = ActiveSupport::OrderedOptions.new
|
31
30
|
config.kubectl.order.roles = role_order
|
32
31
|
config.kubectl.order.kinds = kind_order
|
33
32
|
|
34
|
-
config.suffix_hash = true # append suffix has to ConfigMap and Secret
|
35
|
-
|
36
33
|
config.repo = nil # expected to be set by .kubes/config.rb
|
37
34
|
|
38
|
-
config.
|
35
|
+
config.logger = Logger.new($stdout)
|
36
|
+
config.logger.level = ENV['KUBES_LOG_LEVEL'] || :info
|
37
|
+
|
38
|
+
config.skip = []
|
39
|
+
|
40
|
+
config.state = ActiveSupport::OrderedOptions.new
|
41
|
+
config.state.docker_image_path = "#{Kubes.root}/.kubes/state/docker_image.txt"
|
42
|
+
|
43
|
+
config.suffix_hash = true # append suffix hash to ConfigMap and Secret
|
39
44
|
|
40
45
|
config
|
41
46
|
end
|
@@ -61,7 +61,7 @@ module Kubes::Docker::Strategy
|
|
61
61
|
end
|
62
62
|
|
63
63
|
def custom
|
64
|
-
custom = Kubes::Args::Custom.new(@name, "#{Kubes.root}/.kubes/config/docker
|
64
|
+
custom = Kubes::Args::Custom.new(@name, "#{Kubes.root}/.kubes/config/args/docker.rb")
|
65
65
|
custom.build
|
66
66
|
custom
|
67
67
|
end
|
data/lib/kubes/hooks/builder.rb
CHANGED
@@ -4,39 +4,53 @@ module Kubes::Hooks
|
|
4
4
|
include Dsl
|
5
5
|
include DslEvaluator
|
6
6
|
include Kubes::Logging
|
7
|
-
include Kubes::Util::Sh
|
8
7
|
|
9
8
|
attr_accessor :name
|
10
|
-
def initialize(
|
11
|
-
@
|
12
|
-
@
|
9
|
+
def initialize(dsl_file, options={})
|
10
|
+
@dsl_file, @options = dsl_file, options # IE: .kubes/config/hooks/kubectl.rb
|
11
|
+
@output_file = options[:file] # IE: .kubes/output/web/service.yaml
|
12
|
+
@name = options[:name].to_s
|
13
13
|
@hooks = {before: {}, after: {}}
|
14
14
|
end
|
15
15
|
|
16
16
|
def build
|
17
|
-
return @hooks unless File.exist?(@
|
18
|
-
evaluate_file(@
|
17
|
+
return @hooks unless File.exist?(@dsl_file)
|
18
|
+
evaluate_file(@dsl_file)
|
19
19
|
@hooks.deep_stringify_keys!
|
20
20
|
end
|
21
21
|
memoize :build
|
22
22
|
|
23
23
|
def run_hooks
|
24
24
|
build
|
25
|
-
|
25
|
+
run_each_hook("before")
|
26
26
|
out = yield if block_given?
|
27
|
-
|
27
|
+
run_each_hook("after")
|
28
28
|
out
|
29
29
|
end
|
30
30
|
|
31
|
-
def
|
32
|
-
|
33
|
-
|
31
|
+
def run_each_hook(type)
|
32
|
+
hooks = @hooks.dig(type, @name.to_s) || []
|
33
|
+
hooks.each do |hook|
|
34
|
+
run_hook(type, hook)
|
35
|
+
end
|
36
|
+
end
|
37
|
+
|
38
|
+
def run_hook(type, hook)
|
39
|
+
return unless run?(hook)
|
34
40
|
|
35
|
-
|
36
|
-
|
41
|
+
command = File.basename(@dsl_file).sub('.rb','') # IE: kubes, kubectl, docker
|
42
|
+
id = "#{command} #{type} #{@name}"
|
43
|
+
on = " on: #{hook["on"]}" if hook["on"]
|
44
|
+
label = " label: #{hook["label"]}" if hook["label"]
|
45
|
+
logger.info "Running #{id} hook.#{on}#{label}"
|
46
|
+
logger.debug "Hook options: #{hook}"
|
47
|
+
Runner.new(hook).run
|
48
|
+
end
|
37
49
|
|
38
|
-
|
39
|
-
|
50
|
+
def run?(hook)
|
51
|
+
return false unless hook["execute"]
|
52
|
+
return true unless hook["on"]
|
53
|
+
@output_file.include?(hook["on"])
|
40
54
|
end
|
41
55
|
end
|
42
56
|
end
|