Documentation

Recommendations

Cost optimization recommendations generated from usage patterns, idle detection, and best practices — 337+ audit rules across AWS, GCP, and Azure with workflow remediation.

ZopNight analyzes your cloud resources and generates cost optimization recommendations based on usage patterns, idle detection, and best practices.

How Recommendations Work

The recommender service subscribes to resource discovery events via Redis Streams. When new or updated resources are discovered, it runs audit rules across all three cloud providers to generate actionable recommendations.

Recommendation Categories

CategoryDescription
idleResources with low or no utilization that could be stopped or terminated
rightsizingResources that are over-provisioned and could use a smaller instance type
scheduleResources that would benefit from a start/stop schedule or autoscaler policy
orphanUnattached resources (volumes, snapshots, IPs) with ongoing costs
complianceResources missing required tags or not following naming conventions
discountCommitment-based savings opportunities (Reserved Instances, Savings Plans)
governanceBest practice violations (e.g., public access, missing encryption)

Severity Levels

SeverityDescription
criticalHigh-cost waste requiring immediate attention
highSignificant savings opportunity
mediumModerate optimization potential
lowMinor improvement suggestion
infoInformational — no direct cost impact

List Resource Summaries

GET/recommendations/resources

List distinct resources with aggregate recommendation stats. Used for the resource view.

Query Parameters

ParameterDescription
statusFilter by recommendation status (open, applied, dismissed, optimised)
severityFilter by severity level
categoryFilter by category
providerFilter by cloud provider
resource_typeFilter by resource type (ec2, rds, disk, etc.)
cloud_account_idFilter by cloud account ID (comma-separated for multiple)
searchSearch by resource name or UID
sort_bySort column (savings_usd, updated_at, severity). Default: savings_usd
sort_orderSort direction (asc, desc). Default: desc
pagePage number (default: 1)
sizeItems per page (default: 10, max: 100)
Response · json
{
"data": {
  "items": [
    {
      "resourceUid": "i-0abc123def456",
      "resourceName": "idle-dev-server",
      "resourceType": "ec2",
      "provider": "aws",
      "cloudAccountId": "123456789012",
      "cloudAccountName": "AWS Prod",
      "recommendationCount": 3,
      "openCount": 2,
      "totalSavingsUsd": 52.56
    }
  ],
  "total": 45,
  "page": 1,
  "limit": 10,
  "hasMore": true
}
}

List Rule Summaries

GET/recommendations/rules

List distinct rules with aggregate recommendation stats. Used for the rule view.

Query Parameters

Same filters as /recommendations/resources plus:

ParameterDescription
rule_idFilter by specific rule ID (e.g., RC-212)
Response · json
{
"data": {
  "items": [
    {
      "ruleId": "RC-212",
      "title": "Azure managed disk is unattached — verify and delete if not needed",
      "category": "orphan",
      "severity": "low",
      "resourceCount": 36,
      "openCount": 35,
      "totalSavingsUsd": 5.00
    }
  ],
  "total": 8,
  "page": 1,
  "limit": 10,
  "hasMore": false
}
}

List Recommendations

GET/recommendations

List individual recommendations with filtering and pagination. Use resource_uid or rule_id to scope to a specific group.

Query Parameters

Same filters as /recommendations/resources plus:

ParameterDescription
resource_uidFilter by resource UID (used when expanding a resource card)
rule_idFilter by rule ID (used when expanding a rule card)
Response · json
{
"data": {
  "items": [
    {
      "id": "rec_001",
      "resourceUid": "i-0abc123def456",
      "resourceName": "idle-dev-server",
      "resourceType": "ec2",
      "ruleId": "RC-001",
      "title": "Idle EC2 Instance",
      "description": "This instance has had less than 5% CPU utilization over the past 14 days.",
      "currentCostUsd": 52.56,
      "optimizedCostUsd": 0.00,
      "savingsUsd": 52.56,
      "status": "open",
      "severity": "high",
      "category": "idle",
      "remediation": "Consider stopping or terminating this instance if it is not needed.",
      "consoleUrl": "https://console.aws.amazon.com/ec2/home?region=us-east-1#Instances:instanceId=i-0abc123def456",
      "provider": "aws",
      "cloudAccountId": "123456789012",
      "cloudAccountName": "AWS Prod",
      "actionType": "stop",
      "evidence": {
        "metric": "CPUUtilization",
        "windowDays": 14,
        "p95": 3.1
      },
      "generatedAt": "2025-01-20T08:00:00Z"
    }
  ],
  "total": 36,
  "page": 1,
  "limit": 5,
  "hasMore": true
}
}

Recommendation Summary

GET/recommendations/summary

Get aggregate recommendation statistics.

Response · json
{
"data": {
  "totalOpen": 23,
  "totalSavings": 1250.80,
  "criticalCount": 2,
  "highCount": 8,
  "resourceCount": 18,
  "awsResources": 10,
  "gcpResources": 5,
  "azureResources": 3,
  "appliedCount": 12,
  "dismissedCount": 5,
  "optimisedCount": 3
}
}

