Unexpected Shell Parent Events

osquery
Unexpected process that spawns shell processes (event-based)
Author

Chainguard

Published

January 29, 2025

Description

ODK (osquery-defense-kit) is unique in that the queries are designed to be used as part of a production detection & response pipeline. The detection queries are formulated to return zero rows during normal expected behavior, so that they may be configured to generate alerts when rows are returned.

Query

-- Unexpected process that spawns shell processes (event-based)
--
-- false positives:
--   * IDE's
--
-- references:
--   * https://attack.mitre.org/techniques/T1059/ (Command and Scripting Interpreter)
--   * https://attack.mitre.org/techniques/T1204/002/ (User Execution: Malicious File)
--
-- tags: process events extra
-- interval: 300
-- platform: posix
SELECT
  -- Child
  pe.path AS p0_path,
  REGEX_MATCH (pe.path, '.*/(.*)', 1) AS p0_name,
  TRIM(pe.cmdline) AS p0_cmd,
  pe.cwd AS p0_cwd,
  pe.time AS p0_time,
  pe.pid AS p0_pid,
  pe.euid AS p0_euid,
  p.cgroup_path AS p0_cgroup,
  -- Parent
  pe.parent AS p1_pid,
  p1.cgroup_path AS p1_cgroup,
  TRIM(COALESCE(p1.cmdline, pe1.cmdline)) AS p1_cmd,
  COALESCE(p1.path, pe1.path) AS p1_path,
  COALESCE(p1.euid, pe1.euid) AS p1_euid,
  COALESCE(p_hash1.sha256, pe_hash1.sha256) AS p1_hash,
  REGEX_MATCH (COALESCE(p1.path, pe1.path), '.*/(.*)', 1) AS p1_name,
  -- Grandparent
  COALESCE(p1.parent, pe1.parent) AS p2_pid,
  COALESCE(p1_p2.cgroup_path, pe1_p2.cgroup_path) AS p2_cgroup,
  TRIM(
    COALESCE(p1_p2.cmdline, pe1_p2.cmdline, pe1_pe2.cmdline)
  ) AS p2_cmd,
  COALESCE(p1_p2.path, pe1_p2.path, pe1_pe2.path) AS p2_path,
  COALESCE(
    p1_p2_hash.path,
    pe1_p2_hash.path,
    pe1_pe2_hash.path
  ) AS p2_hash,
  REGEX_MATCH (
    COALESCE(p1_p2.path, pe1_p2.path, pe1_pe2.path),
    '.*/(.*)',
    1
  ) AS p2_name,
  -- Exception key
  REGEX_MATCH (pe.path, '.*/(.*)', 1) || ',' || MIN(pe.euid, 500) || ',' || REGEX_MATCH (COALESCE(p1.path, pe1.path), '.*/(.*)', 1) || ',' || REGEX_MATCH (
    COALESCE(p1_p2.path, pe1_p2.path, pe1_pe2.path),
    '.*/(.*)',
    1
  ) AS exception_key
FROM
  process_events pe
  LEFT JOIN processes p ON pe.pid = p.pid
  -- Parents (via two paths)
  LEFT JOIN processes p1 ON pe.parent = p1.pid
  LEFT JOIN hash p_hash1 ON p1.path = p_hash1.path
  LEFT JOIN process_events pe1 ON pe.parent = pe1.pid
  AND pe1.cmdline != ''
  LEFT JOIN hash pe_hash1 ON pe1.path = pe_hash1.path
  -- Grandparents (via 3 paths)
  LEFT JOIN processes p1_p2 ON p1.parent = p1_p2.pid -- Current grandparent via parent processes
  LEFT JOIN processes pe1_p2 ON pe1.parent = pe1_p2.pid -- Current grandparent via parent events
  LEFT JOIN process_events pe1_pe2 ON pe1.parent = pe1_p2.pid
  AND pe1_pe2.cmdline != '' -- Past grandparent via parent events
  LEFT JOIN hash p1_p2_hash ON p1_p2.path = p1_p2_hash.path
  LEFT JOIN hash pe1_p2_hash ON pe1_p2.path = pe1_p2_hash.path
  LEFT JOIN hash pe1_pe2_hash ON pe1_pe2.path = pe1_pe2_hash.path
