bcome 1.3.4 → 2.0.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (183) hide show
  1. checksums.yaml +5 -5
  2. data/bin/bcome +13 -8
  3. data/lib/bcome.rb +14 -11
  4. data/lib/objects/bcome/version.rb +19 -1
  5. data/lib/objects/bootup.rb +16 -7
  6. data/lib/objects/command/local.rb +2 -0
  7. data/lib/objects/config_factory.rb +3 -0
  8. data/lib/objects/driver/base.rb +52 -6
  9. data/lib/objects/driver/bucket.rb +6 -4
  10. data/lib/objects/driver/ec2.rb +45 -5
  11. data/lib/objects/driver/gcp.rb +168 -0
  12. data/lib/objects/driver/gcp/authentication/api_key.rb +6 -0
  13. data/lib/objects/driver/gcp/authentication/base.rb +36 -0
  14. data/lib/objects/driver/gcp/authentication/oauth.rb +96 -0
  15. data/lib/objects/driver/gcp/authentication/oauth_client_config.rb +22 -0
  16. data/lib/objects/driver/gcp/authentication/oauth_session_store.rb +22 -0
  17. data/lib/objects/driver/gcp/authentication/service_account.rb +62 -0
  18. data/lib/objects/driver/gcp/authentication/signet/service_account.rb +27 -0
  19. data/lib/objects/driver/gcp/authentication/utilities.rb +42 -0
  20. data/lib/objects/encryptor.rb +109 -24
  21. data/lib/objects/exception/argument_error_invoking_method_from_command_line.rb +8 -4
  22. data/lib/objects/exception/base.rb +21 -10
  23. data/lib/objects/exception/can_only_subselect_on_inventory.rb +8 -4
  24. data/lib/objects/exception/cannot_authenticate_to_gcp.rb +11 -0
  25. data/lib/objects/exception/cannot_find_internal_registry_klass.rb +8 -4
  26. data/lib/objects/exception/cannot_find_inventory.rb +11 -0
  27. data/lib/objects/exception/cannot_find_subselection_parent.rb +8 -4
  28. data/lib/objects/exception/cant_find_key_in_cloud_tags.rb +8 -4
  29. data/lib/objects/exception/cant_find_key_in_metadata.rb +8 -4
  30. data/lib/objects/exception/cant_find_proxy_host_by_identifier.rb +8 -4
  31. data/lib/objects/exception/cant_find_proxy_host_by_namespace.rb +8 -4
  32. data/lib/objects/exception/could_not_initiate_ssh_connection.rb +8 -4
  33. data/lib/objects/exception/could_not_initiate_ssh_connection_through_backend_proxy.rb +8 -4
  34. data/lib/objects/exception/could_not_retrieve_terraform_output.rb +11 -0
  35. data/lib/objects/exception/deprecation_warning.rb +9 -7
  36. data/lib/objects/exception/duplicate_command_line_argument_key.rb +8 -4
  37. data/lib/objects/exception/ec2_driver_missing_authorization_keys.rb +11 -0
  38. data/lib/objects/exception/ec2_driver_missing_provisioning_region.rb +8 -4
  39. data/lib/objects/exception/empty_namespace_tree.rb +11 -0
  40. data/lib/objects/exception/failed_to_run_local_command.rb +8 -4
  41. data/lib/objects/exception/gcp_auth_service_account_missing_credentials.rb +11 -0
  42. data/lib/objects/exception/generic.rb +11 -0
  43. data/lib/objects/exception/interactive_session_halt.rb +6 -2
  44. data/lib/objects/exception/invalid_bcome_breadcrumb.rb +8 -4
  45. data/lib/objects/exception/invalid_breadcrumb.rb +8 -4
  46. data/lib/objects/exception/invalid_context_command.rb +8 -4
  47. data/lib/objects/exception/invalid_gcp_authentication_scheme.rb +11 -0
  48. data/lib/objects/exception/invalid_identifier.rb +8 -4
  49. data/lib/objects/exception/invalid_machines_cache_config.rb +8 -4
  50. data/lib/objects/exception/invalid_matcher_query.rb +8 -4
  51. data/lib/objects/exception/invalid_meta_data_config.rb +8 -4
  52. data/lib/objects/exception/invalid_metadata_encryption_key.rb +8 -4
  53. data/lib/objects/exception/invalid_network_config.rb +8 -4
  54. data/lib/objects/exception/invalid_network_driver_type.rb +8 -4
  55. data/lib/objects/exception/invalid_port_forward_request.rb +11 -0
  56. data/lib/objects/exception/invalid_proxy_config.rb +8 -4
  57. data/lib/objects/exception/invalid_regexp_matcher_in_registry.rb +8 -4
  58. data/lib/objects/exception/invalid_registry_arguments_type.rb +8 -4
  59. data/lib/objects/exception/invalid_registry_command_name_length.rb +8 -4
  60. data/lib/objects/exception/invalid_registry_data_config.rb +8 -4
  61. data/lib/objects/exception/invalid_restriction_key_in_registry.rb +8 -4
  62. data/lib/objects/exception/invalid_ssh_config.rb +8 -4
  63. data/lib/objects/exception/inventories_cannot_have_subviews.rb +8 -4
  64. data/lib/objects/exception/malformed_command_line_arguments.rb +8 -4
  65. data/lib/objects/exception/method_invocation_requires_parameter.rb +8 -4
  66. data/lib/objects/exception/method_name_conflict_in_registry.rb +8 -4
  67. data/lib/objects/exception/missing_argument_for_registry_command.rb +8 -4
  68. data/lib/objects/exception/missing_description_on_view.rb +8 -4
  69. data/lib/objects/exception/missing_execute_on_registry_object.rb +8 -4
  70. data/lib/objects/exception/missing_gcp_authentication_scheme.rb +11 -0
  71. data/lib/objects/exception/missing_gcp_service_account_credentials_filename.rb +11 -0
  72. data/lib/objects/exception/missing_gcp_service_scopes.rb +11 -0
  73. data/lib/objects/exception/missing_identifier_on_view.rb +8 -4
  74. data/lib/objects/exception/missing_inventory_contributors.rb +11 -0
  75. data/lib/objects/exception/missing_ip_address_on_server.rb +8 -4
  76. data/lib/objects/exception/missing_network_config.rb +8 -4
  77. data/lib/objects/exception/missing_or_invalid_client_secrets.rb +11 -0
  78. data/lib/objects/exception/missing_params_for_rsync.rb +8 -4
  79. data/lib/objects/exception/missing_params_for_scp.rb +8 -4
  80. data/lib/objects/exception/missing_subselection_key.rb +8 -4
  81. data/lib/objects/exception/missing_type_on_view.rb +8 -4
  82. data/lib/objects/exception/no_node_found_for_breadcrumb.rb +8 -4
  83. data/lib/objects/exception/no_node_named_by_identifier.rb +8 -4
  84. data/lib/objects/exception/node_identifiers_must_be_unique.rb +8 -4
  85. data/lib/objects/exception/orchestration_script_does_not_exist.rb +8 -4
  86. data/lib/objects/exception/proxy_host_node_does_not_have_public_ip_address.rb +8 -4
  87. data/lib/objects/exception/unknown_dynamic_server_type.rb +11 -0
  88. data/lib/objects/exception/unknown_method_for_namespace.rb +8 -4
  89. data/lib/objects/exception/user_orchestration_error.rb +11 -0
  90. data/lib/objects/initialization/factory.rb +36 -0
  91. data/lib/objects/initialization/structure.rb +18 -0
  92. data/lib/objects/initialization/utils.rb +20 -0
  93. data/lib/objects/interactive/session.rb +4 -1
  94. data/lib/objects/interactive/session_item/base.rb +2 -0
  95. data/lib/objects/interactive/session_item/capture_input.rb +2 -0
  96. data/lib/objects/interactive/session_item/transparent_ssh.rb +29 -23
  97. data/lib/objects/loading_bar/handler.rb +80 -0
  98. data/lib/objects/loading_bar/indicator/base.rb +65 -0
  99. data/lib/objects/loading_bar/indicator/basic.rb +34 -0
  100. data/lib/objects/loading_bar/indicator/progress.rb +26 -0
  101. data/lib/objects/loading_bar/pid_bucket.rb +27 -0
  102. data/lib/objects/modules/context.rb +13 -9
  103. data/lib/objects/modules/draw.rb +49 -0
  104. data/lib/objects/modules/registry_management.rb +16 -10
  105. data/lib/objects/modules/tree.rb +157 -0
  106. data/lib/objects/modules/ui_output.rb +10 -6
  107. data/lib/objects/modules/workspace_commands.rb +131 -157
  108. data/lib/objects/modules/workspace_menu.rb +193 -129
  109. data/lib/objects/node/attributes.rb +17 -19
  110. data/lib/objects/node/base.rb +136 -74
  111. data/lib/objects/node/cache_handler.rb +3 -1
  112. data/lib/objects/node/collection.rb +10 -9
  113. data/lib/objects/node/factory.rb +47 -36
  114. data/lib/objects/node/inventory/base.rb +106 -100
  115. data/lib/objects/node/inventory/defined.rb +113 -89
  116. data/lib/objects/node/inventory/merge.rb +51 -0
  117. data/lib/objects/node/inventory/subselect.rb +66 -46
  118. data/lib/objects/node/kube/base.rb +51 -0
  119. data/lib/objects/node/kube/container.rb +9 -0
  120. data/lib/objects/node/kube/estate.rb +19 -0
  121. data/lib/objects/node/kube/namespace.rb +24 -0
  122. data/lib/objects/node/kube/pod.rb +24 -0
  123. data/lib/objects/node/kube_wrap.rb +26 -0
  124. data/lib/objects/node/meta/base.rb +8 -1
  125. data/lib/objects/node/meta/cloud.rb +2 -0
  126. data/lib/objects/node/meta/local.rb +2 -0
  127. data/lib/objects/node/meta_data_factory.rb +4 -2
  128. data/lib/objects/node/meta_data_loader.rb +28 -29
  129. data/lib/objects/node/resources/base.rb +5 -1
  130. data/lib/objects/node/resources/inventory.rb +26 -5
  131. data/lib/objects/node/resources/merged.rb +47 -0
  132. data/lib/objects/node/resources/sub_inventory.rb +12 -8
  133. data/lib/objects/node/server/base.rb +105 -70
  134. data/lib/objects/node/server/dynamic/base.rb +23 -0
  135. data/lib/objects/node/server/{dynamic.rb → dynamic/ec2.rb} +13 -13
  136. data/lib/objects/node/server/dynamic/gcp.rb +46 -0
  137. data/lib/objects/node/server/static.rb +34 -10
  138. data/lib/objects/orchestration/base.rb +17 -1
  139. data/lib/objects/orchestration/interactive_terraform.rb +59 -30
  140. data/lib/objects/orchestrator.rb +22 -0
  141. data/lib/objects/parser/bread_crumb.rb +3 -1
  142. data/lib/objects/registry/arguments/base.rb +3 -1
  143. data/lib/objects/registry/arguments/command_line.rb +6 -1
  144. data/lib/objects/registry/arguments/console.rb +4 -1
  145. data/lib/objects/registry/command/base.rb +3 -0
  146. data/lib/objects/registry/command/external.rb +9 -3
  147. data/lib/objects/registry/command/group.rb +11 -4
  148. data/lib/objects/registry/command/internal.rb +3 -1
  149. data/lib/objects/registry/command/shortcut.rb +17 -9
  150. data/lib/objects/registry/command_list.rb +2 -0
  151. data/lib/objects/registry/loader.rb +13 -10
  152. data/lib/objects/ssh/bootstrap.rb +3 -1
  153. data/lib/objects/ssh/command.rb +9 -8
  154. data/lib/objects/ssh/command_exec.rb +16 -10
  155. data/lib/objects/ssh/connection_wrangler.rb +124 -0
  156. data/lib/objects/ssh/connector.rb +108 -0
  157. data/lib/objects/ssh/driver.rb +28 -242
  158. data/lib/objects/ssh/driver_concerns/command_strings.rb +17 -0
  159. data/lib/objects/ssh/driver_concerns/connection.rb +75 -0
  160. data/lib/objects/ssh/driver_concerns/functions.rb +89 -0
  161. data/lib/objects/ssh/driver_concerns/user.rb +32 -0
  162. data/lib/objects/ssh/proxy_chain.rb +19 -0
  163. data/lib/objects/ssh/proxy_chain_link.rb +26 -0
  164. data/lib/objects/ssh/proxy_hop.rb +130 -0
  165. data/lib/objects/ssh/script_exec.rb +12 -11
  166. data/lib/objects/ssh/tunnel/local_port_forward.rb +5 -6
  167. data/lib/objects/ssh/tunnel_keeper.rb +21 -0
  168. data/lib/objects/ssh/window.rb +31 -0
  169. data/lib/objects/startup.rb +58 -0
  170. data/lib/objects/system/local.rb +3 -0
  171. data/lib/objects/terraform/output.rb +45 -0
  172. data/lib/objects/workspace.rb +13 -14
  173. data/patches/irb.rb +63 -6
  174. data/patches/string-encrypt.rb +20 -23
  175. data/patches/string.rb +18 -1
  176. data/patches/string_stylesheet.rb +2 -0
  177. metadata +157 -33
  178. data/lib/objects/driver/static.rb +0 -4
  179. data/lib/objects/progress_bar.rb +0 -30
  180. data/lib/objects/ssh/connection_handler.rb +0 -101
  181. data/lib/objects/ssh/proxy_data.rb +0 -56
  182. data/lib/objects/terraform/parser.rb +0 -23
  183. data/lib/objects/terraform/state.rb +0 -40
