| name | dtctl |
| description | Investigate incidents, debug performance issues, analyze logs, and manage observability resources in Dynatrace using the dtctl CLI. Use this skill whenever the user asks about error rates, latency spikes, service health, crash-looping pods, web vitals, SLO status, open problems, root cause analysis, log patterns, trace analysis, or building dashboards — even if they don't mention Dynatrace by name. Also covers DQL queries, workflow management, notebook and dashboard creation, settings configuration, and any operations against a Dynatrace environment. |
Dynatrace Control with dtctl
Operate dtctl, the kubectl-style CLI for Dynatrace. This skill teaches core dtctl command patterns and operations.
Recommended Initialization
At the start of a task, run these checks to establish context and permissions:
dtctl commands --brief -o json
dtctl config current-context
dtctl config describe-context $(dtctl config current-context) --plain
dtctl auth whoami --plain
This displays:
- Current context name and environment URL
- Safety level (readonly, readwrite-mine, readwrite-all, dangerously-unrestricted)
- Authenticated user identity (name, email, UUID)
DQL Reference Usage
Before writing, modifying, or executing any DQL that fetches Dynatrace data (for example via dtctl query, dtctl wait query, or query files), you MUST consult references/DQL-reference.md and follow its documented syntax and templates.
If there is any conflict between memory/assumptions and the reference, prefer the reference.
Prerequisites
If dtctl is not installed or not working, see references/troubleshooting.md for installation and setup.
Resources & Commands
Available Resources
dtctl uses a uniform pattern for all resource types. Discover schema from actual output with dtctl describe <resource> <id> -o json --plain.
| Resource | Aliases |
|---|
| analyzer | analyzers |
| anomaly-detector | anomaly-detectors |
| app | apps |
| bucket | bkt |
| copilot-skill | copilot-skills |
| dashboard | dash |
| edgeconnect | ec |
| extension | ext, extensions |
| extension-config | extcfg, extension-configs |
| function | fn, func |
| group | groups |
| intent | intents |
| lookup | lookups, lkup |
| notebook | nb |
| notification | notifications |
| sdk-version | sdk-versions |
| settings | setting |
| settings-schema | schema |
| slo | - |
| slo-template | slo-templates |
| trash | deleted |
| user | users |
| workflow | wf |
| workflow-execution | wfe |
Use IDs whenever possible instead of names to avoid ambiguity.
Command Verbs
| Verb | Description | Example |
|---|
| get | List resources | dtctl get workflows --mine |
| describe | Show resource details | dtctl describe workflow <id> |
| edit | Edit resource interactively | dtctl edit dashboard <id> |
| apply | Create/update from file | dtctl apply -f workflow.yaml --set env=prod |
| delete | Delete resource | dtctl delete workflow <id> |
| exec | Execute workflow/function/analyzer/copilot | dtctl exec workflow <id> |
| query | Run DQL query | dtctl query "fetch logs | limit 10" |
| logs | Print resource logs | dtctl logs workflow-execution <id> |
| wait | Wait for conditions | dtctl wait query "fetch logs" --for=any |
| history | Show document history | dtctl history dashboard <id> |
| restore | Restore document version | dtctl restore dashboard <id> --version 3 |
| share | Share document | dtctl share dashboard <id> --user email@example.com |
| unshare | Remove sharing | dtctl unshare dashboard <id> --user email@example.com |
| find | Discover resources | dtctl find intents --data trace.id=abc |
| open | Open in browser | dtctl open intent <app/intent> --data key=value |
| diff | Compare resources | dtctl diff -f workflow.yaml |
| verify | Validate without executing | dtctl verify query 'fetch logs' --fail-on-warn |
| commands | List all commands (machine-readable) | dtctl commands --brief -o json |
Key Concepts for AI Agents
Output Modes
-A, --agent
--no-agent
-o json
-o yaml
-o csv
-o chart
-o sparkline
-o barchart
-o table
-o wide
--plain
For AI agents, prefer: dtctl <command> --agent (auto-detected) or dtctl <command> -o json --plain
The --agent envelope provides structured metadata alongside results:
{
"ok": true,
"result": [ ... ],
"context": {
"verb": "get", "resource": "workflow",
"total": 5, "has_more": false,
"suggestions": ["Run 'dtctl describe workflow <id>' for details"]
}
}
Template Variables
In YAML/DQL files, use Go template syntax:
title: "{{.environment}} Deployment"
owner: "{{.team}}"
trigger:
schedule:
cron: "{{.schedule | default "0 0 * * *"}}"
# query.dql
fetch logs
| filter host.name == "{{.host}}"
| filter timestamp > now() - {{.timerange | default "1h"}}
Execute with: dtctl apply -f file.yaml --set environment=prod --set team=platform
Copilot, Functions, Analyzers
dtctl get copilot-skills -o json --plain
dtctl get functions -o json --plain
dtctl exec function <id-or-name> --payload '{"key":"value"}' --plain
dtctl get analyzers -o json --plain
dtctl exec analyzer <id-or-name> --input '{"timeframe":"now-2h"}' --plain
dtctl get anomaly-detector <id> -o yaml --plain > detector.yaml
dtctl apply -f detector.yaml --context <target-context> --plain
Prefer get ... -o json --plain first, then describe/exec with explicit IDs.
Authentication & Permissions
dtctl auth whoami --plain
dtctl auth can-i create workflows
dtctl auth can-i delete dashboards
Use can-i to verify permissions before attempting operations.
Quick Reference: DQL Queries
Required workflow for DQL data fetching:
- First consult
references/DQL-reference.md
- Build/validate the query using the documented patterns
- Execute with
dtctl query ... -o json --plain (or dtctl wait query ... when waiting for results)
dtctl query "fetch logs | filter status='ERROR' | limit 100" -o json --plain
dtctl query -f query.dql --set host=h-123 --set timerange=2h -o json --plain
dtctl wait query "fetch spans | filter test_id='test-123'" --for=count=1 --timeout 5m
dtctl query "timeseries avg(dt.host.cpu.usage)" -o chart --plain
Dashboards
For full examples and field-level gotchas, see references/resources/dashboards.md.
Create/update: dtctl apply -f dashboard.yaml --plain. Export for reference: dtctl get dashboard <id> -o yaml --plain.
YAML skeleton
name: "Dashboard Name"
type: dashboard
content:
annotations: []
importedWithCode: false
settings:
defaultTimeframe:
enabled: true
value: { from: now()-2h, to: now() }
layouts:
"1":
x: 0
"y": 0
w: 12
h: 6
tiles:
"1":
title: "Tile Title"
type: data
query: |
fetch logs | limit 10
visualization: lineChart
visualizationSettings:
autoSelectVisualization: false
davis: { enabled: false, davisVisualization: { isAvailable: true } }
Tile types & visualizations
type: data — DQL tile with query + visualization: singleValue, lineChart, areaChart, barChart, pieChart, table, honeycomb, scatterplot
type: markdown — static text via content field (supports markdown)
For detailed visualizationSettings (singleValue, charts, tables, thresholds, unit overrides), see references/resources/dashboards.md.
Gotchas
- Always set
davis.enabled: false on data tiles.
- Use
makeTimeseries for log/span time series; timeseries for metrics.
version field warning on create is benign.
- No
id field → creates new; with id field → updates existing.
Common Issues
Name resolution ambiguity:
- If a name matches multiple resources, dtctl will fail
- Solution: Use IDs instead of names
- Find ID:
dtctl get <resource> -o json --plain | jq -r '.[] | "\(.id) | \(.name)"'
Permission denied:
Context/safety blocks:
- Destructive operations may be blocked by safety level
- Switch context:
dtctl config use-context <name>
- Adjust safety level when creating context
Additional Resources
Safety Reminders
- Use
--plain for machine/AI consumption
- Confirm context + safety level before destructive ops; prefer
get/describe first
- Use
--mine flag to filter resources you own
- For multi-tenant work, see references/config-management.md