WHERE
  pe.time > (strftime('%s', 'now') -300)
  AND pe.cmdline != ''
  AND pe.parent > 0
  AND p0_name IN ('sh', 'fish', 'zsh', 'bash', 'dash')
  AND NOT (
    p1_name IN (
      'abrt-handle-eve',
      'alacritty',
      'at-spi-bus-launcher',
      'BambuStudio',
      'bash',
      'build-script-build',
      'chainctl',
      'chezmoi',
      'clang-11',
      'Code - Insiders Helper (Renderer)',
      'Code Helper (Renderer)',
      'code',
      'collect2',
      'com.docker.backend',
      'conmon',
      'containerd-shim-runc-v2',
      'containerd-shim',
      'cpptools',
      'dash',
      'dbus-run-session',
      'demoit',
      'direnv',
      'doas',
      'Docker Desktop',
      'docker-credential-desktop',
      'docker-credential-gcr',
      'docker',
      'dotnet',
      'dpkg',
      'Emacs-arm64-11',
      'env',
      'erl_child_setup',
      'find',
      'FinderSyncExtension',
      'fish',
      'gatherheaderdoc',
      'gdm-session-worker',
      'gdm-wayland-session',
      'gdm-x-session',
      'gdm3',
      'git',
      'gke-gcloud-auth-plugin',
      'gnome-session-binary',
      'gnome-shell',
      'gnome-terminal-server',
      'go',
      'goland',
      'gopls',
      'helm',
      'HP Diagnose & Fix',
      'i3bar',
      'i3blocks',
      'idea',
      'java',
      'jetbrains-toolbox',
      'kitty',
      'ko',
      'konsole',
      'kubectl',
      'lazygit',
      'lightdm',
      'local-path-provisioner',
      'login',
      'MacVim',
      'make',
      'mc',
      'monorail',
      'my_print_defaults',
      'ninja',
      'nix-build',
      'nix-daemon',
      'nix',
      'nm-dispatcher',
      'node',
      'nu',
      'nvim',
      'obs',
      'package_script_service',
      'pacman',
      'perl',
      'PK-Backend',
      'provisio',
      'pulumi',
      -- 'python' - do not include this, or you won't detect supply-chain attacks.
      'ression-arm64',
      'roxterm',
      'sddm-helper',
      'sdk',
      'sdzoomplugin',
      'sh',
      'ShellLauncher',
      'skhd',
      'snyk-macos',
      'snyk',
      'sshd',
      'stable',
      'Stream Deck',
      'su',
      'sudo',
      'swift',
      'systemd-sleep',
      'systemd',
      'terminator',
      'terraform-ls',
      'terraform',
      'test2json',
      'tmux:server',
      'tmux',
      'update-notifier',
      'vi',
      'vim.nox',
      'vim',
      'Vim',
      'watch',
      'wezterm-gui',
      'xargs',
      'xcrun',
      'xfce4-terminal',
      'xinit',
      'Xorg',
      'xterm',
      'yacls',
      'yay',
      'yum',
      'zed',
      'zellij',
      'zsh'
    )
    OR p1_name LIKE 'terraform-provider-%'
    OR p1_name LIKE 'iTermServer-%'
    -- Do not add shells to this list if you want your query to detect
    -- bad programs that were started from a shell.
    OR p2_name IN ('env', 'git')
    -- Homebrew, except we don't want to allow all of ruby
    OR p0_cmd IN (
      '/bin/bash /usr/bin/xdg-settings set default-url-scheme-handler slack Slack.desktop',
      '/bin/bash /usr/local/bin/mount-product-files',
      '/bin/sh -c black .',
      '/bin/sh -c ioreg -rd1 -c IOPlatformExpertDevice',
      '/bin/sh -c lsb_release -a --short',
      '/bin/sh -c ps ax -ww -o pid,ppid,uid,gid,args',
      '/bin/sh -c scutil --get ComputerName',
      '/bin/sh -c sysctl hw.model kern.osrelease',
      '/bin/sh -c uname -m 2>/dev/null',
      '/bin/sh /usr/bin/lsb_release -a --short',
      '/bin/sh /usr/bin/lsb_release -a',
      '/bin/zsh -c ls',
      '/usr/bin/python3 /usr/bin/terminator',
      'sh -c /Applications/Xcode.app/Contents/Developer/usr/bin/xcodebuild -sdk /Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX.sdk -find python3 2> /dev/null',
      'sh -c /bin/stty size 2>/dev/null',
      'sh -c /usr/bin/xcrun clang 2>&1',
      'sh -c cat /proc/sys/kernel/pid_max',
      'sh -c pactl --version',
      'sh -c python3.7 --version 2>&1',
      'sh -c xcode-select --print-path >/dev/null 2>&1 && xcrun --sdk macosx --show-sdk-path 2>/dev/null',
      "/bin/sh -c defaults delete 'com.cisco.webexmeetingsapp'",
      "sh -c osascript -e 'user locale of (get system info)'",
      "sh -c pacmd list-sinks |grep 'name:\\|module:'"
    )
    OR (
      p1_name = 'WhatsApp'
      -- WhatsApp grabs the serial number from people's machines :(
      AND p0_cmd = '/bin/sh -c ioreg -c IOPlatformExpertDevice -d 2'
    )
    OR (
      p1_name LIKE 'emacs-%'
      AND p1_path LIKE '%/bin/emacs%'
    )
    OR p1_cmd IN (
      '/usr/bin/python3 /usr/share/apport/apport-gtk',
      'php ./autodocs update images'
    )
    OR (
      p1_cmd LIKE '%Python% /opt/homebrew/bin/jupyter%'
      AND p0_cmd = '/bin/sh -c osascript'
    )
    OR (
      p1_name = 'osqueryd'
      AND p0_cmd LIKE '/bin/sh /etc/NetworkManager/dispatcher.d/%'
    )
    OR (
      p1_name = 'ssh'
      AND p0_cmd LIKE '%gcloud.py compute start-iap-tunnel%'
    )
    OR exception_key IN (
      'bash,0,auditd,launchd',
      'bash,0,etcd,containerd-shim-runc-v2',
      'bash,0,kube-apiserver,containerd-shim-runc-v2',
      'bash,0,mutter-x11-frames,gnome-shell',
      'bash,0,perl5.30,system_installd',
      'bash,0,pia-daemon,launchd',
      'bash,0,udevadm,udevadm',
      'bash,500,.man-wrapped,zsh',
      'bash,500,accounts-daemon,systemd',
      'bash,500,busybox,bwrap',
      'bash,500,bwrap,bwrap',
      'bash,500,com.docker.dev-envs,com.docker.backend',
      'bash,500,docker-builder,bash',
      'bash,500,Foxit PDF Reader,launchd',
      'bash,500,gdb,perl',
      'bash,500,gnome-session-binary,systemd',
      'bash,500,gpg-agent,launchd',
      'bash,500,Hyprland,gdm-wayland-session',
      'bash,500,incusd,incusd',
      'bash,500,lazygit,nvim',
      'bash,500,plasmashell,systemd',
      'bash,500,Private Internet Access,launchd',
      'bash,500,ruby,zsh',
      'bash,500,screen,screen',
      'bash,500,script,bash',
      'bash,500,steam,bash',
      'bash,500,xdg-desktop-portal,systemd',
      'bash,500,xdg-permission-store,systemd',
      'dash,0,anacron,systemd',
      'dash,0,dpkg,apt',
      'dash,0,dpkg,python3.10',
      'dash,0,kindnetd,containerd-shim-runc-v2',
      'dash,0,kube-proxy,containerd-shim-runc-v2',
      'dash,0,run-parts,dash',
      'dash,0,snapd,systemd',
      'dash,500,gdm-wayland-session,gdm-session-worker',
      'dash,500,python3.12,firefox-bin',
      'sh,0,auditd,launchd',
      'sh,0,Ecamm Live,launchd',
      'sh,0,expect,kandji-daemon',
      'sh,500,cloud_sql_proxy,zsh',
      'sh,500,docs,zsh',
      'sh,500,Google Drive,launchd',
      'sh,500,LogiTune,launchd',
      'sh,500,Meeting Center,launchd',
      'sh,500,snyk-macos,snyk',
      'sh,500,splunkd,splunkd',
      'sh,500,ssh,Code Helper (Plugin)',
      'sh,500,ssh,mosh-client',
      'sh,500,updater,Foxit PDF Reader',
      'sh,500,viddy,zsh',
      'sh,500,yabai,launchd',
      'zsh,500,Hyper,launchd',
      'zsh,500,old,launchd',
      'zsh,500,old,old',
      'zsh,500,OpenLens,launchd',
      'zsh,500,pycharm,launchd',
      'zsh,500,python3.10,gnome-shell',
      'zsh,500,rubymine,launchd',
      'zsh,500,stable,launchd'
    )
    OR p0_cmd LIKE '/bin/bash /Users/%/homebrew/Library/Homebrew/shims/shared/curl %'
    OR p0_cmd LIKE '/bin/bash /usr/bin/xdg-settings check %'
    OR p0_cmd LIKE '/bin/bash /usr/local/Homebrew/%'
    OR p0_cmd LIKE '/bin/bash %/opt/homebrew/%'
    OR p0_cmd LIKE '/bin/bash %git credential-osxkeychain get'
    OR p0_cmd LIKE '/bin/sh -c /Applications/%'
    OR p0_cmd LIKE '/bin/sh -c pkg-config %'
    OR p0_cmd LIKE '/bin/sh /usr/bin/xdg-open %'
    OR p0_cmd LIKE '/bin/sh /usr/bin/xdg-settings check %'
    OR p0_cmd LIKE '/bin/sh /usr/bin/xdg-settings get %'
    OR p0_cmd LIKE '/bin/sh /usr/bin/xdg-settings set %'
    OR p0_cmd LIKE '/bin/sh %/bin/gcloud%config config-helper%'
    OR p0_cmd LIKE '/bin/sh %/bin/xvim block %'
    OR p0_cmd LIKE '/bin/sh %/docker-credential-gcloud get'
    OR p0_cmd LIKE '/bin/sh %/google-cloud-sdk/bin/gcloud config get project'
    OR p0_cmd LIKE '%/bash -e%/bin/as -arch%'
    OR p0_cmd LIKE '%/google-chrome% --flag-switches-begin % --product-version'
    OR p0_cmd LIKE '%/usr/bin/python3 /usr/bin/terminator%'
    OR p0_cmd LIKE '%gcloud config config-helper --format=json'
    OR p0_cmd LIKE '%gcloud config get-value%'
    OR p0_cmd LIKE '%sh -c ntia-checker %'
    OR p1_cmd LIKE '/%google-cloud-sdk/lib/gcloud.py%'
    OR p1_cmd LIKE '/System/Library/Frameworks/Ruby.framework/Versions/2.6/usr/bin/ruby -W1 --disable=gems,rubyopt -- /Users/%/homebrew/Library/Homebrew/build.rb%'
    OR p1_cmd LIKE '%/bin/pipenv shell'
    OR p1_cmd LIKE '%/usr/bin/terminator%'
    OR p1_cmd LIKE 'gcloud% auth%login%'
    OR (
      exception_key IN ('sh,500,ruby,zsh', 'bash,500,ruby,zsh')
      AND p1_cmd LIKE '%brew.rb'
    )
    OR (
      exception_key = 'sh,500,ruby,ruby'
      AND p1_cmd LIKE '%homebrew%'
    )
    OR p1_cmd LIKE '%Python /opt/homebrew/bin/aws configure sso'
    OR p2_cmd LIKE '/bin/bash /usr/local/bin/brew%'
    OR p2_cmd LIKE '/usr/bin/python3 -m py_compile %'
  )
  AND NOT p0_cgroup LIKE '/system.slice/docker-%'
  AND NOT p1_cgroup LIKE '/system.slice/docker-%'
  AND NOT p2_cgroup LIKE '/system.slice/docker-%'
GROUP BY
  pe.pid

tags: SStagSS

Reference

https://github.com/chainguard-dev/osquery-defense-kit