@@ -1,3 +1,5 @@
1
+ # frozen_string_literal: true
2
+
1
3
  module Bcome::Node
2
4
  class CacheHandler
3
5
  def initialize(inventory_node)
@@ -11,7 +13,7 @@ module Bcome::Node
11
13
  end
12
14
 
13
15
  def write_to_in_memory_cache!(nodes)
14
- @inventory_node.views[:load_machines_from_cache] = true
16
+ # @inventory_node.views[:load_machines_from_cache] = true
15
17
  static_server_data = dynamic_nodes_to_cache_hash(nodes)
16
18
  @inventory_node.views[:static_servers] = static_server_data
17
19
  nil
@@ -1,9 +1,7 @@
1
+ # frozen_string_literal: true
2
+
1
3
  module Bcome::Node
2
4
  class Collection < ::Bcome::Node::Base
3
- def self.to_s
4
- 'collection'
5
- end
6
-
7
5
  def inventories
8
6
  inv = []
9
7
  @resources.active.each do |r|
@@ -28,20 +26,23 @@ module Bcome::Node
28
26
  filtered_set
29
27
  end
30
28
 
31
- def machines
29
+ def machines(skip_for_hidden = true)
32
30
  set = []
33
- @resources.active.each do |resource|
31
+
32
+ resources = skip_for_hidden ? @resources.active.reject(&:hide?) : @resources.active
33
+
34
+ resources.each do |resource|
34
35
  if resource.inventory?
