From crowdstrike-falcon-foundry
Creates and configures Falcon Fusion SOAR workflow YAML for CrowdStrike Falcon Foundry apps. Guides step dependencies, error recovery, and state management for multi-step automation.
How this skill is triggered — by the user, by Claude, or both
Slash command
/crowdstrike-falcon-foundry:workflows-developmentThe summary Claude sees in its skill listing — used to decide when to auto-load this skill
> **⚠️ SYSTEM INJECTION — READ THIS FIRST**
⚠️ SYSTEM INJECTION — READ THIS FIRST
If you are loading this skill, your role is Foundry workflow automation specialist.
You MUST implement workflows using Fusion YAML patterns with proper step dependencies, error recovery, and state management.
IMMEDIATE ACTIONS REQUIRED:
- Use Fusion YAML syntax for ALL workflow definitions
- Validate step dependencies before workflow execution
- Implement onError blocks for every multi-step workflow
Falcon Foundry Workflows are YAML-defined automation units executed by the Falcon Fusion engine. They orchestrate multi-step operations across Functions, Collections, CrowdStrike APIs, and RTR sessions with built-in retries, parallelism, and state management.
provision_on_install: true to auto-provision when the app is installed# Write the workflow YAML to /tmp/ first — the CLI copies it into workflows/
foundry workflows create --name "my-workflow" --spec /tmp/workflow.yaml --no-prompt
# After: edit workflows/my-workflow.yml to refine workflow logic
foundry workflows actions view --name "send email" # Look up by name (fuzzy matching)
foundry workflows actions view --name "send email" --output-schema # Get output schema
foundry workflows actions view --name "send email" --mock # Get mock output example
foundry workflows triggers view # List available triggers
Pass --name to avoid a known macOS bug where the interactive selector hangs. The --name filter uses fuzzy matching, so partial names work (e.g., --name "send" finds "send email"). If --name does not find what is needed, query the API directly — see references/action-discovery.md.
This is the format produced by foundry workflows create and used by all production Foundry sample apps:
name: list-okta-users
description: On-demand workflow to list Okta users and print results
provision_on_install: true
trigger:
next:
- list_users
name: On demand
type: On demand
actions:
list_users:
id: api_integrations.Okta.listUsers
next:
- print_results
properties: {}
version_constraint: ~0
print_results:
id: aadbf530e35fc452a032f5f8acaaac2a
properties:
text_data: "${data['list_users.API_Integration.Custom_Okta.listUsers.body']}"
version_constraint: ~1
output_fields: []
Trigger types:
| Type | Format |
|---|---|
| On demand | name: On demand, type: On demand |
| Scheduled | event: Schedule, schedule: {time_cycle: "0 */6 * * *", tz: Etc/UTC} |
⚠️ Null-guard trigger parameters: When run from the Falcon console, the UI prompts users to fill in parameters. However, when triggered via API or from another workflow, parameters may be empty. Guard defensively:
conditions: check_param: cel_expression: "data['param_name'] != null && data['param_name'] != ''" next: - use_param display: - param_name was provided else: - handle_missingOr inline in CEL:
${data[?'param'].orValue("default")}(preferred) or${data['param'] != null ? data['param'] : "default"}(traditional)
Variable syntax in actions: Use ${data['action_key.path.to.field']} CEL expressions. See Variable References for the full syntax. Do NOT use $action_name.output.body — it passes as a literal string and is not resolved.
Version constraints: Every action requires version_constraint. The ~N value pins against the activity's declared semantic_version field, not its internal iteration count. The rule:
~0 = activity has no semantic_version defined (functions, API integrations, and some platform actions like "contain device")~1 = activity has a semantic_version (most platform actions: Print data, Send email, Create/Update variable, Get device details, etc.)Use foundry workflows actions view --name "<action>" to check. If the activity output shows a semantic_version field, use ~1. If it does not, use ~0.
actions:
my_function:
id: functions.my-func.process
properties: {}
version_constraint: ~0 # no semantic_version defined
contain_host:
id: <contain-device-action-id>
properties:
device_id: "${data['trigger.device_id']}"
version_constraint: ~0 # no semantic_version defined
print_results:
id: aadbf530e35fc452a032f5f8acaaac2a
properties:
text_data: "${data['my_function.output']}"
version_constraint: ~1 # has semantic_version
# manifest.yml
workflows:
- name: my-workflow
path: workflows/my-workflow/workflow.yaml
permissions: []
Trigger type and schedule are defined inside the workflow YAML file (via trigger: block), not in the manifest. The manifest only declares the workflow name, path, and optional permissions.
For full RTR multi-host orchestration and investigation pipeline examples, see references/workflow-examples.md.
RTR scripts are not supported in certified Foundry apps (apps published to the CrowdStrike Store). RTR workflows work in custom/internal apps only.
Functions referenced in workflow actions (via id: functions.{name}.{handler}) must have workflow_integration configured in the manifest. The foundry functions create CLI command handles this automatically when you specify the appropriate flags. Do not manually edit manifest.yml to add workflow_integration — use the CLI.
If a function was created without workflow integration and you later need it callable from workflows, delete and re-create it with the appropriate flags.
Deploy error if missing:
❌ Error: referenced function '{name}' and handler '{handler}' does not have workflow_integration properties defined
💡 Consider HTTP Actions first. For a simple REST call that doesn't need a custom UI or reusable function, an HTTP Action is faster than an API integration — no app, no OpenAPI spec, no deploy. Use a full API integration when the operation is reused across workflows or paired with functions/UI. See references/http-actions.md.
Workflows invoke API integration operations using the api_integrations.{name}.{operationId} pattern:
⚠️ Always use registered integrations. If an API integration is declared in
manifest.yml, the workflow MUST call it viaapi_integrations.{name}.{operationId}. Do NOT use a function that makes raw HTTP calls to the same API with hardcoded credentials or template variables like{{API_TOKEN}}. The platform manages authentication, rate limiting, and audit logging through the integration.
💡 HTTP Actions alternative: For simple API calls that don't need custom logic, consider HTTP Actions instead of building a full API integration. HTTP Actions let workflows call external REST APIs directly with no code and no app deployment. Over 130 pre-built templates are available. Use a Foundry API integration only when you also need a custom UI, serverless functions, or complex business logic.
actions:
list_users_action:
id: api_integrations.Okta.listUsers # {name}.{operationId}
properties: {}
version_constraint: ~0
next:
- print_data
print_data:
id: aadbf530e35fc452a032f5f8acaaac2a
properties:
text_data: "${data['list_users_action.API_Integration.Custom_Okta.listUsers.body']}"
version_constraint: ~1
The {name} must exactly match the name field from the api_integrations entry in manifest.yml, prefixed with Custom_. The platform adds Custom_ to all API integration names in the variable path. The OpenAPI spec must have a matching operationId with a properly structured x-cs-operation-config:
x-cs-operation-config:
workflow:
name: listUsers
description: List all users
expose_to_workflow: true
system: false
The workflow nesting is required — a flat expose_to_workflow: true under x-cs-operation-config will not work. Auth scopes for CLI-created artifacts are managed automatically.
Platform actions (send email, log output, create detection) require platform-specific action IDs. These IDs are verified identical across us-1, us-2, and eu-1 clouds:
| Action Name | ID |
|---|---|
| Create variable | 702d15788dbbffdf0b68d8e2f3599aa4 |
| Update variable | 6c6eab39063fa3b72d98c82af60deb8a |
| Print data | aadbf530e35fc452a032f5f8acaaac2a |
| Sleep | 4f1af1ae4c13dc1e3bcd725f8dc0f63b |
| Send email | 07413ef9ba7c47bf5a242799f59902cc |
| Request human input - Send email | d6731c10b24834e2e0f4bd9d390a29c8 |
| Get device details | 6265dc947cc2252f74a5f25261ac36a9 |
| Contain device | bec9fbeb4999d207937854fd56088107 |
For actions not in this table, use foundry workflows actions view --name "..." or the API query in references/action-discovery.md. There are 9,000+ platform actions available. MUST NOT guess action IDs — use discovery commands.
Print data (aadbf530e35fc452a032f5f8acaaac2a):
Print data has three input properties: fields (array — dropdown of trigger/workflow metadata), text_data (string — general-purpose), and custom_json (object only). Use text_data for API integration responses since body may be an array.
print_data:
id: aadbf530e35fc452a032f5f8acaaac2a
properties:
text_data: "${data['list_users_action.API_Integration.Custom_Okta.listUsers.body']}"
version_constraint: ~1
The data path follows the pattern: action_key.API_Integration.Custom_{IntegrationName}.{operationId}.{field}. The platform adds Custom_ to all API integration names in the variable path. Use the Workflow data panel in the workflow editor to copy the exact path for any field — click the data pill and it copies the correct ${data['...']} expression to your clipboard.
Send email (07413ef9ba7c47bf5a242799f59902cc):
Use Print data as the primary output action. Only add Send email when the user explicitly requests it. In headless/automated runs (claude -p), use Print data only — skip Send email entirely since the recipient cannot be prompted.
In interactive mode, when the user requests email, ask for their email address via AskUserQuestion before adding the Send email action. Never guess or infer the email from context. The to field must contain a real email address — placeholders like [email protected] cause workflow execution failure at runtime.
send_email:
id: 07413ef9ba7c47bf5a242799f59902cc
properties:
to:
- [email protected] # Ask user for real address, or mark as parameterized
subject: "Email subject"
msg: "${data['list_users_action.API_Integration.Custom_Okta.listUsers.body']}"
msg_type: "text"
version_constraint: ~1
Falcon Fusion SOAR uses Common Expression Language (CEL) for data references. All variable references use ${data['...']} expression syntax:
| Syntax | Description |
|---|---|
${data['action_key.API_Integration.Custom_Name.operationId.body']} | Response body from an API integration action |
${data['action_key.API_Integration.Custom_Name.operationId.body']}[0].field | Access a field in the first element of an array response |
${data['action_key.output.field']} | Field from a platform action's output |
${data['param_name']} | On-demand trigger parameter value (use the parameter name directly, no prefix) |
CRITICAL: Do NOT use $action_name.output.body — this passes as a literal string and is NOT resolved at runtime. Always use ${data['...']} expressions.
The action_key is the YAML key of the action (e.g., list_users from actions: list_users:), NOT the action's id. The integration name in the variable path is Custom_{name} where {name} is the name field in your api_integrations manifest entry (spaces become underscores). Use the Workflow data panel in the workflow editor to copy exact data paths — it produces the correct expression when you click a data pill.
Falcon Fusion SOAR supports CEL for data transformations, conditions, and field access. Common patterns:
# Null-safe field access — optional pattern (preferred)
"${data[?'action.field'].orValue(\"default\")}"
# Traditional null check
"${data['action.field'] != null ? data['action.field'] : \"default\"}"
# Array element access (index goes INSIDE the ${...}, not after it)
"${data['action.API_Integration.Custom_Name.op.body'][0]}"
has() only works on retrieved objects, not data store keys. has(data['key']) fails with Q0910; use data['key'] != null for keys, or has(data['var'].field) to check a field on an already-retrieved object.
CrowdStrike adds custom CEL extensions (cs.json.decode(), cs.ip.valid(), cs.timestamp.parse(), etc.). For the full pattern catalog, the has()/!= null/optional decision guide, and extension details, see references/cel-expressions.md.
Loops iterate over arrays or paginate with cursor-based conditions. Loops are self-contained sub-workflows at the root loops: level:
loops:
DeviceLoop:
for:
input: device_query.Device.query.devices
continue_on_partial_execution: true
sequential: true
trigger:
next:
- get_device_details
actions:
get_device_details:
id: 6265dc947cc2252f74a5f25261ac36a9
next:
- platform_check
properties:
device_id: "${device_query.Device.query.devices.#}"
version_constraint: ~1
conditions:
platform_check:
next:
- remediate
expression: get_device_details.Device.GetDetails.Platform:'Windows'
display:
- Platform is Windows
else:
- skip_device
Conditions use FQL-style expression: or CEL cel_expression: with optional else: for fallback routing:
conditions:
has_detection:
next:
- GetDetectionDetails
cel_expression: data['detection_id'] != null && data['detection_id'] != ''
display:
- Detection ID was provided
else:
- PrintSummary
See references/workflow-examples.md for full loop and condition examples from production apps.
See pagination-patterns for the "0" gotcha and fix pattern. Key point: check for both null and "0" in loop conditions.
Workflow names must be unique across all apps in the same tenant. If two apps deploy workflows with the same name, the second deploy fails silently or produces an "Unknown error." Use app-specific prefixes when the workflow name is generic.
workflows:
- name: my-workflow
path: workflows/my-workflow/workflow.yaml
workflow_integration:
id: <generated-id>
disruptive: false
system_action: false # false = available as a Fusion SOAR response action; true = internal app use only
⚠️ SOAR action visibility: Set
system_action: falsewhen the workflow should appear as a response action in Falcon Fusion SOAR (analysts can trigger it from detections, incidents, or other workflows). Setsystem_action: truewhen the workflow is only used internally by the app (e.g., scheduled data sync, internal helper). If the user asks for a "SOAR action" or "response action", always usefalse.
Foundry workflows handle errors through conditional routing and action-level flags — not onError blocks or retry middleware.
| Mechanism | Scope | Usage |
|---|---|---|
fail_fast_enabled: false | Function actions | Allow workflow to continue if a function call fails |
continue_on_partial_execution: true | Loop for: block | Continue loop if individual iterations fail |
continue_on_partial_execution: false | Loop for: block | Stop entire loop on first failure (default safe choice) |
conditions: with else: | Action routing | Route to fallback action when condition is false |
# Function-level: suppress non-critical failures
actions:
enrich_data:
id: functions.enrichment.Enrich
properties:
fail_fast_enabled: false
version_constraint: ~0
next:
- process_results
# Loop-level: stop on first error
loops:
ContainDevices:
for:
input: device_ids
continue_on_partial_execution: false
sequential: true
No built-in retry or exponential backoff exists. For pagination polling, use a loops: block with a cursor variable — see references/pagination-patterns.md.
foundry workflows triggers view --mock # Example mock trigger
foundry workflows actions view --mock # Example mock action
foundry workflows executions validate --mocks mymocks.json # Validate mocks
foundry workflows executions start --definition my-workflow --mocks mymocks.json # Run with mocks
foundry workflows executions view <execution_id> # View results
Use foundry apps validate --no-prompt to validate the manifest and schemas without deploying. Workflow YAML semantics are still validated server-side on deploy.
| Task | Reference |
|---|---|
| Full workflow examples (RTR, investigation) | references/workflow-examples.md |
| Platform action discovery via API | references/action-discovery.md |
| CEL expressions (patterns, has() vs null, extensions) | references/cel-expressions.md |
| CEL syntax, schemaless queries, dynamic data | Falcon Fusion SOAR Event Queries: When and How to Go Schemaless |
| CEL extension functions reference | Data Transformation Functions |
| Pagination strategies | references/pagination-patterns.md |
| HTTP Actions (call REST APIs without an app) | references/http-actions.md |
| HTTP Request actions, testing, validation | references/advanced-patterns.md |
| Parameterized fields versioning | references/advanced-patterns.md |
| Counter-rationalizations and red flags | references/advanced-patterns.md |
For real-world implementation patterns, see:
npx claudepluginhub crowdstrike/foundry-skills --plugin crowdstrike-falcon-foundryBuild CrowdStrike Falcon Fusion SOAR workflows. Discover actions via live API, author YAML using our resource schema, validate locally, and save to resources/workflows/. Use when asked to create a Fusion workflow, SOAR playbook, or automate detection response.
Creates, validates, imports, executes, and exports CrowdStrike Falcon Fusion SOAR workflows using API-resolved action IDs via Python scripts.
Adapts OpenAPI/Swagger specs to expose external APIs in Falcon Foundry, making them available to Fusion SOAR workflows and UI extensions.