Resource Recommendations (Deprecated)

GET/recommendations/resources/{resourceUID}

Get all recommendations for a specific resource.

Update Recommendation

PATCH/recommendations/{recommendationID}

Mark a recommendation as applied or dismissed, or reopen one that was previously applied or dismissed.

Request · bash
curl -X PATCH https://zopnight.com/api/recommendations/rec_001 \
-H "Authorization: Bearer <token>" \
-H "Content-Type: application/json" \
-d '{ "status": "applied" }'

Valid status transitions:

  • open -> applied
  • open -> dismissed
  • applied -> open (reopen)
  • dismissed -> open (reopen)

Filter Fields

GET/recommendations/filter-fields

Get available filter field values for building filter UIs.

Response · json
{
"data": {
  "statuses": ["open", "applied", "dismissed", "optimised"],
  "severities": ["critical", "high", "medium", "low", "info"],
  "categories": ["idle", "rightsizing", "schedule", "orphan", "compliance", "discount", "governance"],
  "providers": ["aws", "gcp", "azure"]
}
}

Provider Breakdown

GET/recommendations/summary/providers

Get recommendation counts and savings broken down by cloud provider.

Response · json
{
"data": [
  { "provider": "aws", "totalOpen": 12, "totalSavings": 820.50, "criticalCount": 1, "highCount": 5 },
  { "provider": "gcp", "totalOpen": 6, "totalSavings": 310.00, "criticalCount": 0, "highCount": 2 },
  { "provider": "azure", "totalOpen": 5, "totalSavings": 120.30, "criticalCount": 1, "highCount": 1 }
]
}

Recommendation Detail

GET/recommendations/{recID}/detail

Get full detail for a single recommendation including metrics and remediation steps.

Refresh Recommendations

POST/recommendations/refresh

Trigger a recommendation recomputation for the organization.

Auto-Remediation Workflows

For auto-remediable rules (rightsizing, resize, oversized) a recommendation can be applied through a multi-step workflow that the provisioner executes against the cloud provider. Workflows are previewed, started, and approved through the endpoints below. See Provisioning for the apply-side detail.

GET/recommendations/{recID}/workflow

Preview the workflow plan that would be created for this recommendation — steps, target spec, and approval gates — without persisting anything.

POST/recommendations/{recID}/workflow

Start a remediation workflow for this recommendation. Returns a workflow job ID; poll /workflows/{jobID} for status.

GET/workflows/pending-approval

List workflow steps across the org that are waiting on a human approver. Used by the approvals queue widget.

GET/workflows/{jobID}

Get a workflow's status, current step, and per-step audit trail.

POST/workflows/{jobID}/cancel

Cancel an in-flight workflow. Steps already applied are not rolled back; subsequent steps are skipped.

POST/workflows/{jobID}/steps/{stepName}/approve

Approve a paused workflow step by its declared name. Use the by-id variant when you have the step's stable ID instead of the template name.

POST/workflows/{jobID}/steps/{stepName}/reject

Reject a paused workflow step. The workflow halts and no further steps are dispatched.

POST/workflows/{jobID}/steps-by-id/{stepID}/approve

Approve a paused workflow step by its stable ID. Prefer this when the UI already has the step ID — it survives template renames.

POST/workflows/{jobID}/steps-by-id/{stepID}/reject

Reject a paused workflow step by its stable ID.

Auto-Tagging

ZopNight can predict environment tags (prod, staging, dev) and no-stop flags for resources based on naming patterns, tags, and resource characteristics.

GET/autotagging

List auto-tagging suggestions.

Response · json
{
"data": {
  "items": [
    {
      "id": "tag_001",
      "resourceUID": "i-0abc123def456",
      "resourceName": "prod-api-server",
      "resourceType": "aws-ec2",
      "provider": "aws",
      "predictedEnv": "prod",
      "predictedNoStop": true,
      "confidence": 92,
      "reason": "Name contains 'prod', has production tags, runs 24/7"
    }
  ],
  "total": 15,
  "page": 1,
  "limit": 20
}
}
POST/autotagging/accept

Accept an auto-tag suggestion and apply it.

Request · bash
curl -X POST https://zopnight.com/api/autotagging/accept \
-H "Authorization: Bearer <token>" \
-H "Content-Type: application/json" \
-d '{
  "resourceUID": "i-0abc123def456",
  "env": "prod",
  "noStop": true
}'
POST/autotagging/reject

Reject an auto-tag suggestion.

Request · bash
curl -X POST https://zopnight.com/api/autotagging/reject \
-H "Authorization: Bearer <token>" \
-H "Content-Type: application/json" \
-d '{ "resourceUID": "i-0abc123def456" }'

The recommender runs 337+ audit rules across AWS (155), GCP (75), and Azure (107), including six autoscaler rules (RC-ASC-001..006) that feed into VM Autoscaling. Auto-tagging lives on its own page — Auto-Tagging. See Cloud Support Matrix for provider-specific details.