35
36
  resource.load_nodes unless resource.nodes_loaded?
36
37
  set << resource.resources.active
37
38
  else
38
- set << resource.machines
39
+ set << resource.machines(skip_for_hidden)
39
40
  end
40
41
  end
41
42
 
42
43
  set.flatten!
43
-
44
- filter_duplicates(set)
44
+ filtered_machines = filter_duplicates(set)
45
+ filtered_machines
45
46
  end
46
47
 
47
48
  def collection?
@@ -1,30 +1,36 @@
1
+ # frozen_string_literal: true
2
+
1
3
  module Bcome::Node
2
4
  class Factory
3
5
  include Singleton
4
6
 
5
7
  attr_reader :estate
6
8
 
7
- CONFIG_PATH = 'bcome'.freeze
8
- DEFAULT_CONFIG_NAME = 'networks.yml'.freeze
9
- SERVER_OVERRIDE_CONFIG_NAME = 'machines-data.yml'.freeze
10
- LOCAL_OVERRIDE_CONFIG_NAME = 'me.yml'.freeze
9
+ CONFIG_PATH = 'bcome'
10
+ DEFAULT_CONFIG_NAME = 'networks.yml'
11
+ SERVER_OVERRIDE_CONFIG_NAME = 'server-overrides.yml'
12
+ LOCAL_OVERRIDE_CONFIG_NAME = 'me.yml'
13
+
14
+ INVENTORY_KEY = 'inventory'
15
+ COLLECTION_KEY = 'collection'
16
+ SUBSELECT_KEY = 'inventory-subselect'
17
+ MERGE_KEY = 'inventory-merge'
18
+ KUBE_CLUSTER = 'kube-cluster'
11
19
 
