Forum Discussion
Anonymous
3 years agoCreate a Read Only API token. Make the note on the token something like "Used to populate SDT status". Copy the id and key to a temporary clipboard.
Add [yourportal].logicmonitor.com as a monitored device (if you haven't already).
Add sdt_api.id, sdt_api.pass, and sdt_api.account as properties on that device. Use the values you copied earlier for the id and pass. For company use your portal name.
Create a property source that applies to all devices (true()). Use this script. I've tested the script, but haven't tested the whole propertysource.
import javax.crypto.Mac import javax.crypto.spec.SecretKeySpec import org.apache.commons.codec.binary.Hex import groovy.json.JsonSlurper Map credentials = [ "id" : hostProps.get("sdt_api.id"), "key" : hostProps.get("sdt_api.pass"), "account": hostProps.get("sdt_api.account") ] deviceId = hostProps.get("system.deviceId") Map resources = [ "devices": ["path": "/device/devices", "details": ["fields": "displayName,id,sdtStatus", "filter":"id:${deviceId}"]], ] resources.each() { k, v -> Map headers = generate_headers(credentials.id, credentials.key, v.path) if (headers) { Map response = get_response(k, v, credentials.account, headers) if (response?.success) {resources[k]["data"] = response.response} } } Map groups = [:] resources.devices.data.each(){ statuses = it.sdtStatus.tokenize("-") println("auto.group.sdt=${statuses[0]}") println("auto.device.sdt=${statuses[1]}") println("auto.instance.sdt=${statuses[2]}") } return 0 def generate_headers(id, key, path) { try { // Create encryption signature for authorization request Long epoch_time = System.currentTimeMillis() // Get current system time (epoch time) Mac hmac = Mac.getInstance("HmacSHA256") hmac.init(new SecretKeySpec(key.getBytes(), "HmacSHA256")) signature = Hex.encodeHexString(hmac.doFinal("GET${epoch_time}${path}".getBytes())).bytes.encodeBase64() // return headers to main function return ["Authorization": "LMv1 $id:$signature:$epoch_time", "Content-Type": "application/json"] } catch (Exception err) { // If error occurred, print the error message println("ERROR: Unable to establish encryption for $path. Attempting next resource...\n${err.message}") } } def get_response(resource, parameters, account, headers) { try { boolean proceed = true // Boolean used to determine if additional pagination is required // Map to store query results for each endpoint. Contains a list to store actual returned values and a boolean to determine if successful Map results = ["response": [], "success" : true] add_query_parameters(resource, parameters) // Add initial offset and size values to appropriate categories (skips metrics category since it's stagnate) while (proceed) { // Used for paginating through all availabe results. Grabs 1000 at a time and moves offset if another query is required. Map query = query_resource(account, parameters, headers) // Query each API endpoint for a response (Should receive as Map) // If the response was successful (including status and error messages), proceed to printing results if (query && query?.data && query?.status == 200 && query?.errmsg?.toUpperCase() == "OK") { if (resource != "metrics") { results.response.addAll(query.data.items) // Add all the data items found to our results map data list if (query?.data?.items?.size() < parameters.details.size) { // If we received less than 1000 results proceed = false // There is no need to execute another API query with a shifted offset } else { // Otherwise parameters.details.offset += parameters.details.size // Shift the offset to start 1000 numbers from current position } } else { results.response = query.data // Add all the data items found to our results map data list proceed = false // We've successfully queried all values. End while loop } } else { // If response was not successful, print eror message for each category that failed and continue to next endpoint // If response error and status can be determined, print them. Otherwise, use UNKNOWN println("ERROR: Failed to query $resource API Endpoint...\n" + "${query?.errmsg?.toUpperCase() ?: 'UNKNOWN'} (STATUS: ${query?.status ?: 'UNKNOWN'})") results.success = false // Set success value to false since we failed our API query proceed = false // End while loop because of failure and proceed to next endpoint } } return results // Return results to main function } catch (Exception err) { println("ERROR: Script failed while attempting to query $resource API endpoint...\n${err?.message}") } } def add_query_parameters(category, parameters) { // Add size and offset field to map (only if collectors or admins category) if (category != "metrics") { Map query_details = ["size" : 1000, "offset": 0] // If there's already a details key in the details map if (parameters.details) { parameters.details << query_details // Append the query details information to the pre-existing details map } else { // Otherwise, create a details key and assign it the query details map as a value parameters.put("details", query_details) } } } def query_resource(account, details, headers) { try { // Configure request url from account, path, and authorization headers String url = "https://${account}.logicmonitor.com/santaba/rest${details.path}?${pack_parameters(details.details)}" // Return query response, converted from JSON to usable map return new JsonSlurper().parseText(url.toURL().getText(useCaches: true, allowUserInteraction: false, requestProperties: headers)) } catch (Exception err) { // If error occurred, print the error message println("ERROR: Unable to query ${details.path} for details.\n${err.message}") } } def pack_parameters(query_details) { // If additional query details are located in map, include them in url string List pairs = [] query_details?.each { k, v -> pairs.add("${k}=${v}")} return pairs.join("&") }
Related Content
- 8 months ago