Class: ServiceNowHelper
- Inherits:
-
BaseHelper
- Object
- BaseHelper
- ServiceNowHelper
- Defined in:
- lib/nexpose_ticketing/helpers/servicenow_helper.rb
Overview
Serves as the ServiceNow interface for creating/updating issues from vulnerabilities found in Nexpose.
Constant Summary collapse
- NEW_STATE =
1
- RESOLVED_STATE =
6
- CLOSED_STATE =
7
Instance Attribute Summary collapse
-
#log ⇒ Object
Returns the value of attribute log.
-
#transform ⇒ Object
Returns the value of attribute transform.
Attributes inherited from BaseHelper
Instance Method Summary collapse
-
#close_tickets(nxid_requests) ⇒ Object
Sends ticket closure (in JSON format) to ServiceNow individually (each ticket in the list as a separate HTTP post).
-
#create_tickets(tickets) ⇒ Object
Sends a list of tickets (in JSON format) to ServiceNow individually (each ticket in the list as a separate HTTP post).
- #generate_identifier_request(nxid) ⇒ Object
-
#generate_post_request(ticket, forbid_connection_reuse) ⇒ Object
Method generates a HTTP POST request containing the provided ticket to send to ServiceNow.
-
#generate_ticket_request(ticket, forbid_connection_reuse) ⇒ Object
Method generates a HTTP POST request containing the provided ticket to send to ServiceNow.
-
#initialize(service_data, options, mode) ⇒ ServiceNowHelper
constructor
A new instance of ServiceNowHelper.
- #parse_identifier_response(response, nxid) ⇒ Object
-
#prepare_close_tickets(vulnerability_list, nexpose_identifier_id) ⇒ Object
Prepare ticket closures from the CSV of vulnerabilities exported from Nexpose.
-
#prepare_create_tickets(vulnerability_list, nexpose_identifier_id) ⇒ Object
Prepare tickets from the CSV of vulnerabilities exported from Nexpose.
-
#prepare_tickets(vulnerability_list, nexpose_identifier_id) ⇒ Object
Prepares a list of vulnerabilities into a list of JSON-formatted tickets (incidents) for ServiceNow.
-
#prepare_update_tickets(vulnerability_list, nexpose_identifier_id) ⇒ Object
Prepare ticket updates from the CSV of vulnerabilities exported from Nexpose.
-
#update_tickets(tickets) ⇒ Object
Sends ticket updates (in JSON format) to ServiceNow by placing each request on a Typhoeus queue (each ticket in the list as a separate HTTP post).
Methods inherited from BaseHelper
Constructor Details
#initialize(service_data, options, mode) ⇒ ServiceNowHelper
Returns a new instance of ServiceNowHelper.
21 22 23 |
# File 'lib/nexpose_ticketing/helpers/servicenow_helper.rb', line 21 def initialize(service_data, , mode) super(service_data, , mode) end |
Instance Attribute Details
#log ⇒ Object
Returns the value of attribute log.
20 21 22 |
# File 'lib/nexpose_ticketing/helpers/servicenow_helper.rb', line 20 def log @log end |
#transform ⇒ Object
Returns the value of attribute transform.
20 21 22 |
# File 'lib/nexpose_ticketing/helpers/servicenow_helper.rb', line 20 def transform @transform end |
Instance Method Details
#close_tickets(nxid_requests) ⇒ Object
Sends ticket closure (in JSON format) to ServiceNow individually (each ticket in the list as a separate HTTP post).
-
Args :
-
requests
- Hash containing NXIDs and associated Typheous requests.
-
345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 |
# File 'lib/nexpose_ticketing/helpers/servicenow_helper.rb', line 345 def close_tickets(nxid_requests) if nxid_requests.nil? || nxid_requests.empty? @log.('No tickets to close.') return end ticket = { 'sysparm_action' => 'insert', 'u_rpd_id' => nil, 'u_state' => CLOSED_STATE } requests = [] final_ticket = nxid_requests.count - 1 hydra = Typhoeus::Hydra.new nxid_requests.each_with_index do |(nxid, request), i| request.on_complete do |response| u_rpd_id = parse_identifier_response(response, nxid)[:id] ticket['u_rpd_id'] = u_rpd_id ticket_request = generate_post_request(ticket.to_json, i == final_ticket) hydra.queue ticket_request requests << ticket_request end hydra.queue request end hydra.run @metrics.closed requests.count @log.('Closed ticket batch.') requests.map(&:response) end |
#create_tickets(tickets) ⇒ Object
Sends a list of tickets (in JSON format) to ServiceNow individually (each ticket in the list as a separate HTTP post).
-
Args :
-
tickets
- List of JSON-formatted ticket creates (new tickets).
-
116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 |
# File 'lib/nexpose_ticketing/helpers/servicenow_helper.rb', line 116 def create_tickets(tickets) fail 'Ticket(s) cannot be empty' if tickets.nil? || tickets.empty? final_ticket = tickets.count - 1 ticket_index = 0 hydra = Typhoeus::Hydra.new requests = tickets.map do |ticket| ticket['u_rpd_id'] = SecureRandom.uuid request = generate_post_request(ticket.to_json, ticket_index == final_ticket) hydra.queue request ticket_index += 1 request end hydra.run @metrics.created tickets.count @log.('Created ticket batch.') requests.map(&:response) end |
#generate_identifier_request(nxid) ⇒ Object
274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 |
# File 'lib/nexpose_ticketing/helpers/servicenow_helper.rb', line 274 def generate_identifier_request(nxid) query = "incident.do?JSONv2&sysparm_query=active=true^u_nxid=#{nxid}" url = URI.join(@service_data[:servicenow_url], "/").to_s + query userpwd = "#{@service_data[:username]}:#{@service_data[:password]}" headers = { 'Content-Type' => 'application/json' } = { method: :get, userpwd: userpwd, headers: headers, accept_encoding: 'application/json', maxredirs: @service_data[:redirect_limit], ssl_verifyhost: 0 } Typhoeus::Request.new(url, ) end |
#generate_post_request(ticket, forbid_connection_reuse) ⇒ Object
Method generates a HTTP POST request containing the provided ticket to
send to ServiceNow. Provides error handling via on_complete functionality
-
Args :
-
ticket
- The ticket to be sent to ServiceNow -
forbid_connection_reuse
- Whether the current HTTP connection can bereused to send tickets to ServiceNow.
-
-
Returns :
-
A HTTP post request object to be placed on the queue for sending
-
225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 |
# File 'lib/nexpose_ticketing/helpers/servicenow_helper.rb', line 225 def generate_post_request(ticket, forbid_connection_reuse) request = generate_ticket_request(ticket, forbid_connection_reuse) request.on_complete do |response| unless response.success? msg = if response.timed_out? 'Time out has occurred.' elsif response.code == 0 response. else "HTTP request failed: #{response.code}" end @log. msg raise msg end end request end |
#generate_ticket_request(ticket, forbid_connection_reuse) ⇒ Object
Method generates a HTTP POST request containing the provided ticket to
send to ServiceNow. Provides error handling via on_complete functionality
-
Args :
-
ticket
- The ticket to be sent to ServiceNow -
forbid_connection_reuse
- Whether the current HTTP connection can bereused to send tickets to ServiceNow.
-
-
Returns :
-
A HTTP request object to be placed on the queue for sending
-
255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 |
# File 'lib/nexpose_ticketing/helpers/servicenow_helper.rb', line 255 def generate_ticket_request(ticket, forbid_connection_reuse) address = @service_data[:servicenow_url] userpwd = "#{@service_data[:username]}:#{@service_data[:password]}" headers = { 'Content-Type' => 'application/json' } = { method: :post, userpwd: userpwd, headers: headers, accept_encoding: 'application/json', maxredirs: @service_data[:redirect_limit], ssl_verifyhost: 0, forbid_reuse: forbid_connection_reuse, body: ticket } Typhoeus::Request.new(address, ) end |
#parse_identifier_response(response, nxid) ⇒ Object
291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 |
# File 'lib/nexpose_ticketing/helpers/servicenow_helper.rb', line 291 def parse_identifier_response(response, nxid) tickets = JSON.parse(response.body) records = tickets['records'] if records.count > 1 @log.("Found more than one result for NXID #{nxid}. " \ 'Updating first result.') records.each { |r| @log.("NXID #{nxid} found with " \ "Rapid7 Identifier #{r['u_rpd_id']}") } elsif records.count == 0 @log.("No results found for NXID #{nxid}.") return { id: nil, state: nil } end ticket_id = records.first['u_rpd_id'] state = records.first['state'] @log.("Found ticket for NXID #{nxid} ID is: #{ticket_id}") if ticket_id.nil? @log.("ID is nil for ticket with NXID #{nxid}.") state = nil end { id: ticket_id, state: state } end |
#prepare_close_tickets(vulnerability_list, nexpose_identifier_id) ⇒ Object
Prepare ticket closures from the CSV of vulnerabilities exported from Nexpose. This method currently only supports updating default mode tickets in ServiceNow.
-
Args :
-
vulnerability_list
- CSV of vulnerabilities within Nexpose.
-
-
Returns :
-
List of JSON-formated tickets for closing within ServiceNow.
-
325 326 327 328 329 330 331 332 333 334 335 336 337 |
# File 'lib/nexpose_ticketing/helpers/servicenow_helper.rb', line 325 def prepare_close_tickets(vulnerability_list, nexpose_identifier_id) @log.("Preparing ticket closures for mode #{@options[:ticket_mode]}.") requests = {} CSV.parse(vulnerability_list.chomp, headers: :first_row) do |row| nxid = @mode_helper.get_nxid(nexpose_identifier_id, row) @log.("Closing ticket with NXID: #{nxid}.") request = generate_identifier_request(nxid) requests[nxid] = request end requests end |
#prepare_create_tickets(vulnerability_list, nexpose_identifier_id) ⇒ Object
Prepare tickets from the CSV of vulnerabilities exported from Nexpose. This method determines how to prepare the tickets (either by default or by IP address) based on config options.
-
Args :
-
vulnerability_list
- CSV of vulnerabilities within Nexpose.
-
-
Returns :
-
List of JSON-formated tickets for creating within ServiceNow.
-
106 107 108 |
# File 'lib/nexpose_ticketing/helpers/servicenow_helper.rb', line 106 def prepare_create_tickets(vulnerability_list, nexpose_identifier_id) prepare_tickets(vulnerability_list, nexpose_identifier_id) end |
#prepare_tickets(vulnerability_list, nexpose_identifier_id) ⇒ Object
Prepares a list of vulnerabilities into a list of JSON-formatted tickets (incidents) for ServiceNow.
-
Args :
-
vulnerability_list
- CSV of vulnerabilities within Nexpose.
-
-
Returns :
-
List of JSON-formated tickets for creating within ServiceNow.
-
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 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 |
# File 'lib/nexpose_ticketing/helpers/servicenow_helper.rb', line 34 def prepare_tickets(vulnerability_list, nexpose_identifier_id) @metrics.start matching_fields = @mode_helper.get_matching_fields @ticket = Hash.new(-1) @log.("Preparing tickets in #{[:ticket_mode]} mode format.") tickets = [] previous_row = nil description = nil action = 'insert' CSV.parse(vulnerability_list.chomp, headers: :first_row) do |row| if previous_row.nil? previous_row = row.dup nxid = @mode_helper.get_nxid(nexpose_identifier_id, row) action = if row['comparison'].nil? || row['comparison'] == 'New' 'insert' else 'update' end @ticket = { 'sysparm_action' => action, 'u_caller_id' => "#{@service_data[:username]}", 'u_category' => 'Software', 'u_impact' => '1', 'u_urgency' => '1', 'u_short_description' => @mode_helper.get_title(row), 'u_work_notes' => "", 'u_nxid' => nxid, 'u_rpd_id' => nil } description = @mode_helper.get_description(nexpose_identifier_id, row) elsif matching_fields.any? { |x| previous_row[x].nil? || previous_row[x] != row[x] } info = @mode_helper.get_field_info(matching_fields, previous_row) @log.("Generated ticket with #{info}") @ticket['u_work_notes'] = @mode_helper.print_description(description) tickets.push(@ticket) previous_row = nil description = nil redo else unless row['comparison'].nil? || row['comparison'] == 'New' @ticket['sysparm_action'] = 'update' end description = @mode_helper.update_description(description, row) end end unless @ticket.nil? || @ticket.empty? info = @mode_helper.get_field_info(matching_fields, previous_row) @log.("Generated ticket with #{info}") @ticket['u_work_notes'] = @mode_helper.print_description(description) unless (@ticket.size == 0) tickets.push(@ticket) end @log.("Generated <#{tickets.count.to_s}> tickets.") tickets end |
#prepare_update_tickets(vulnerability_list, nexpose_identifier_id) ⇒ Object
Prepare ticket updates from the CSV of vulnerabilities exported from Nexpose. The list of vulnerabilities are ordered depending on the ticketing mode and then by ticket_status, allowing the method to loop through and display new, old, and same vulnerabilities in that order.
- +vulnerability_list+ - CSV of vulnerabilities within Nexpose.
-
Returns :
-
List of JSON-formated tickets for updating within ServiceNow.
-
147 148 149 |
# File 'lib/nexpose_ticketing/helpers/servicenow_helper.rb', line 147 def prepare_update_tickets(vulnerability_list, nexpose_identifier_id) prepare_tickets(vulnerability_list, nexpose_identifier_id) end |
#update_tickets(tickets) ⇒ Object
Sends ticket updates (in JSON format) to ServiceNow by placing each request on a Typhoeus queue (each ticket in the list as a separate HTTP post).
-
Args :
-
tickets
- List of Hash-formatted ticket updates.
-
157 158 159 160 161 162 163 164 165 166 167 168 169 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 |
# File 'lib/nexpose_ticketing/helpers/servicenow_helper.rb', line 157 def update_tickets(tickets) if tickets.nil? || tickets.empty? @log.('No tickets to update.') return end requests = [] final_ticket = tickets.count - 1 hydra = Typhoeus::Hydra.new tickets.each_with_index do |ticket, i| if ticket['sysparm_action'] == 'update' id_request = generate_identifier_request(ticket['u_nxid']) id_request.on_complete do |response| ticket['sysparm_action'] = 'insert' current_data = parse_identifier_response(response, ticket['u_nxid']) u_rpd_id = current_data[:id] ticket['u_rpd_id'] = u_rpd_id || SecureRandom.uuid if current_data[:state] == RESOLVED_STATE ticket['u_state'] = NEW_STATE title = "(Reopened) #{ticket['u_short_description']}" ticket['u_short_description'] = title current_notes = ticket['u_work_notes'].rpartition("\n\n\nNXID: ") new_notes = '++ Reopened by Nexpose Ticketing Gem ' \ "++\n#{current_notes.first}" nxid = current_notes[1,2].join('').lstrip description = @mode_helper.finalize_description(new_notes, nxid) ticket['u_work_notes'] = description end ticket_request = generate_post_request(ticket.to_json, i == final_ticket) hydra.queue ticket_request ticket['u_rpd_id'] == u_rpd_id ? @metrics.updated : @metrics.created requests << ticket_request end hydra.queue id_request elsif ticket['sysparm_action'] == 'insert' ticket['u_rpd_id'] ||= SecureRandom.uuid ticket_request = generate_post_request(ticket.to_json, i == final_ticket) hydra.queue ticket_request @metrics.created requests << ticket_request end end hydra.run @log.('Updated ticket batch.') requests.map(&:response) end |