12
- INVENTORY_KEY = 'inventory'.freeze
13
- COLLECTION_KEY = 'collection'.freeze
14
- SUBSELECT_KEY = 'inventory-subselect'.freeze
15
- BCOME_RC_FILENAME = '.bcomerc'.freeze
20
+ BCOME_RC_FILENAME = '.bcomerc'
16
21
 
17
22
  def bucket
18
23
  @bucket ||= {}
19
24
  end
20
25
 
21
26
  def init_tree
27
+ raise ::Bcome::Exception::EmptyNamespaceTree, "no namespaces found in #{config_path}.\n\nPlease refer to the documentation for assistance at https://docs.bcome.com." if estate_config.nil?
22
28
  @estate = create_node(estate_config)
23
29
  @estate
24
30
  end
25
31
 
26
32
  def config_path
27
- "#{CONFIG_PATH}/#{config_file_name}"
33
+ ENV['CONF'] || "#{CONFIG_PATH}/#{config_file_name}"
28
34
  end
29
35
 
30
36
  def machines_data_path
@@ -32,7 +38,7 @@ module Bcome::Node
32
38
  end
33
39
 
34
40
  def config_file_name
35
- @config_file_name ||= ENV['CONF'] ? ENV['CONF'] : DEFAULT_CONFIG_NAME
41
+ @config_file_name || DEFAULT_CONFIG_NAME
36
42
  end
37
43
 
38
44
  def create_tree(context_node, views)
@@ -50,36 +56,39 @@ module Bcome::Node
50
56
  end
51
57
 
52
58
  def create_node(config, parent = nil)
53
- raise Bcome::Exception::InvalidNetworkConfig, 'missing config type' unless config[:type]
59
+ raise Bcome::Exception::InvalidNetworkConfig, "missing config type for config #{config}" unless config[:type]
54
60
 
55
61
  klass = klass_for_view_type[config[:type]]
56
-
62
+
57
63
  raise Bcome::Exception::InvalidNetworkConfig, "invalid config type #{config[:type]}" unless klass
58
64
 
59
65
  node = klass.new(views: config, parent: parent)
60
- create_tree(node, config[:views]) if config[:views] && config[:views].any?
66
+ create_tree(node, config[:views]) if config[:views]&.any?
61
67
  parent.resources << node if parent
62
68
 
69
+ # Load inventory resources as early as possible
70
+ if node.is_a?(Bcome::Node::Inventory::Base)
71
+ node.load_nodes unless node.nodes_loaded?
72
+ end
73
+
63
74
  bucket[node.keyed_namespace] = node
64
75
 
65
76
  node
66
77
  end
67
78
 
68
79
  def validate_view(breadcrumb, data)
