Module: Jenkins::Api
- Includes:
- HTTParty
- Defined in:
- lib/jenkins/api.rb
Constant Summary collapse
- JobAlreadyExistsError =
http_proxy ‘localhost’, ‘8888’
Class.new(Exception)
Class Method Summary collapse
-
.add_node(options = {}) ⇒ Object
Adds SSH nodes only, for now.
-
.build_details(job_name, build_number) ⇒ Object
Return a hash of information about a build.
- .build_job(name) ⇒ Object
- .cache_configuration! ⇒ Object
-
.console(job_name, axe, build_number) ⇒ Object
Return the console log information about a build.
-
.create_job(name, job_config, options = {}) ⇒ Object
returns true if successfully create a new job on Jenkins
job_config
is a Jenkins::JobConfigBuilder instanceoptions
are: :override - true, will delete any existing job with same name, else error. -
.delete_job(name) ⇒ Object
Attempts to delete a job
name
. - .delete_node(name) ⇒ Object
-
.get_plain(path, options = {}) ⇒ Object
Helper for GET that don’t barf at Jenkins’s crappy API responses.
-
.job(name) ⇒ Object
Return hash of job statuses.
- .job_names ⇒ Object
- .nodes ⇒ Object
-
.post_plain(path, data = "", options = {}) ⇒ Object
Helper for POST that don’t barf at Jenkins’s crappy API responses.
- .setup_base_url(options = {}) ⇒ Object
- .summary ⇒ Object
-
.update_job(name, job_config) ⇒ Object
returns true if successfully updated a job on Jenkins
job_config
is a Jenkins::JobConfigBuilder instance.
Class Method Details
.add_node(options = {}) ⇒ Object
Adds SSH nodes only, for now
170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 |
# File 'lib/jenkins/api.rb', line 170 def self.add_node( = {}) = .with_clean_keys = Hash.new if [:vagrant] .merge!( :slave_port => 2222, :slave_user => 'vagrant', :master_key => "/Library/Ruby/Gems/1.8/gems/vagrant-0.6.7/keys/vagrant", # FIXME - hardcoded master username assumption :slave_fs => "/vagrant/tmp/jenkins-slave/", :description => "Automatically created by Jenkins.rb", :executors => 2, :exclusive => true ) else .merge!( :slave_port => 22, :slave_user => 'deploy', :master_key => "/home/deploy/.ssh/id_rsa", # FIXME - hardcoded master username assumption :slave_fs => "/data/jenkins-slave/", :description => "Automatically created by Jenkins.rb", :executors => 2, :exclusive => true ) end = .merge() slave_host = [:slave_host] name = [:name] || slave_host labels = [:labels].split(/\s*,\s*/).join(' ') if [:labels] type = "hudson.slaves.DumbSlave$DescriptorImpl" fields = { "name" => name, "type" => type, "json" => { "name" => name, "nodeDescription" => [:description], "numExecutors" => [:executors], "remoteFS" => [:slave_fs], "labelString" => labels, "mode" => [:exclusive] ? "EXCLUSIVE" : "NORMAL", "type" => type, "retentionStrategy" => { "stapler-class" => "hudson.slaves.RetentionStrategy$Always" }, "nodeProperties" => { "stapler-class-bag" => "true" }, "launcher" => { "stapler-class" => "hudson.plugins.sshslaves.SSHLauncher", "host" => slave_host, "port" => [:slave_port], "username" => [:slave_user], "privatekey" => [:master_key], } }.to_json } url = URI.parse("#{base_uri}/computer/doCreateItem") req = Net::HTTP::Post.new(url.path) req.set_form_data(fields) http = Net::HTTP.new(url.host, url.port) response = http.request(req) case response when Net::HTTPFound { :name => name, :slave_host => slave_host } else # error message looks like: # <td id="main-panel"> # <h1>Error</h1><p>Slave called 'localhost' already exists</p> require "hpricot" error = Hpricot(response.body).search("td#main-panel p").text unless error.blank? puts error else puts response.body # so we can find other errors end false end end |
.build_details(job_name, build_number) ⇒ Object
Return a hash of information about a build.
143 144 145 146 147 148 149 150 151 |
# File 'lib/jenkins/api.rb', line 143 def self.build_details(job_name, build_number) begin json = get "/job/#{job_name}/#{build_number}/api/json" cache_configuration! json rescue Crack::ParseError false end end |
.build_job(name) ⇒ Object
116 117 118 119 |
# File 'lib/jenkins/api.rb', line 116 def self.build_job(name) res = get_plain "/job/#{name}/build" res.code.to_i == 302 end |
.cache_configuration! ⇒ Object
276 277 278 279 280 |
# File 'lib/jenkins/api.rb', line 276 def self.cache_configuration! Jenkins::Config.config["base_uri"] = base_uri Jenkins::Config.config["basic_auth"] = [:basic_auth] Jenkins::Config.store! end |
.console(job_name, axe, build_number) ⇒ Object
Return the console log information about a build.
154 155 156 157 158 159 160 161 |
# File 'lib/jenkins/api.rb', line 154 def self.console(job_name, axe, build_number) path = "/job/#{job_name}/#{build_number}/" path << "#{axe}/" if axe path << "consoleText" log = get_plain path cache_configuration! log.body end |
.create_job(name, job_config, options = {}) ⇒ Object
returns true if successfully create a new job on Jenkins job_config
is a Jenkins::JobConfigBuilder instance options
are:
:override - true, will delete any existing job with same name, else error
returns true if successful, else false
TODO Exceptions?
71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 |
# File 'lib/jenkins/api.rb', line 71 def self.create_job(name, job_config, = {}) = .with_clean_keys delete_job(name) if [:override] begin res = post "/createItem/api/xml?name=#{CGI.escape(name)}", { :body => job_config.to_xml, :format => :xml, :headers => { 'content-type' => 'application/xml' } } if res.code.to_i == 200 cache_configuration! true else show_me_the_error(res) false end rescue REXML::ParseException => e # For some reason, if the job exists we get back half a page of HTML raise JobAlreadyExistsError.new(name) end end |
.delete_job(name) ⇒ Object
Attempts to delete a job name
111 112 113 114 |
# File 'lib/jenkins/api.rb', line 111 def self.delete_job(name) res = post_plain "#{job_url name}/doDelete" res.code.to_i == 302 end |
.delete_node(name) ⇒ Object
252 253 254 |
# File 'lib/jenkins/api.rb', line 252 def self.delete_node(name) post_plain("#{base_uri}/computer/#{CGI::escape(name).gsub('+', '%20')}/doDelete/api/json") end |
.get_plain(path, options = {}) ⇒ Object
Helper for GET that don’t barf at Jenkins’s crappy API responses
270 271 272 273 274 |
# File 'lib/jenkins/api.rb', line 270 def self.get_plain(path, = {}) = .with_clean_keys uri = URI.parse base_uri res = Net::HTTP.start(uri.host, uri.port) { |http| http.get(path, ) } end |
.job(name) ⇒ Object
Return hash of job statuses
132 133 134 135 136 137 138 139 140 |
# File 'lib/jenkins/api.rb', line 132 def self.job(name) begin json = get "/job/#{name}/api/json" cache_configuration! json rescue Crack::ParseError false end end |
.job_names ⇒ Object
127 128 129 |
# File 'lib/jenkins/api.rb', line 127 def self.job_names summary["jobs"].map {|job| job["name"]} end |
.nodes ⇒ Object
163 164 165 166 167 |
# File 'lib/jenkins/api.rb', line 163 def self.nodes json = get "/computer/api/json" cache_configuration! json end |
.post_plain(path, data = "", options = {}) ⇒ Object
Helper for POST that don’t barf at Jenkins’s crappy API responses
257 258 259 260 261 262 263 264 265 266 267 |
# File 'lib/jenkins/api.rb', line 257 def self.post_plain(path, data = "", = {}) = .with_clean_keys uri = URI.parse base_uri res = Net::HTTP.start(uri.host, uri.port) do |http| if RUBY_VERSION =~ /1.8/ http.post(path, ) else http.post(path, data, ) end end end |
.setup_base_url(options = {}) ⇒ Object
22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 |
# File 'lib/jenkins/api.rb', line 22 def self.setup_base_url( = {}) # Handle single strings = { :host => } if .is_a? String = .with_clean_keys # Thor's HashWithIndifferentAccess is based on string keys which URI::HTTP.build ignores = .inject({}) { |mem, (key, val)| mem[key.to_sym] = val; mem } = setup_authentication() # Handle URL style hosts by parsing the URL if .keys.length == 1 && .key?(:host) parsed_uri = URI::parse([:host]) = { :host => parsed_uri.host, :port => parsed_uri.port, :path => parsed_uri.path, :ssl => parsed_uri.scheme == 'https' } if parsed_uri.user && parsed_uri.password basic_auth parsed_uri.user, parsed_uri.password end end [:host] ||= ENV['JENKINS_HOST'] [:port] ||= ENV['JENKINS_PORT'] [:port] &&= [:port].to_i if [:host] uri_class = .delete(:ssl) ? URI::HTTPS : URI::HTTP uri = uri_class.build() else if Jenkins::Config.config["base_uri"] uri = Jenkins::Config.config["base_uri"] else return false # Nothing to work with. end end base_uri uri.to_s uri end |
.summary ⇒ Object
121 122 123 124 125 |
# File 'lib/jenkins/api.rb', line 121 def self.summary json = get "/api/json" cache_configuration! json end |
.update_job(name, job_config) ⇒ Object
returns true if successfully updated a job on Jenkins job_config
is a Jenkins::JobConfigBuilder instance
returns true if successful, else false
TODO Exceptions?
97 98 99 100 101 102 103 104 105 106 107 108 |
# File 'lib/jenkins/api.rb', line 97 def self.update_job(name, job_config) res = post "#{job_url name}/config.xml", { :body => job_config.to_xml, :format => :xml, :headers => { 'content-type' => 'application/xml' } } if res.code.to_i == 200 cache_configuration! true else show_me_the_error(res) false end end |