escli 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (40) hide show
  1. checksums.yaml +7 -0
  2. data/.gitignore +13 -0
  3. data/.travis.yml +5 -0
  4. data/Gemfile +4 -0
  5. data/LICENSE.txt +21 -0
  6. data/README.md +41 -0
  7. data/Rakefile +10 -0
  8. data/bin/escli +5 -0
  9. data/bin/setup +8 -0
  10. data/escli.gemspec +34 -0
  11. data/lib/canzea/cli/canzea.rb +323 -0
  12. data/lib/canzea/cli/escli.rb +207 -0
  13. data/lib/canzea/commands/add-env.rb +64 -0
  14. data/lib/canzea/commands/apply-config.rb +50 -0
  15. data/lib/canzea/commands/config-git-commit.rb +78 -0
  16. data/lib/canzea/commands/ecosystem/ecosystem.rb +66 -0
  17. data/lib/canzea/commands/ecosystem/resources.rb +82 -0
  18. data/lib/canzea/commands/gen-user.rb +22 -0
  19. data/lib/canzea/commands/get-catalog.rb +46 -0
  20. data/lib/canzea/commands/login.rb +50 -0
  21. data/lib/canzea/commands/prepare-plan.rb +82 -0
  22. data/lib/canzea/commands/push-config.rb +283 -0
  23. data/lib/canzea/commands/register-metadata.rb +79 -0
  24. data/lib/canzea/commands/remote-bootstrap.rb +38 -0
  25. data/lib/canzea/commands/remote-run.rb +37 -0
  26. data/lib/canzea/commands/update-config.rb +26 -0
  27. data/lib/canzea/config.rb +35 -0
  28. data/lib/canzea/core/audit.rb +86 -0
  29. data/lib/canzea/core/prepare-environment.rb +194 -0
  30. data/lib/canzea/core/registry.rb +191 -0
  31. data/lib/canzea/core/ssh-base-cmd-class.rb +103 -0
  32. data/lib/canzea/core/template-runner.rb +41 -0
  33. data/lib/canzea/core/trace-component.rb +171 -0
  34. data/lib/canzea/core/trace-runner.rb +108 -0
  35. data/lib/canzea/environment.rb +6 -0
  36. data/lib/canzea/helper-run-class.rb +89 -0
  37. data/lib/canzea/plan-step-class.rb +210 -0
  38. data/lib/canzea/registry.rb +12 -0
  39. data/lib/canzea/version.rb +3 -0
  40. metadata +201 -0