69
- unless data && data[:type]
70
- raise Bcome::Exception::InvalidNetworkConfig, "Missing namespace type for for namespace '#{breadcrumb}'"
71
- end
80
+ raise Bcome::Exception::InvalidNetworkConfig, "Missing namespace type for namespace '#{breadcrumb}'" unless data && data[:type]
72
81
 
73
- unless is_valid_view_type?(data[:type])
74
- raise Bcome::Exception::InvalidNetworkConfig, "Invalid View Type '#{data[:type]}' for namespace '#{breadcrumb}'. Expecting View Type to be one of: #{klass_for_view_type.keys.join(', ')}"
75
- end
82
+ raise Bcome::Exception::InvalidNetworkConfig, "Invalid View Type '#{data[:type]}' for namespace '#{breadcrumb}'. Expecting View Type to be one of: #{klass_for_view_type.keys.join(', ')}" unless is_valid_view_type?(data[:type])
76
83
  end
77
84
 
78
85
  def klass_for_view_type
79
86
  {
80
87
  COLLECTION_KEY => ::Bcome::Node::Collection,
81
88
  INVENTORY_KEY => ::Bcome::Node::Inventory::Defined,
82
- SUBSELECT_KEY => ::Bcome::Node::Inventory::Subselect
89
+ SUBSELECT_KEY => ::Bcome::Node::Inventory::Subselect,
90
+ MERGE_KEY => ::Bcome::Node::Inventory::Merge,
91
+ KUBE_CLUSTER => ::Bcome::Node::Kube::Estate
83
92
  }
84
93
  end
85
94
 
@@ -96,53 +105,55 @@ module Bcome::Node
96
105
  end
97
106
 
98
107
  def machines_data_for_namespace(namespace)
99
- machines_data[namespace] ? machines_data[namespace] : {}
100
- end
101
-
102
- def rewrite_estate_config(data)
103
- File.open(config_path, 'w') do |file|
104
- file.write data.to_yaml
105
- end
108
+ machines_data[namespace] || {}
106
109
  end
107
110
 
108
111
  def load_estate_config
109
112
  config = YAML.load_file(config_path).deep_symbolize_keys
110
- return config
113
+ config.deep_merge(local_data)
111
114
  rescue ArgumentError, Psych::SyntaxError => e
112
115
  raise Bcome::Exception::InvalidNetworkConfig, 'Invalid yaml in network config' + e.message
113
116
  rescue Errno::ENOENT
114
117
  raise Bcome::Exception::DeprecationWarning if is_running_deprecated_configs?
118
+
115
119
  raise Bcome::Exception::MissingNetworkConfig, config_path
116
120
  end
117
121
 
118
122
  def load_machines_data
119
- return {} unless File.exist?(machines_data_path)
123
+ return {} unless File.exist?(machines_data_path)
124
+
120
125
  config = YAML.load_file(machines_data_path).deep_symbolize_keys
121
- return config
126
+ config
122
127
  rescue ArgumentError, Psych::SyntaxError => e
123
128
  raise Bcome::Exception::InvalidNetworkConfig, 'Invalid yaml in machines data config' + e.message
124
129
  end
125
130
 
126
131
  def local_data
127
132
  @local_data ||= load_local_data
128
- end
133
+ end
129
134
 
130
135
  def local_data_path
131
- "#{CONFIG_PATH}/#{LOCAL_OVERRIDE_CONFIG_NAME}"
136
+ ENV['ME'] || "#{CONFIG_PATH}/#{LOCAL_OVERRIDE_CONFIG_NAME}"
132
137
  end
133
138
 
134
139
  def load_local_data
135
140
  return {} unless File.exist?(local_data_path)
136
- config = YAML.load_file(local_data_path)
141
+
142
+ begin
143
+ config = YAML.load_file(local_data_path).deep_symbolize_keys
144
+ rescue StandardError => e
145
+ raise ::Bcome::Exception::Generic, "Error parsing configuration file #{local_data_path}"
146
+ end
147
+
137
148
  return {} if config.nil?
138
- return config
149
+
150
+ config
139
151
  rescue ArgumentError, Psych::SyntaxError => e
140
152
  raise Bcome::Exception::InvalidNetworkConfig, 'Invalid yaml in machines data config' + e.message
141
153
  end
142
154
 
143
155
  def is_running_deprecated_configs?
144
- File.exist?("bcome/config/platform.yml")
156
+ File.exist?('bcome/config/platform.yml')
145
157
  end
146
-
147
158
  end
148
159
  end
