xp5k 0.0.1

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.
data/.gitignore ADDED
@@ -0,0 +1,2 @@
1
+ # Gem package directory
2
+ pkg/
data/Gemfile ADDED
@@ -0,0 +1,3 @@
1
+ source "http://rubygems.org"
2
+
3
+ gemspec
data/Gemfile.lock ADDED
@@ -0,0 +1,38 @@
1
+ PATH
2
+ remote: .
3
+ specs:
4
+ xp5k (0.0.1)
5
+ json (>= 1.5.1)
6
+ restfully (~> 0.6)
7
+ term-ansicolor (>= 1.0.7)
8
+
9
+ GEM
10
+ remote: http://rubygems.org/
11
+ specs:
12
+ addressable (2.3.2)
13
+ backports (2.6.5)
14
+ json (1.7.5)
15
+ mime-types (1.19)
16
+ rack (1.4.1)
17
+ rack-cache (1.2)
18
+ rack (>= 0.4)
19
+ rest-client (1.6.7)
20
+ mime-types (>= 1.16)
21
+ rest-client-components (1.2.0)
22
+ rack (>= 1.0.1)
23
+ rest-client (>= 1.6.0, < 1.7.0)
24
+ restfully (0.8.8)
25
+ addressable
26
+ backports
27
+ json (~> 1.5)
28
+ rack-cache
29
+ rest-client (~> 1.6)
30
+ rest-client-components
31
+ term-ansicolor (1.0.7)
32
+
33
+ PLATFORMS
34
+ ruby
35
+
36
+ DEPENDENCIES
37
+ bundler (>= 1.0.0)
38
+ xp5k!
data/LICENCE ADDED
@@ -0,0 +1,21 @@
1
+ Copyright (c) 2012 Pascal Morillon
2
+ Copyright (c) 2012 Université Rennes 1
3
+
4
+ Permission is hereby granted, free of charge, to any person obtaining
5
+ a copy of this software and associated documentation files (the
6
+ "Software"), to deal in the Software without restriction, including
7
+ without limitation the rights to use, copy, modify, merge, publish,
8
+ distribute, sublicense, and/or sell copies of the Software, and to
9
+ permit persons to whom the Software is furnished to do so, subject to
10
+ the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be
13
+ included in all copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
16
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
17
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
18
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
19
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
20
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
21
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
data/README.md ADDED
@@ -0,0 +1,2 @@
1
+ Work in progress.
2
+
@@ -0,0 +1,42 @@
1
+ module XP5K
2
+ class Config
3
+ # Create a singleton
4
+ @@config = Hash.new
5
+
6
+ @@config[:debug] = false
7
+ @@config[:user] = ENV['USER']
8
+ @@config[:public_key] = File.join(ENV['HOME'], '.ssh', 'id_rsa.pub')
9
+
10
+ @@config[:loaded] = false
11
+
12
+ # Load the experiment configuration file
13
+ def self.load
14
+ if File.exist?(File.expand_path(File.join(Dir.pwd, 'xp.conf')))
15
+ file_path = File.expand_path(File.join(Dir.pwd, 'xp.conf'))
16
+ self.instance_eval(IO.read(file_path),file_path, 1)
17
+ end
18
+ @@config[:loaded] = true
19
+ end # def:: self.load
20
+
21
+ def self.loaded?
22
+ @@config[:loaded]
23
+ end
24
+
25
+ # Configuration options getter
26
+ def self.[](opt)
27
+ @@config[opt.to_sym]
28
+ end # def:: self.[](opt)
29
+
30
+ # Configuration options setter
31
+ def self.[]=(opt, value)
32
+ @@config[opt.to_sym] = value
33
+ end # def:: self.[]=(opt, value)
34
+
35
+ # Using methods in configuration file and transform into hash keys
36
+ def self.method_missing(method_symbol, *args)
37
+ @@config[method_symbol] = args[0]
38
+ @@config[method_symbol]
39
+ end # def:: self.method_missing(method_symbol), *args)
40
+
41
+ end # class:: Config
42
+ end # module:: XP
@@ -0,0 +1,3 @@
1
+ module XP5K
2
+ VERSION='0.0.1'
3
+ end
data/lib/xp5k/xp.rb ADDED
@@ -0,0 +1,137 @@
1
+ require "json"
2
+ require "restfully"
3
+ require "fileutils"
4
+ require "term/ansicolor"
5
+
6
+ module XP5K
7
+ class XP
8
+
9
+ include Term::ANSIColor
10
+
11
+ attr_accessor :jobs, :jobs2submit, :deployments, :todeploy, :connection
12
+ attr_reader :starttime
13
+
14
+ def initialize(options = {})
15
+ @jobs = []
16
+ @jobs2submit = []
17
+ @deployments = []
18
+ @todeploy = []
19
+ @starttime = Time.now
20
+ @logger = options[:logger] || Logger.new(STDOUT)
21
+ XP5K::Config.load unless XP5K::Config.loaded?
22
+
23
+ @connection = Restfully::Session.new(
24
+ :configuration_file => "~/.restfully/api.grid5000.fr.yml",
25
+ :logger => begin
26
+ tmplogger = ::Logger.new(STDERR)
27
+ tmplogger.level = ::Logger::WARN
28
+ tmplogger
29
+ end
30
+ )
31
+ end
32
+
33
+ def timer
34
+ Time.now - self.starttime
35
+ end
36
+
37
+ def define_deployment(deployment_hash)
38
+ self.todeploy << deployment_hash
39
+ end
40
+
41
+ def deploy
42
+ self.todeploy.each do |x|
43
+ x[:nodes] ||= []
44
+ # Get assigned resources to deploy
45
+ x[:jobs].each do |jobname|
46
+ job = self.job_with_name(jobname)
47
+ x[:nodes] += job["assigned_nodes"]
48
+ end
49
+ deployment = @connection.root.sites[x[:site].to_sym].deployments.submit(x)
50
+ self.deployments << { :uid => deployment["uid"], :site => deployment["site_uid"] }
51
+ update_cache
52
+ puts "Waiting for the deployment ##{deployment['uid']} to be terminated..."
53
+ while deployment.reload['status'] == 'processing'
54
+ print(".")
55
+ sleep 10
56
+ end
57
+ print(" [#{green("OK")}]\n")
58
+ update_cache
59
+ end
60
+ end
61
+
62
+ def define_job(job_hash)
63
+ self.jobs2submit << job_hash
64
+
65
+ if File.exists?(".xp_cache")
66
+ # Reload job
67
+ datas = JSON.parse(File.read(".xp_cache"))
68
+ uid = datas["jobs"].select { |x| x["name"] == job_hash[:name] }.first["uid"]
69
+ unless uid.nil?
70
+ job = @connection.root.sites[job_hash[:site].to_sym].jobs["#{uid}".to_sym]
71
+ unless (job.nil? or job["state"] != "running")
72
+ j = job.reload
73
+ self.jobs << j
74
+ end
75
+ end
76
+ end
77
+ end
78
+
79
+ def submit
80
+ self.jobs2submit.each do |job2submit|
81
+ job = self.job_with_name(job2submit[:name])
82
+ if job.nil?
83
+ job = @connection.root.sites[job2submit[:site].to_sym].jobs.submit(job2submit)
84
+ self.jobs << job
85
+ update_cache
86
+ logger.info "Waiting for the job #{job["name"]} ##{job['uid']} to be running at #{job2submit[:site]}..."
87
+ while job.reload["state"] != "running"
88
+ print(".")
89
+ sleep 3
90
+ end
91
+ print(" [#{green("OK")}]\n")
92
+ update_cache
93
+ end
94
+ end
95
+ end
96
+
97
+ def job_with_name(name)
98
+ self.jobs.select { |x| x["name"] == name }.first
99
+ end
100
+
101
+ def status
102
+ # self.jobs2submit.each do |job2submit|
103
+ # job = self.job_with_name(job2submit[:name])
104
+
105
+ # end
106
+ self.jobs.each do |job|
107
+ logger.info "Job #{job["name"]} ##{job["uid"]} status : #{job["state"]}"
108
+ end
109
+ end
110
+
111
+ def clean
112
+ self.jobs.each do |job|
113
+ if job.reload["state"] == "running"
114
+ job.delete
115
+ logger.info "Job ##{job["uid"]} deleted !"
116
+ end
117
+ end
118
+ FileUtils.rm(".xp_cache")
119
+ end
120
+
121
+ private
122
+
123
+ def logger
124
+ @logger
125
+ end
126
+
127
+ def update_cache
128
+ cache = { :jobs => self.jobs.collect { |x| x.properties },
129
+ :deployments => self.deployments
130
+ }
131
+ open(".xp_cache", "w") do |f|
132
+ f.puts cache.to_json
133
+ end
134
+ end
135
+
136
+ end
137
+ end
data/lib/xp5k.rb ADDED
@@ -0,0 +1,6 @@
1
+ require 'xp5k/config'
2
+ require 'xp5k/xp'
3
+
4
+ module XP5K
5
+ Config.load
6
+ end # module:: XP
data/xp5k.gemspec ADDED
@@ -0,0 +1,26 @@
1
+ # -*- encoding: utf-8 -*-
2
+ $:.push File.expand_path('../lib', __FILE__)
3
+ require 'xp5k/version'
4
+
5
+ Gem::Specification.new do |s|
6
+ s.name = 'xp5k'
7
+ s.version = XP5K::VERSION
8
+ s.platform = Gem::Platform::RUBY
9
+ s.authors = ['Pascal Morillon']
10
+ s.email = ['pascal.morillon@irisa.fr']
11
+ s.homepage = 'https://github.com/pmorillon/xp5k'
12
+ s.summary = %q{A small Grid'5000 helper to submit jobs and deploy environments via REST API}
13
+ s.description = %q{A small Grid'5000 helper to submit jobs and deploy environments via REST API}
14
+
15
+ s.required_rubygems_version = '>= 1.3.6'
16
+
17
+ s.add_dependency 'restfully', '~>0.6'
18
+ s.add_dependency 'term-ansicolor', '>= 1.0.7'
19
+ s.add_dependency 'json', '>= 1.5.1'
20
+
21
+ s.add_development_dependency 'bundler', '>= 1.0.0'
22
+
23
+ s.files = `git ls-files`.split("\n")
24
+ s.require_paths = ['lib']
25
+
26
+ end
metadata ADDED
@@ -0,0 +1,122 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: xp5k
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.0.1
5
+ prerelease:
6
+ platform: ruby
7
+ authors:
8
+ - Pascal Morillon
9
+ autorequire:
10
+ bindir: bin
11
+ cert_chain: []
12
+ date: 2012-12-04 00:00:00.000000000 Z
13
+ dependencies:
14
+ - !ruby/object:Gem::Dependency
15
+ name: restfully
16
+ requirement: !ruby/object:Gem::Requirement
17
+ none: false
18
+ requirements:
19
+ - - ~>
20
+ - !ruby/object:Gem::Version
21
+ version: '0.6'
22
+ type: :runtime
23
+ prerelease: false
24
+ version_requirements: !ruby/object:Gem::Requirement
25
+ none: false
26
+ requirements:
27
+ - - ~>
28
+ - !ruby/object:Gem::Version
29
+ version: '0.6'
30
+ - !ruby/object:Gem::Dependency
31
+ name: term-ansicolor
32
+ requirement: !ruby/object:Gem::Requirement
33
+ none: false
34
+ requirements:
35
+ - - ! '>='
36
+ - !ruby/object:Gem::Version
37
+ version: 1.0.7
38
+ type: :runtime
39
+ prerelease: false
40
+ version_requirements: !ruby/object:Gem::Requirement
41
+ none: false
42
+ requirements:
43
+ - - ! '>='
44
+ - !ruby/object:Gem::Version
45
+ version: 1.0.7
46
+ - !ruby/object:Gem::Dependency
47
+ name: json
48
+ requirement: !ruby/object:Gem::Requirement
49
+ none: false
50
+ requirements:
51
+ - - ! '>='
52
+ - !ruby/object:Gem::Version
53
+ version: 1.5.1
54
+ type: :runtime
55
+ prerelease: false
56
+ version_requirements: !ruby/object:Gem::Requirement
57
+ none: false
58
+ requirements:
59
+ - - ! '>='
60
+ - !ruby/object:Gem::Version
61
+ version: 1.5.1
62
+ - !ruby/object:Gem::Dependency
63
+ name: bundler
64
+ requirement: !ruby/object:Gem::Requirement
65
+ none: false
66
+ requirements:
67
+ - - ! '>='
68
+ - !ruby/object:Gem::Version
69
+ version: 1.0.0
70
+ type: :development
71
+ prerelease: false
72
+ version_requirements: !ruby/object:Gem::Requirement
73
+ none: false
74
+ requirements:
75
+ - - ! '>='
76
+ - !ruby/object:Gem::Version
77
+ version: 1.0.0
78
+ description: A small Grid'5000 helper to submit jobs and deploy environments via REST
79
+ API
80
+ email:
81
+ - pascal.morillon@irisa.fr
82
+ executables: []
83
+ extensions: []
84
+ extra_rdoc_files: []
85
+ files:
86
+ - .gitignore
87
+ - Gemfile
88
+ - Gemfile.lock
89
+ - LICENCE
90
+ - README.md
91
+ - lib/xp5k.rb
92
+ - lib/xp5k/config.rb
93
+ - lib/xp5k/version.rb
94
+ - lib/xp5k/xp.rb
95
+ - xp5k.gemspec
96
+ homepage: https://github.com/pmorillon/xp5k
97
+ licenses: []
98
+ post_install_message:
99
+ rdoc_options: []
100
+ require_paths:
101
+ - lib
102
+ required_ruby_version: !ruby/object:Gem::Requirement
103
+ none: false
104
+ requirements:
105
+ - - ! '>='
106
+ - !ruby/object:Gem::Version
107
+ version: '0'
108
+ required_rubygems_version: !ruby/object:Gem::Requirement
109
+ none: false
110
+ requirements:
111
+ - - ! '>='
112
+ - !ruby/object:Gem::Version
113
+ version: 1.3.6
114
+ requirements: []
115
+ rubyforge_project:
116
+ rubygems_version: 1.8.24
117
+ signing_key:
118
+ specification_version: 3
119
+ summary: A small Grid'5000 helper to submit jobs and deploy environments via REST
120
+ API
121
+ test_files: []
122
+ has_rdoc: