aws-wtf

SKILL.md

AWS WTF Skill for Opsy

Explains every charge on your AWS bill — what it is, why you're paying, and what resource caused it.

⚠️ Mandatory Output

You MUST automatically generate and save a CSV file at the end of every analysis. Do not wait for the user to ask. The analysis is incomplete until the CSV exists.

Step 1: Cost Explorer (Two Queries Required)

Run BOTH queries to detect credit coverage:

# Query 1: Normal (shows $0 if credits cover costs)
aws ce get-cost-and-usage \
  --time-period Start=$(date -v-30d +%Y-%m-%d),End=$(date +%Y-%m-%d) \
  --granularity MONTHLY \
  --metrics "UnblendedCost" "UsageQuantity" \
  --group-by Type=DIMENSION,Key=SERVICE

# Query 2: Exclude credits (shows ACTUAL usage cost)
aws ce get-cost-and-usage \
  --time-period Start=$(date -v-30d +%Y-%m-%d),End=$(date +%Y-%m-%d) \
  --granularity MONTHLY \
  --metrics "UnblendedCost" "UsageQuantity" \
  --group-by Type=DIMENSION,Key=SERVICE \
  --filter '{"Not": {"Dimensions": {"Key": "RECORD_TYPE", "Values": ["Credit", "Refund"]}}}'

Interpretation:

  • Query 1 = $0, Query 2 = $X → Credits covering $X actual usage
  • Query 1 = Query 2 = $0 → Real free tier
  • Query 1 = Query 2 = $X → Normal billing

If credits detected, warn user: "Your bill shows $0 but actual usage is $X/month. When credits run out, you WILL be charged."

Step 2: Identify ALL Regions

aws ce get-cost-and-usage \
  --time-period Start=$(date -v-30d +%Y-%m-%d),End=$(date +%Y-%m-%d) \
  --granularity MONTHLY \
  --metrics "UnblendedCost" \
  --group-by Type=DIMENSION,Key=REGION \
  --filter '{"Not": {"Dimensions": {"Key": "RECORD_TYPE", "Values": ["Credit", "Refund"]}}}'

You MUST enumerate resources in EVERY region showing charges > $0.01.

Step 3: Enumerate ALL Resources

For each region with charges, query ALL applicable services using --region $REGION:

  • aws ec2 describe-instances
  • aws ec2 describe-volumes
  • aws ec2 describe-snapshots --owner-ids self
  • aws ec2 describe-addresses
  • aws ec2 describe-nat-gateways
  • aws rds describe-db-instances
  • aws elbv2 describe-load-balancers
  • aws ecs list-clustersdescribe-clusterslist-serviceslist-tasks
  • aws eks list-clustersdescribe-cluster
  • aws lambda list-functions
  • aws s3api list-buckets (global) → get-bucket-location per bucket
  • aws ecr describe-repositories
  • aws secretsmanager list-secrets
  • aws logs describe-log-groups
  • aws kms list-keys
  • aws route53 list-hosted-zones (global)

ARN Construction

For resources without ARN in response, construct: arn:aws:{service}:{region}:{account}:{resource-type}/{id}

Examples:

  • EC2: arn:aws:ec2:us-east-1:123456789012:instance/i-abc123
  • EBS: arn:aws:ec2:us-east-1:123456789012:volume/vol-abc123
  • S3: arn:aws:s3:::bucket-name

⚠️ CRITICAL: Every Charge Must Have Identification

Every row in the CSV MUST have a resource identifier (ARN + resource_id) UNLESS it is truly untraceable.

What CAN Be Traced (MUST have ARN)

ANY charge from these services MUST be traced to a specific resource:

Service Has ARN Example
EC2 ✅ Always Instance, Volume, Snapshot, EIP, NAT Gateway
S3 ✅ Always Bucket
RDS ✅ Always Instance
ECS/EKS ✅ Always Cluster, Service, Task
Lambda ✅ Always Function
ALB/NLB ✅ Always Load Balancer
CloudWatch Logs ✅ Always Log Group
Secrets Manager ✅ Always Secret
KMS ✅ Always Key
ECR ✅ Always Repository
Route 53 ✅ Always Hosted Zone

If Cost Explorer shows charges for these but you can't find the resource → it was deleted mid-period. Put ARN as DELETED - {service} and note in description.

What CANNOT Be Traced (N/A allowed)

Only these charges are truly untraceable to a single resource:

Charge Type Why Untraceable
Data Transfer Out Aggregated from multiple sources
Data Transfer Inter-Region No single source
Data Transfer Inter-AZ No single source
Support Plan Account-level
Tax Account-level
CloudWatch Custom Metrics (aggregated) No single dimension

For these only: use arn: N/A - Service-level charge or N/A - Account-level charge

Verification Rule

Before marking ANY charge as N/A, ask: "Is there a specific AWS resource that caused this?"

  • If YES → find it, get its ARN
  • If NO (only data transfer, support, tax) → N/A is acceptable

Elastic IP Verification

Check AssociationId before calling an IP "unattached":

  • AssociationId present → attached (even if InstanceId is empty)
  • NetworkInterfaceOwnerId = "amazon-..." → service-managed (ALB, RDS, NAT)

Public IPv4 Charges

AWS charges $0.005/hr ($3.60/mo) per public IPv4. Find all sources:

  • EC2 public IPs, Elastic IPs, internet-facing ALBs, NAT Gateways

CSV Output (Mandatory)

Filename: aws-wtf-{account-id}-{date}.csv

account_id,resource_name,charge_category,charge_explanation,monthly_cost_usd,status,resource_id,arn,region,resource_type,tags,description

Column Definitions

Column Description
account_id AWS account ID
resource_name Name tag (empty if untagged)
charge_category Compute, Storage, Database, Networking, Container, Serverless, Monitoring, DNS, Security, Data Transfer, Support, Tax
charge_explanation What you're paying for: EC2 t3.small, EBS gp3 20GB, ALB hourly
monthly_cost_usd Actual cost (not $0 even if credit-covered)
status Billed, Free-Tier, Credit-Offset
resource_id AWS resource ID or N/A for non-resource charges
arn Full ARN or N/A - Service-level charge
region AWS region or global
resource_type EC2, EBS, RDS, S3, Lambda, etc.
tags key=value,key=value
description Cost breakdown: 720 hrs × $0.10/hr, details

One Row Per Charge Type

A resource can have multiple rows:

  • EC2: Compute hours + Public IPv4
  • RDS: Instance hours + Storage
  • ALB: Hourly + LCU + Public IPv4
  • ECS Fargate: vCPU + Memory

Example

account_id,resource_name,charge_category,charge_explanation,monthly_cost_usd,status,resource_id,arn,region,resource_type,tags,description
550435500798,api-server,Compute,EC2 t3.small,15.18,Credit-Offset,i-abc123,arn:aws:ec2:us-east-1:550435500798:instance/i-abc123,us-east-1,EC2,"Env=prod","720 hrs × $0.0211/hr"
550435500798,api-server,Networking,Public IPv4,3.60,Credit-Offset,i-abc123,arn:aws:ec2:us-east-1:550435500798:instance/i-abc123,us-east-1,EC2-IPv4,,"720 hrs × $0.005/hr"
550435500798,/aws/lambda/func,Monitoring,CloudWatch Logs,0.45,Credit-Offset,/aws/lambda/func,arn:aws:logs:us-east-1:550435500798:log-group:/aws/lambda/func,us-east-1,CloudWatch-Logs,,"15GB × $0.03/GB"
550435500798,,Data Transfer,Egress to Internet,12.00,Billed,N/A,N/A - Service-level charge,us-east-1,DataTransfer,,"133GB × $0.09/GB"

Summary Report

## AWS Bill Breakdown
**Account:** {id} | **Period:** {start} to {end}

### Cost Summary
| Metric | Amount |
|--------|--------|
| Actual Usage | $XXX |
| Credits Applied | -$XXX |
| **You Pay** | **$X.XX** |

### By Category
| Category | Amount | % |
|----------|--------|---|
| Compute | $XX | X% |
| Storage | $XX | X% |
...

### Top Charges
| Resource | Type | Cost | Description |
|----------|------|------|-------------|
| {name} | {type} | $XX | {explanation} |

Completion Checklist

  • Cost Explorer: both queries (with/without credits)
  • All regions identified and enumerated
  • All resources listed with ARNs
  • CSV file saved
  • Summary shown to user

Analysis is INCOMPLETE until CSV file exists.

Weekly Installs
6
Repository
opsyhq/opsy
First Seen
Jan 22, 2026
Installed on
claude-code4
opencode3
cursor3
codex3
antigravity3
gemini-cli3