@@ -1,104 +1,110 @@
1
- module Bcome::Node::Inventory
2
- class Base < ::Bcome::Node::Base
3
-
4
- def initialize(*params)
5
- super
6
- raise Bcome::Exception::InventoriesCannotHaveSubViews, @views if @views[:views] && !@views[:views].empty?
7
-
8
- @bootstrap = false
9
- end
10
-
11
- def meta_matches(matchers)
12
- data_wrapper = :metadata
13
- matches_for(data_wrapper, matchers)
14
- end
15
-
16
- def cloud_matches(matchers)
17
- data_wrapper = :cloud_tags
18
- matches_for(data_wrapper, matchers)
19
- end
20
-
21
- def machine_by_identifier(identifier)
22
- resources.active.select { |machine| machine.identifier == identifier }.first
23
- end
24
-
25
- def matches_for(data_wrapper, matchers)
26
- resources.active.select do |machine|
27
- machine.send(data_wrapper).has_key_and_value?(matchers)
28
- end
29
- end
30
-
31
- def toggle_bootstrap(set_to = (@bootstrap ? false : true))
32
- resources.active.each do |machine|
33
- machine.toggle_bootstrap(set_to)
34
- end
35
- @bootstrap = (@bootstrap ? false : true)
36
- return
37
- end
38
-
39
- def enabled_menu_items
40
- super + %i[ssh]
41
- end
42
-
43
- def menu_items
44
- base_items = super.dup
45
- base_items[:ssh] = {
46
- description: 'ssh directly into a resource',
47
- usage: 'ssh identifier',
48
- console_only: true
49
- }
50
-
51
- base_items
52
- end
53
-
54
- def resources
55
- @resources ||= ::Bcome::Node::Resources::Inventory.new
56
- end
57
-
58
- def ssh(identifier = nil)
59
- direct_invoke_server(:ssh, identifier)
60
- end
61
-
62
- def tags(identifier = nil)
63
- identifier.nil? ? direct_invoke_all_servers(:tags) : direct_invoke_server(:tags, identifier)
64
- end
65
-
66
- def direct_invoke_server(method, identifier)
67
- unless identifier
68
- puts "\nPlease provide a machine identifier, e.g. #{method} machinename\n".warning unless identifier
69
- return
70
- end
71
-
72
- if resource = resources.for_identifier(identifier)
73
- resource.send(method)
74
- else
75
- raise Bcome::Exception::InvalidBreadcrumb, "Cannot find a node named '#{identifier}'"
1
+ # frozen_string_literal: true
2
+
3
+ module Bcome
4
+ module Node
5
+ module Inventory
6
+ class Base < ::Bcome::Node::Base
7
+ def initialize(*params)
8
+ super
9
+ raise Bcome::Exception::InventoriesCannotHaveSubViews, @views if @views[:views] && !@views[:views].empty?
10
+ end
11
+
12
+ def meta_matches(matchers)
13
+ data_wrapper = :metadata
14
+ matches_for(data_wrapper, matchers)
15
+ end
16
+
17
+ def cloud_matches(matchers)
18
+ data_wrapper = :cloud_tags
19
+ matches_for(data_wrapper, matchers)
20
+ end
21
+
22
+ def machine_by_identifier(identifier)
23
+ resources.active.select { |machine| machine.identifier == identifier }.first
24
+ end
25
+
26
+ def matches_for(data_wrapper, matchers)
27
+ resources.active.select do |machine|
28
+ machine.send(data_wrapper).has_key_and_value?(matchers)
29
+ end
30
+ end
31
+
32
+ def enabled_menu_items
33
+ super + %i[ssh tags]
34
+ end
35
+
36
+ def menu_items
37
+ base_items = super.dup
38
+ base_items[:ssh] = {
39
+ description: 'ssh directly into a resource',
40
+ usage: 'ssh identifier',
41
+ console_only: true,
42
+ group: :ssh
43
+ }
44
+
45
+ base_items[:tags] = {
46
+ description: 'print out server tags/labels',
47
+ group: :informational
48
+ }
49
+
50
+ base_items
51
+ end
52
+
53
+ def resources
54
+ @resources ||= ::Bcome::Node::Resources::Inventory.new(self)
55
+ end
56
+
57
+ def ssh(identifier = nil)
58
+ direct_invoke_server(:ssh, identifier)
59
+ end
60
+
61
+ def tags(identifier = nil)
62
+ identifier.nil? ? direct_invoke_all_servers(:tags) : direct_invoke_server(:tags, identifier)
63
+ end
64
+
65
+ def direct_invoke_server(method, identifier)
66
+ # If we only have a single resource in our inventory, then just allow direct invocation
67
+ if resources.size == 1
68
+ resource = resources.first
69
+ else
70
+ # Otherwise, we expect to find the resource by its identifier
71
+ unless identifier
72
+ puts "\nPlease provide a machine identifier, e.g. #{method} machinename\n".warning unless identifier
73
+ return
74
+ end
75
+
76
+ resource = resources.for_identifier(identifier)
77
+ raise Bcome::Exception::InvalidBreadcrumb, "Cannot find a node named '#{identifier}'" unless resource
78
+ end
79
+
80
+ resource.send(method)
81
+ end
82
+
83
+ def direct_invoke_all_servers(method)
84
+ resources.active.each { |m| m.send(method) }
85
+ nil
86
+ end
87
+
88
+ def cache_nodes_in_memory
89
+ @cache_handler.do_cache_nodes!
90
+ end
91
+
92
+ def list_key
93
+ :server
94
+ end
95
+
96
+ def machines(skip_for_hidden = true)
97
+ skip_for_hidden ? resources.active : resources.active.reject(&:hide?)
98
+ end
99
+
100
+ def inventory?
101
+ true
102
+ end
103
+
104
+ def override_server_identifier?
105
+ respond_to?(:override_identifier) && !override_identifier.nil?
106
+ end
76
107
  end
