Skip to content

Commit

Permalink
Devo v2 dev (#32989) (#33282)
Browse files Browse the repository at this point in the history
* write_to_table_command & write_to_lookup_table_command changes

* updated test cases & bug fixes

* updated devo_write_to_table test case

* prevoius code reverted

* diff changes

* Bug fixes

* Validation checks & Unit test cases added

* code reverted & working on readme file

* readme file changes updated

* optional parameters check added

* code reverted

* updated test cases & implemented the requested suggestions

* reverted changes

* added additional unit test cases

* Docker image tag updated

* Updated the release note for latest docker image

* Docker Image updated

Co-authored-by: namrata-metron <[email protected]>
Co-authored-by: JudithB <[email protected]>
  • Loading branch information
3 people authored Mar 11, 2024
1 parent 9399dcd commit 217cead
Show file tree
Hide file tree
Showing 6 changed files with 452 additions and 98 deletions.
172 changes: 119 additions & 53 deletions Packs/Devo/Integrations/Devo_v2/Devo_v2.py
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,7 @@
TIMEOUT = demisto.params().get("timeout", "60")
PORT = arg_to_number(demisto.params().get("port", "443") or "443")
ITEMS_PER_PAGE = 50
LIMIT = 100
HEALTHCHECK_WRITER_RECORD = [{"hello": "world", "from": "demisto-integration"}]
HEALTHCHECK_WRITER_TABLE = "test.keep.free"
RANGE_PATTERN = re.compile("^[0-9]+ [a-zA-Z]+")
Expand Down Expand Up @@ -823,13 +824,42 @@ def multi_table_query_command(offset, items):
return entry


def convert_to_str(value):
if isinstance(value, list) and not value:
return_warning("Empty list encountered.")
return '[]'
elif isinstance(value, dict) and not value:
return_warning("Empty dictionary encountered.")
return '{}'
elif isinstance(value, list | dict):
return json.dumps(value)
return str(value)


def write_to_table_command():
table_name = demisto.args()["tableName"]
records = check_type(demisto.args()["records"], list)
tag = demisto.args().get("tag", None)
tableName = demisto.args()["tableName"]
records_str = demisto.args()["records"]
linq_base = demisto.args().get("linqLinkBase", None)

final_tag = tag or tableName

# Use json.loads to parse the records string
try:
records = json.loads(records_str)
except json.JSONDecodeError:
return_error("Error decoding JSON. Please ensure the records are valid JSON.")

# Ensure records is a list
if not isinstance(records, list):
return_error("The 'records' parameter must be a list.")

# Check if all records are empty
if all(not record for record in records):
return_error("All records are empty.")

creds = get_writer_creds()
linq = f"from {table_name}"
linq = f"from {tableName}"

sender = Sender(
SenderConfigSSL(
Expand All @@ -840,53 +870,55 @@ def write_to_table_command():
)
)

total_events = 0
total_bytes_sent = 0

for r in records:
try:
sender.send(tag=table_name, msg=json.dumps(r))
except TypeError:
sender.send(tag=table_name, msg=f"{r}")
# Convert each record to a JSON string or string
formatted_record = convert_to_str(r)

# If the record is empty, skip sending it
if not formatted_record.strip():
continue

# Send each record to Devo with the specified tag
sender.send(tag=final_tag, msg=formatted_record)

# Update totals
total_events += 1
total_bytes_sent += len(formatted_record.encode("utf-8"))

current_ts = int(time.time())
start_ts = (current_ts - 30) * 1000
end_ts = (current_ts + 30) * 1000

querylink = {
"DevoTableLink": build_link(
linq,
int(1000 * time.time()) - 3600000,
int(1000 * time.time()),
start_ts,
end_ts,
linq_base=linq_base,
)
}

entry = {
"Type": entryTypes["note"],
"Contents": {"recordsWritten": records},
"ContentsFormat": formats["json"],
"ReadableContentsFormat": formats["markdown"],
"EntryContext": {"Devo.RecordsWritten": records, "Devo.LinqQuery": linq},
}
entry_linq = {
"Type": entryTypes["note"],
"Contents": querylink,
"ContentsFormat": formats["json"],
"ReadableContentsFormat": formats["markdown"],
"EntryContext": {"Devo.QueryLink": createContext(querylink)},
}
headers: list = []
resultRecords: list = []
innerDict: dict = {}
for obj in records:
record = json.loads(obj)
currKey = list(record.keys())
currValue = list(record.values())

headers.extend(currKey)

innerDict.update(dict(zip(currKey, currValue))) # Create a dictionary using zip
resultRecords.append(innerDict) # Append the dictionary to the list

demisto.debug("final array :")
demisto.debug(resultRecords)
entry = {
"Type": entryTypes["note"],
"Contents": {"TotalRecords": total_events, "TotalBytesSent": total_bytes_sent},
"ContentsFormat": formats["json"],
"ReadableContentsFormat": formats["markdown"],
"EntryContext": {"Devo.TotalRecords": total_events, "Devo.TotalBytesSent": total_bytes_sent, "Devo.LinqQuery": linq},
}

md = tableToMarkdown("Entries to load into Devo", resultRecords, headers)
entry["HumanReadable"] = md
md_message = f"Total Records Sent: {total_events}.\nTotal Bytes Sent: {total_bytes_sent}."
entry["HumanReadable"] = md_message

md_linq = tableToMarkdown(
"Link to Devo Query",
Expand All @@ -898,9 +930,19 @@ def write_to_table_command():


def write_to_lookup_table_command():
lookup_table_name = demisto.args()["lookupTableName"]
headers = check_type(demisto.args()["headers"], list)
records = check_type(demisto.args()["records"], list)
lookup_table_name = demisto.args().get("lookupTableName")
headers_param = demisto.args().get("headers")
records_param = demisto.args().get("records")

if not lookup_table_name or not headers_param or not records_param:
return_error("Missing required arguments. Please provide lookupTableName, headers, and records.")

# Parse JSON strings to Python objects
try:
headers = json.loads(headers_param)
records = json.loads(records_param)
except json.JSONDecodeError as e:
return_error(f"Failed to parse JSON: {str(e)}")

creds = get_writer_creds()

Expand All @@ -911,35 +953,59 @@ def write_to_lookup_table_command():
chain=creds["chain"].name,
)

con = None
total_events = 0
total_bytes = 0

try:
con = Sender(config=engine_config, timeout=60)
# Validate headers
if not isinstance(headers, dict) or "headers" not in headers or not isinstance(headers["headers"], list):
raise ValueError("Invalid headers format. 'headers' must be a list.")

columns = headers["headers"]

lookup = Lookup(name=lookup_table_name, historic_tag=None, con=con)
# Order sensitive list
pHeaders = json.dumps(headers)
# Set default values for optional parameters
key_index = int(headers.get("key_index", 0)) # Ensure it's casted to integer
action = headers.get("action", "INC") # Set default value to 'INC'

lookup.send_control("START", pHeaders, "INC")
# Validate key_index
if key_index < 0:
raise ValueError("key_index must be a non-negative integer value.")

# Validate action
if action not in {"INC", "FULL"}:
raise ValueError("action must be either 'INC' or 'FULL'.")

con = Sender(config=engine_config, timeout=60)
lookup = Lookup(name=lookup_table_name, con=con)

lookup.send_headers(headers=columns, key_index=key_index, event="START", action=action)

# Send data lines
for r in records:
lookup.send_data_line(key_index=0, fields=r["values"])
fields = r.get("fields", [])
delete = r.get("delete", False)
lookup.send_data_line(key_index=key_index, fields=fields, delete=delete)
total_events += 1
total_bytes += len(json.dumps(r))

lookup.send_control("END", pHeaders, "INC")
finally:
# Send end event
lookup.send_headers(headers=columns, key_index=key_index, event="END", action=action)

# Flush buffer and shutdown connection
con.flush_buffer()
con.socket.shutdown(0)

entry = {
"Type": entryTypes["note"],
"Contents": {"recordsWritten": records},
"ContentsFormat": formats["json"],
"ReadableContentsFormat": formats["markdown"],
"EntryContext": {"Devo.RecordsWritten": records},
}
# Return three lines as output
return f"Lookup Table Name: {lookup_table_name}.\nTotal Records Sent: {total_events}.\nTotal Bytes Sent: {total_bytes}."

md = tableToMarkdown("Entries to load into Devo", records)
entry["HumanReadable"] = md
except Exception as e:
return_error(f"Failed to execute command write-to-lookup-table. Error: {str(e)}")

return [entry]
finally:
if con:
con.flush_buffer()
con.socket.shutdown(0)


def main():
Expand Down
7 changes: 4 additions & 3 deletions Packs/Devo/Integrations/Devo_v2/Devo_v2.yml
Original file line number Diff line number Diff line change
Expand Up @@ -194,12 +194,14 @@ script:
description: Queries multiple tables for a given token and returns relevant results.
- name: devo-write-to-table
arguments:
- name: tag
description: The tag to assign to the records.
- name: tableName
required: true
description: The name of the table to write to.
- name: records
required: true
description: Records to write to the specified table.
required: true
isArray: true
- name: linqLinkBase
description: Overrides the global Devo base domain for linq linking.
Expand All @@ -221,7 +223,6 @@ script:
- name: headers
required: true
description: Headers for lookup table control.
isArray: true
- name: records
required: true
description: Records to write to the specified table.
Expand All @@ -232,7 +233,7 @@ script:
type: unknown
description: Writes lookup table entry records to a specified Devo table.
execution: true
dockerimage: demisto/devo:1.0.0.86778
dockerimage: demisto/devo:1.0.0.89201
isfetch: true
subtype: python3
tests:
Expand Down
Loading

0 comments on commit 217cead

Please sign in to comment.