@@ -0,0 +1,82 @@
1
+
2
+ =begin
3
+
4
+ curl -v -X GET -H "Content-Type: application/json" https://canzea.com/console-app/api/api/v2/saasexpress/ecosystem
5
+
6
+ Canzea::config[:canzea_platform_uri] + "/api/v2/saasexpress/ecosystem"
7
+
8
+ escli ecosystem resource test.yaml intg-0
9
+
10
+ escli resource plus test.yaml prod-1
11
+
12
+ escli resource apply f67c70b0-2706-48b5-ac4b-c76eeb72effe
13
+
14
+ =end
15
+
16
+ require 'net/http'
17
+ require 'json'
18
+ require 'canzea/commands/login'
19
+
20
+ class Resources
21
+ def plus(manifest, segment)
22
+ token = Login.new.get()
23
+
24
+ esFile = Canzea::config[:config_location] + '/.scope'
25
+ es = JSON.parse(File.read(esFile))
26
+
27
+ puts "Using #{es['id']}"
28
+
29
+ uri = URI(Canzea::config[:canzea_platform_uri] + "/api/v2/saasexpress/buildingblocks/#{es['id']}/buildingblocks/configure")
30
+
31
+ config = File.read(manifest)
32
+
33
+ req = Net::HTTP::Post.new(uri, 'Authorization' => "Bearer #{token}", 'Content-Type' => 'application/json')
34
+ req.body = {config: config, segment: segment, resources: []}.to_json
35
+
36
+ https = Net::HTTP.new(uri.hostname,uri.port)
37
+ https.use_ssl = uri.instance_of? URI::HTTPS
38
+
39
+ res = https.request(req)
40
+
41
+ case res
42
+ when Net::HTTPSuccess, Net::HTTPRedirection
43
+ es = JSON.parse(res.body)
44
+ puts JSON.pretty_generate(es)
45
+ puts "Run 'escli resource apply #{es['id']}'"
46
+ else
47
+ puts "Error #{res}"
48
+ puts res.body
49
+ end
50
+
51
+ end
52
+
53
+ def apply(resourceID)
54
+ token = Login.new.get()
55
+
56
+ esFile = Canzea::config[:config_location] + '/.scope'
57
+ es = JSON.parse(File.read(esFile))
58
+
59
+ uri = URI(Canzea::config[:canzea_platform_uri] + "/api/v2/saasexpress/resources/plus/#{es['id']}/#{resourceID}")
60
+
61
+ puts "Calling #{uri}"
62
+
63
+ req = Net::HTTP::Post.new(uri, 'Authorization' => "Bearer #{token}", 'Content-Type' => 'application/json')
64
+ req.body = {}.to_json
65
+
66
+ https = Net::HTTP.new(uri.hostname,uri.port)
67
+ https.use_ssl = uri.instance_of? URI::HTTPS
68
+
69
+ res = https.request(req)
70
+
71
+ case res
72
+ when Net::HTTPSuccess, Net::HTTPRedirection
73
+ es = JSON.parse(res.body)
74
+ puts JSON.pretty_generate(es)
75
+ puts "Successfully scheduled. Batch reference # #{es['ref']}."
76
+ else
77
+ puts "Error #{res}"
78
+ puts res.body
79
+ end
80
+
81
+ end
82
+ end
@@ -0,0 +1,22 @@
1
+ require "git"
2
+ require "fileutils"
3
+ require "pathname"
4
+ require "canzea/helper-run-class"
5
+
6
+ class GenUser
7
+ def do(name)
8
+ puts "Generating #{name}"
9
+
10
+ home = "/home/#{name}"
11
+ key = "general"
12
+
13
+ plans = [
14
+ {:solution => "users", :action => "add-user", :parameters => { "name" => name }},
15
+ {:solution => "users", :action => "new-key", :parameters => { "name" => name, "home" => home, "key" => key }},
16
+ ]
17
+
18
+ helper = HelperRun.new(false)
19
+ helper.runPlan(plans)
20
+
21
+ end
22
+ end
@@ -0,0 +1,46 @@
1
+ require "git"
2
+ require "fileutils"
3
+ require "logger"
4
+ require "pathname"
5
+
6
+
7
+ class GetCatalog
8
+ def initialize ()
9
+ @log = Logger.new(Canzea::config[:logging_root] + '/plans.log')
10
+ end
11
+ def do(tag = nil)
12
+ catalogLocation = Canzea::config[:catalog_location]
13
+ FileUtils.rm_rf "#{catalogLocation}"
14
+ FileUtils.mkdir_p "#{catalogLocation}"
15
+ catalogLocation = Pathname.new(catalogLocation).parent().realpath
16
+ g = Git.clone(Canzea::config[:catalog_git], "catalog", :path => catalogLocation)
17
+ if (tag != nil)
18
+ puts "-- (RESET) For tag #{tag}"
19
+ g.checkout(tag)
20
+ else
21
+ g.checkout(g.branch(Canzea::config[:catalog_branch]))
22
+ end
23
+ branch = g.current_branch
24
+ puts "-- (RESET) Branch '#{branch}' Commit #{g.log[0]}"
25
+ puts "-- (RESET) Source #{catalogLocation}/catalog"
26
+ end
27
+
28
+ def state()
29
+ g = Git.open(Canzea::config[:catalog_location])
30
+
31
+ branch = g.current_branch
32
+ commit = g.log[0]
33
+ g.tags.each do | c |
34
+ # Doesn't seem to link tag to the branch so excluding for now
35
+ # ENV.store('CATALOG_TAG', c.name)
36
+ # puts "-- Tag #{c.name}"
37
+ end
38
+
39
+ ENV.store('CATALOG_BRANCH', branch)
40
+ ENV.store('CATALOG_COMMIT', "#{g.log[0]}")
41
+
42
+ # puts "-- Branch '#{ENV['CATALOG_BRANCH']}' Commit #{ENV['CATALOG_COMMIT']}"
43
+
44
+ end
45
+
46
+ end
@@ -0,0 +1,50 @@
1
+
2
+ =begin
3
+ Login should use an ecosystem.
4
+
5
+ curl -v -X POST -H "Content-Type: application/json" -d '{"name":"","password":""}' https://canzea.com/console-app/api/x/user/login
6
+
7
+ Canzea::config[:canzea_platform_uri] + "/api/x/user/login"
8
+ =end
9
+
10
+ require 'net/http'
11
+ require 'json'
12
+
13
+ class Login
14
+ def do(name, password)
15
+ credFile = "#{Dir.home}/.canzearc"
16
+
17
+ uri = URI(Canzea::config[:canzea_platform_uri] + "/api/x/user/login")
18
+
19
+ req = Net::HTTP::Post.new(uri, 'Content-Type' => 'application/json')
20
+ req.body = {name: name, password: password}.to_json
21
+
22
+ https = Net::HTTP.new(uri.hostname,uri.port)
23
+ https.use_ssl = uri.instance_of? URI::HTTPS
24
+
25
+ res = https.request(req)
26
+
27
+ case res
28
+ when Net::HTTPSuccess, Net::HTTPRedirection
29
+ token = JSON.parse(res.body)
30
+
31
+ File.open(credFile, 'w') { |file| file.write(token['token']) }
32
+ puts "Login successful. Authorization cached."
33
+ else
34
+ puts res.body
35
+ end
36
+
37
+ end
38
+
39
+ def logout()
40
+ credFile = "#{Dir.home}/.canzearc"
41
+ File.delete(credFile)
42
+ puts "Authorization cache cleared."
43
+ end
44
+
45
+ def get()
46
+ credFile = "#{Dir.home}/.canzearc"
47
+ token = File.read(credFile)
48
+ return token
49
+ end
50
+ end
@@ -0,0 +1,82 @@
1
+ require 'json'
2
+ require 'base64'
3
+ require 'yaml'
4
+ require 'canzea/config'
5
+ require 'canzea/plan-step-class'
6
+ require 'canzea/commands/remote-run'
7
+ require 'canzea/commands/add-env'
8
+
9
+ class PreparePlan
10
+ def initialize ()
11
+ @log = Logger.new(Canzea::config[:logging_root] + '/plans.log')
12
+ @basePath = "#{Pathname.new(Canzea::config[:catalog_location]).realpath}"
13
+ end
14
+
15
+ def doPatch (paramsString, test, privateKey, serverBase, serverNumber)
16
+
17
+ publicIp = File.read("#{Canzea::config[:pwd]}/vps-#{serverBase}-#{serverNumber}.json")
18
+ publicIp.strip!
19
+
20
+ params = JSON.parse(paramsString)
21
+
22
+ patches = Base64.decode64(params["patches"])
23
+ params['patches'] = JSON.parse(patches)
24
+
25
+ params['patches'].each do | patch |
26
+ RemoteRun.new.doCommand publicIp, privateKey, patch["command"]
27
+ end
28
+ end
29
+
30
+ # Read the blueprint instructions and prepare plan for a particular segment
31
+ def do (blueprint, segment, step, task, test, privateKey, serverBase, serverNumber)
32
+ planStep = PlanStep.new
33
+
34
+ log "Processing configure.json for #{segment} in #{blueprint} from #{@basePath}"
35
+
36
+ instructions = YAML.load_file("#{@basePath}/blueprints/#{blueprint}/instruction.yml")
37
+ segment = instructions['instructions']['segments'][segment]
38
+
39
+ log segment['abbreviation']
40
+
41
+ first = true
42
+ index = 1
43
+ segment['image'].each { |item|
44
+ if item.start_with?("step:")
45
+ parts = item.split(':')
46
+
47
+ if (index < Integer(step))
48
+ log "[#{index.to_s.rjust(2, "0")}] #{item} SKIPPING"
49
+ msg = { "message" => { "task" => "skip", "context" => { "step" => index } } }
50
+ puts msg.to_json
51
+ else
52
+
53
+ log "[#{index.to_s.rjust(2, "0")}] #{item}"
54
+ publicIp = File.read("#{Canzea::config[:pwd]}/vps-#{serverBase}-#{serverNumber}.json")
55
+ publicIp.strip!
56
+ if (test == false)
57
+ if (task == nil)
58
+ RemoteRun.new.do publicIp, privateKey, parts[1], parts[2], index.to_s.rjust(2, "0")
59
+ else
60
+ RemoteRun.new.doTask publicIp, privateKey, parts[1], parts[2], task, index.to_s.rjust(2, "0")
61
+ end
62
+ # Keep track of what we have done; parsing the response and looking at the JSON
63
+ else
64
+ RemoteRun.new.test publicIp, privateKey, parts[1], parts[2], index.to_s.rjust(2, "0")
65
+ end
66
+
67
+ if first == true
68
+ # task is only relevant for the first invocation
69
+ task = nil
70
+ first = false
71
+ end
72
+ end
73
+ index = index + 1
74
+ end
75
+ }
76
+ end
77
+
78
+ def log (msg)
79
+ puts "-- #{msg}"
80
+ @log.info(msg)
81
+ end
82
+ end
@@ -0,0 +1,283 @@
1
+
2
+ require 'git'
3
+ require 'json'
4
+ require 'fileutils'
5
+ require 'logger'
6
+ require 'canzea/config'
7
+
8
+ # canzea --lifecycle=wire --solution=gogs --action=push-content --args='{"file":"metadata.json","path":"test/a/metadata.json", "comment":"V1.0"}'
9
+ # canzea --util=push-config git@165.227.87.135:root/ecosystem.git "Latest Input" --args='{"files":[{"file":"metadata.json","path":"test/a/metadata.json"}]}'
10
+
11
+ # PushConfig.new.write "Test", "nr-global.flow", "new content and more and more"
12
+
13
+ # PushConfig.new.cp "Another bit of file", "nr-global.flow", "node-red/nr-global.flow"
14
+
15
+
16
+ class PushConfig
17
+ def initialize (_relPath = "instances/#{ENV['HOSTNAME']}/")
18
+ @relPath = _relPath
19
+ @log = Logger.new(Canzea::config[:logging_root] + '/plans.log')
20
+ end
21
+
22
+ def cat(filePath)
23
+ write filePath, File.read(filePath)
24
+ end
25
+
26
+ def cp(filePath, destPath)
27
+ if File.directory?(filePath)
28
+ Dir.foreach(filePath) { |x|
29
+ if x == '.' or x == '..' or x.start_with?('.')
30
+ else
31
+ if File.directory?("#{filePath}/#{x}")
32
+ cp "#{filePath}/#{x}", "#{destPath}/#{x}"
33
+ else
34
+ write "#{destPath}/#{x}", File.read("#{filePath}/#{x}")
35
+ end
36
+ end
37
+ }
38
+ else
39
+ write destPath, File.read(filePath)
40
+ end
41
+ end
42
+
43
+ def backupAndRemove(filePath)
44
+ backup(filePath)
45
+ rm(filePath)
46
+ end
47
+
48
+ def backup(filePath)
49
+ path = "ecosystems/#{ENV['ECOSYSTEM']}/#{@relPath}#{filePath}"
50
+
51
+ folder = "sc"
52
+ if ENV.has_key? 'ECOSYSTEM_CONFIG_GIT'
53
+ folder = "_working"
54
+ end
55
+
56
+ if File.exists? folder
57
+ g = Git.init(folder)
58
+ else
59
+ if ENV.has_key? 'ECOSYSTEM_CONFIG_GIT'
60
+ g = Git.clone(ENV['ECOSYSTEM_CONFIG_GIT'], folder, :path => '.')
61
+ else
62
+ g = Git.init(folder)
63
+ end
64
+ end
65
+
66
+ puts "Exists? #{folder}/#{path}"
67
+ if File.exists? "#{folder}/#{path}"
68
+ numb = 1
69
+ bkup = "#{folder}/#{path}.archived"
70
+ while File.exists? "#{bkup}#{numb}"
71
+ numb = numb + 1
72
+ end
73
+ puts "Backed up to #{bkup}#{numb}"
74
+ File.write "#{bkup}#{numb}", File.read("#{folder}/#{path}");
75
+
76
+ g.add("#{path}.archived#{numb}")
77
+ end
78
+ end
79
+
80
+ def write(filePath, contents)
81
+
82
+ path = "ecosystems/#{ENV['ECOSYSTEM']}/#{@relPath}#{filePath}"
83
+
84
+ begin
85
+ folder = "sc"
86
+ if ENV.has_key? 'ECOSYSTEM_CONFIG_GIT'
87
+ folder = "_working"
88
+ end
89
+
90
+
91
+ if File.exists? folder
92
+ g = Git.init(folder)
93
+ else
94
+ if ENV.has_key? 'ECOSYSTEM_CONFIG_GIT'
95
+ g = Git.clone(ENV['ECOSYSTEM_CONFIG_GIT'], folder, :path => '.')
96
+ else
97
+ g = Git.init(folder)
98
+ end
99
+ end
100
+ FileUtils.mkdir_p File.dirname("#{folder}/#{path}")
101
+
102
+ pth = File.expand_path "#{folder}/#{path}"
103
+
104
+ @log.info "Writing to: #{pth}"
105
+
106
+ open("#{folder}/#{path}", 'w') { |f|
107
+ f.puts contents
108
+ }
109
+
110
+ g.add("#{path}")
111
+
112
+ rescue => exception
113
+ @log.error("PushConfig Failed")
114
+ @log.error(exception.to_s)
115
+ @log.error(exception.backtrace)
116
+ abort()
117
+ end
118
+
119
+ end
120
+
121
+ def rm(filePath)
122
+ path = "ecosystems/#{ENV['ECOSYSTEM']}/#{@relPath}#{filePath}"
123
+
124
+ begin
125
+ folder = "sc"
126
+ if ENV.has_key? 'ECOSYSTEM_CONFIG_GIT'
127
+ folder = "_working"
128
+ end
129
+
130
+ if File.exists? folder
131
+ g = Git.init(folder)
132
+ else
133
+ g = Git.clone(ENV['ECOSYSTEM_CONFIG_GIT'], folder, :path => '.')
134
+ end
135
+
136
+ if File.directory? "#{folder}/#{path}"
137
+ puts "DELETE DIRECTORY: #{folder}/#{path}"
138
+ FileUtils.rmdir_p File.dirname("#{folder}/#{path}")
139
+
140
+ g.add("#{path}")
141
+
142
+ else
143
+ if File.exists? "#{folder}/#{path}"
144
+ puts "DELETE FILE: #{folder}/#{path}"
145
+ FileUtils.rm_f "#{folder}/#{path}"
146
+ g.remove("#{path}")
147
+ end
148
+ end
149
+ rescue => exception
150
+ @log.error("PushConfig Failed")
151
+ @log.error(exception.to_s)
152
+ @log.error(exception.backtrace)
153
+ abort()
154
+ end
155
+
156
+ end
157
+
158
+ def commit(comment)
159
+ folder = "sc"
160
+ if ENV.has_key? 'ECOSYSTEM_CONFIG_GIT'
161
+ folder = "_working"
162
+ else
163
+ return
164
+ end
165
+
166
+ begin
167
+ count = 0
168
+
169
+
170
+ g = Git.init(folder)
171
+
172
+ begin
173
+ g.chdir do
174
+ g.status.changed.each do |file|
175
+ count = count +1
176
+ end
177
+ g.status.added.each do |file|
178
+ count = count +1
179
+ end
180
+ g.status.deleted.each do |file|
181
+ count = count +1
182
+ end
183
+ end
184
+ rescue => exception_ignore
185
+ count = 1
186
+ end
187
+
188
+ if count == 0
189
+ log "No changes"
190
+ else
191
+ log "#{count} changes committed."
192
+ g.commit(comment)
193
+
194
+ if ENV.has_key? 'ECOSYSTEM_CONFIG_GIT'
195
+ log "Pushing changes to SCM."
196
+ g.push
197
+ end
198
+ end
199
+
200
+ begin
201
+ if ENV.has_key? 'ECOSYSTEM_CONFIG_GIT'
202
+ log "Removing #{folder}"
203
+ FileUtils.rm_rf(folder)
204
+ end
205
+ rescue => exception
206
+ @log.error("PushConfig Failed")
207
+ @log.error(exception.backtrace)
208
+ @log.error(exception.to_s)
209
+ end
210
+
211
+ rescue => exception
212
+ @log.error("PushConfig Failed")
213
+ @log.error(exception.backtrace)
214
+ @log.error(exception.to_s)
215
+ abort()
216
+ end
217
+ end
218
+
219
+ def do(url, comment, params)
220
+ begin
221
+
222
+ # "git@#{ENV["GOGS_ADDRESS"]}:root/ecosystem.git"
223
+
224
+ folder = "sc"
225
+ if ENV.has_key? 'ECOSYSTEM_CONFIG_GIT'
226
+ folder = "_working"
227
+ end
228
+
229
+ FileUtils.rm_rf(folder)
230
+
231
+ g = Git.clone(url, folder, :path => '.')
232
+
233
+ params['files'].each { | f |
234
+ file = f['file']
235
+
236
+ path = f['path']
237
+
238
+ FileUtils.mkdir_p File.dirname("#{folder}/#{path}")
239
+
240
+ open("#{folder}/#{path}", 'w') { |f|
241
+ f.puts File.read(file)
242
+ }
243
+ g.add("#{path}")
244
+ }
245
+
246
+
247
+ count = 0
248
+ g.chdir do
249
+ g.status.changed.each do |file|
250
+ count = count +1
251
+ end
252
+ g.status.added.each do |file|
253
+ count = count +1
254
+ end
255
+ g.status.deleted.each do |file|
256
+ count = count +1
257
+ end
258
+ end
259
+
260
+
261
+ if count == 0
262
+ log "No changes"
263
+ else
264
+ log "#{count} changes committed."
265
+ g.commit(comment)
266
+
267
+ log "Pushing changes to SCM."
268
+ g.push
269
+ log "Changes pushed"
270
+ end
271
+ rescue => exception
272
+ @log.error("PushConfig Failed")
273
+ @log.error(exception.to_s)
274
+ @log.error(exception.backtrace)
275
+ abort()
276
+ end
277
+ end
278
+
279
+ def log (msg)
280
+ puts msg
281
+ @log.info(msg)
282
+ end
283
+ end