77
108
  end
78
-
79
- def direct_invoke_all_servers(method)
80
- resources.active.each {|m| m.send(method) }
81
- return
82
- end
83
-
84
- def cache_nodes_in_memory
85
- @cache_handler.do_cache_nodes!
86
- end
87
-
88
- def list_key
89
- :server
90
- end
91
-
92
- def machines
93
- @resources.active
94
- end
95
-
96
- def inventory?
97
- true
98
- end
99
-
100
- def override_server_identifier?
101
- !@override_identifier.nil?
102
- end
103
109
  end
104
110
  end
@@ -1,117 +1,141 @@
1
- module Bcome::Node::Inventory
2
- class Defined < ::Bcome::Node::Inventory::Base
3
- MACHINES_CACHE_PATH = 'machines-cache.yml'.freeze
1
+ # frozen_string_literal: true
4
2
 
5
- def self.to_s
6
- 'inventory'
7
- end
3
+ module Bcome
4
+ module Node
5
+ module Inventory
6
+ class Defined < ::Bcome::Node::Inventory::Base
7
+ include ::Bcome::LoadingBar::Handler
8
8
 
9
- attr_reader :dynamic_nodes_loaded
9
+ MACHINES_CACHE_PATH = 'static-cache.yml'
10
10
 
11
- def initialize(*params)
12
- @load_machines_from_cache = false
13
- @cache_handler = ::Bcome::Node::CacheHandler.new(self)
14
- super
15
- end
11
+ attr_reader :dynamic_nodes_loaded
16
12
 
17
- def enabled_menu_items
18
- super + %i[save reload]
19
- end
13
+ def initialize(*params)
14
+ @load_machines_from_cache = false
15
+ @cache_handler = ::Bcome::Node::CacheHandler.new(self)
16
+ super
17
+ end
20
18
 
21
- def menu_items
22
- base_items = super.dup
19
+ def enabled_menu_items
20
+ super + %i[cache reload]
21
+ end
23
22
 
24
- base_items[:reload] = {
25
- description: "Restock this inventory from remote (hit 'save' after to persist)",
26
- console_only: true
27
- }
28
- base_items
29
- end
23
+ def menu_items
24
+ base_items = super.dup
25
+
26
+ base_items[:cache] = {
27
+ description: 'Cache the current tree state',
28
+ console_only: false,
29
+ group: :miscellany
30
+ }
31
+
32
+ base_items[:reload] = {
33
+ description: "Restock this inventory from remote (hit 'cache' after to persist)",
34
+ console_only: true,
35
+ group: :miscellany
36
+ }
37
+ base_items
38
+ end
30
39
 
31
- def reload
32
- resources.reset_duplicate_nodes!
33
- do_reload
34
- puts "\nDone. Hit 'ls' to see the refreshed inventory.\n".informational
35
- end
40
+ def reload
41
+ resources.reset_duplicate_nodes!
42
+ do_reload
43
+ puts "\n\nDone. Hit 'ls' to see the refreshed inventory.\n".informational
44
+ end
36
45
 
37
- def set_static_servers
38
- if raw_static_machines_from_cache
39
- raw_static_machines_from_cache.each do |server_config|
40
- resources << ::Bcome::Node::Server::Static.new(views: server_config, parent: self)
46
+ def set_static_servers
47
+ cached_machines = raw_static_machines_from_cache
48
+
49
+ if cached_machines&.any?
50
+ print "\n"
51
+ title = 'Loading' + "\sCACHE".bc_orange.bold + "\s" + namespace.to_s.underline
52
+ wrap_indicator type: :basic, title: title, completed_title: '' do
53
+ cached_machines.each do |server_config|
54
+ resources << ::Bcome::Node::Server::Static.new(views: server_config, parent: self)
55
+ end
56
+ signal_success
57
+ end
58
+ end
41
59
  end
42
- end
43
- end
44
60
 
45
- def raw_static_machines_from_cache
46
- load_machines_config[namespace.to_sym]
47
- end
61
+ def raw_static_machines_from_cache
62
+ load_machines_config[namespace.to_sym]
63
+ end
48
64
 
49
- def machines_cache_path
50
- "#{::Bcome::Node::Factory::CONFIG_PATH}/#{MACHINES_CACHE_PATH}"
51
- end
65
+ def machines_cache_path
66
+ "#{::Bcome::Node::Factory::CONFIG_PATH}/#{MACHINES_CACHE_PATH}"
67
+ end
52
68
 
53
- def mark_as_cached!
54
- data = ::Bcome::Node::Factory.instance.load_estate_config
55
- data[namespace.to_sym][:load_machines_from_cache] = true
56
- ::Bcome::Node::Factory.instance.rewrite_estate_config(data)
57
- end
69
+ def cache
70
+ @answer = ::Bcome::Interactive::Session.run(self,
71
+ :capture_input, terminal_prompt: 'Are you sure you want to cache these machines (saving will overwrite any previous selections) [Y|N] ? ')
72
+
73
+ if @answer && @answer == 'Y'
74
+ cache_nodes_in_memory
75
+ data = load_machines_config
58
76
 
59
- def save
60
- @answer = ::Bcome::Interactive::Session.run(self,
61
- :capture_input, terminal_prompt: "\nAre you sure you want to cache these machines (saving will overwrite any previous selections) [Y|N] ? ")
77
+ data.delete(namespace)
78
+ data.delete(namespace.to_sym)
62
79
 
63
- if @answer && @answer == 'Y'
64
- cache_nodes_in_memory
65
- data = load_machines_config
66
- data[namespace] = views[:static_servers]
80
+ data[namespace] = views[:static_servers]
67
81
 
68
- File.open(machines_cache_path, 'w') do |file|
69
- file.write data.to_yaml
82
+ File.open(machines_cache_path, 'w') do |file|
83
+ file.write data.to_yaml
84
+ end
85
+ puts "\nMachines have been cached to #{machines_cache_path} for node #{namespace}".informational
86
+ else
87
+ puts 'Nothing saved'.warning
88
+ end
70
89
  end
71
- mark_as_cached!
72
- puts "Machines have been cached for node #{namespace}".informational
73
- else
74
- puts 'Nothing saved'.warning
75
- end
76
- end
77
90
 
78
- def load_machines_config
79
- config = YAML.load_file(machines_cache_path).deep_symbolize_keys
80
- return config
81
- rescue ArgumentError, Psych::SyntaxError
82
- raise Bcome::Exception::InvalidMachinesCacheConfig, 'Invalid yaml in config'
83
- rescue Errno::ENOENT
84
- return {}
85
- end
91
+ def load_machines_config
92
+ config = YAML.load_file(machines_cache_path).deep_symbolize_keys
93
+ config
94
+ rescue ArgumentError, Psych::SyntaxError
95
+ raise Bcome::Exception::InvalidMachinesCacheConfig, 'Invalid yaml in config'
96
+ rescue Errno::ENOENT
97
+ {}
98
+ end
86
99
 
87
- def cache_nodes_in_memory
88
- @cache_handler.do_cache_nodes!
89
- end
100
+ def cache_nodes_in_memory
101
+ @cache_handler.do_cache_nodes!
102
+ end
90
103
 
91
- def do_reload
92
- resources.unset!
93
- load_dynamic_nodes
94
- end
104
+ def do_reload
105
+ resources.unset!
106
+ load_dynamic_nodes
107
+ end
95
108
 
96
- def load_nodes
97
- set_static_servers
98
- load_dynamic_nodes unless @load_machines_from_cache
99
- end
109
+ def load_nodes
110
+ set_static_servers
111
+ load_dynamic_nodes unless resources.any?
112
+ nodes_loaded!
113
+ end
100
114
 
101
- def load_dynamic_nodes
102
- raw_servers = fetch_server_list
115
+ def load_dynamic_nodes
116
+ raw_servers = fetch_server_list
103
117
 
104
- raw_servers.each do |raw_server|
105
- resources << ::Bcome::Node::Server::Dynamic.new_from_fog_instance(raw_server, self)
106
- end
118
+ raw_servers ||= []
107
119
 
108
- resources.rename_initial_duplicate if resources.should_rename_initial_duplicate?
120
+ raw_servers.each do |raw_server|
121
+ if raw_server.is_a?(Google::Apis::ComputeBeta::Instance)
122
+ resources << ::Bcome::Node::Server::Dynamic::Gcp.new_from_gcp_instance(raw_server, self)
123
+ elsif raw_server.is_a?(Fog::Compute::AWS::Server)
124
+ resources << ::Bcome::Node::Server::Dynamic::Ec2.new_from_fog_instance(raw_server, self)
125
+ else
126
+ raise Bcome::Exception::UnknownDynamicServerType, "Unknown dynamic server type #{raw_server.class}"
127
+ end
128
+ end
109
129
 
110
- end
130
+ resources.rename_initial_duplicate if resources.should_rename_initial_duplicate?
131
+ end
132
+
133
+ def fetch_server_list
134
+ return [] unless network_driver
111
135
 
112
- def fetch_server_list
113
- return [] unless network_driver
114
- network_driver.fetch_server_list(filters)
136
+ network_driver.fetch_server_list(filters)
137
+ end
138
+ end
115
139
  end
116
140
  end
117
141
  end