Skip to content

Commit

Permalink
feat: playbook action consumer & playbook_action_agent_data table
Browse files Browse the repository at this point in the history
  • Loading branch information
adityathebe committed Jan 22, 2024
1 parent 97b748c commit 9333a63
Show file tree
Hide file tree
Showing 4 changed files with 71 additions and 11 deletions.
22 changes: 17 additions & 5 deletions models/playbooks.go
Original file line number Diff line number Diff line change
Expand Up @@ -11,13 +11,14 @@ import (
type PlaybookRunStatus string

const (
PlaybookRunStatusPending PlaybookRunStatus = "pending"
PlaybookRunStatusScheduled PlaybookRunStatus = "scheduled"
PlaybookRunStatusRunning PlaybookRunStatus = "running"
PlaybookRunStatusCancelled PlaybookRunStatus = "cancelled"
PlaybookRunStatusFailed PlaybookRunStatus = "failed"
PlaybookRunStatusCompleted PlaybookRunStatus = "completed"
PlaybookRunStatusFailed PlaybookRunStatus = "failed"
PlaybookRunStatusPending PlaybookRunStatus = "pending" // pending approval
PlaybookRunStatusRunning PlaybookRunStatus = "running"
PlaybookRunStatusScheduled PlaybookRunStatus = "scheduled"
PlaybookRunStatusSleeping PlaybookRunStatus = "sleeping"
PlaybookRunStatusWaiting PlaybookRunStatus = "waiting" // waiting for a consumer
)

// PlaybookRunStatus are statuses for a playbook run and its actions.
Expand All @@ -27,6 +28,7 @@ const (
PlaybookActionStatusCompleted PlaybookActionStatus = "completed"
PlaybookActionStatusFailed PlaybookActionStatus = "failed"
PlaybookActionStatusRunning PlaybookActionStatus = "running"
PlaybookActionStatusScheduled PlaybookActionStatus = "scheduled"
PlaybookActionStatusSkipped PlaybookActionStatus = "skipped"
PlaybookActionStatusSleeping PlaybookActionStatus = "sleeping"
)
Expand Down Expand Up @@ -67,7 +69,7 @@ type PlaybookRun struct {
PlaybookID uuid.UUID `json:"playbook_id"`
Status PlaybookRunStatus `json:"status,omitempty"`
CreatedAt time.Time `json:"created_at,omitempty" time_format:"postgres_timestamp" gorm:"<-:false"`
StartTime time.Time `json:"start_time,omitempty" time_format:"postgres_timestamp"`
StartTime *time.Time `json:"start_time,omitempty" time_format:"postgres_timestamp"`
ScheduledTime time.Time `json:"scheduled_time,omitempty" time_format:"postgres_timestamp" gorm:"default:NOW(), NOT NULL"`
EndTime *time.Time `json:"end_time,omitempty" time_format:"postgres_timestamp"`
CreatedBy *uuid.UUID `json:"created_by,omitempty"`
Expand Down Expand Up @@ -111,3 +113,13 @@ type PlaybookApproval struct {
func (p PlaybookApproval) AsMap(removeFields ...string) map[string]any {
return asMap(p, removeFields...)
}

type PlaybookActionAgentData struct {
ActionID uuid.UUID `json:"action_id"`
Spec types.JSON `json:"spec"`
Env types.JSON `json:"env,omitempty"`
}

func (t *PlaybookActionAgentData) TableName() string {
return "playbook_action_agent_data"
}
34 changes: 29 additions & 5 deletions schema/playbooks.hcl
Original file line number Diff line number Diff line change
Expand Up @@ -221,6 +221,31 @@ table "playbook_runs" {
}
}

table "playbook_action_agent_data" {
schema = schema.public
comment = "saves the necessary details for the agent to run a playbook action (eg: template env vars). Only applicable to agent runners."
column "action_id" {
null = false
type = uuid
}
column "spec" {
comment = "Action spec provided by upstream"
null = false
type = jsonb
}
column "env" {
comment = "templateEnv for the action provided by the upstream"
null = true
type = jsonb
}
foreign_key "playbook_action_template_env_agent_action_id_fkey" {
columns = [column.action_id]
ref_columns = [table.playbook_run_actions.column.id]
on_update = NO_ACTION
on_delete = CASCADE
}
}

table "playbook_run_actions" {
schema = schema.public
column "id" {
Expand All @@ -238,14 +263,13 @@ table "playbook_run_actions" {
default = "running"
}
column "playbook_run_id" {
null = true
type = uuid
null = true
type = uuid
comment = "a run id is mandatory except for an agent"
}
column "start_time" {
null = true
type = timestamptz
default = sql("now()")
null = true
type = timestamptz
}
column "scheduled_time" {
null = false
Expand Down
6 changes: 5 additions & 1 deletion upstream/commands.go
Original file line number Diff line number Diff line change
Expand Up @@ -195,7 +195,11 @@ func InsertUpstreamMsg(ctx context.Context, req *PushData) error {
"error": req.PlaybookActions[i].Error,
}
if err := db.Model(&models.PlaybookRunAction{}).Where("id = ?", req.PlaybookActions[i].ID).Updates(updates).Error; err != nil {
return fmt.Errorf("error updating playbook action [%s]: %w", req.PlaybookActions[i].ID.String(), err)
return fmt.Errorf("error updating playbook action [%s]: %w", req.PlaybookActions[i].ID, err)
}

if err := db.Exec("UPDATE playbook_runs SET status = ? WHERE id = (SELECT playbook_run_id FROM playbook_run_actions WHERE id = ?)", models.PlaybookRunStatusScheduled, req.PlaybookActions[i].ID).Error; err != nil {
return fmt.Errorf("error updating playbook run [%s] status to %s : %w", req.PlaybookActions[i].PlaybookRunID, models.PlaybookRunStatusScheduled, err)
}
}

Expand Down
20 changes: 20 additions & 0 deletions views/018_playbooks.sql
Original file line number Diff line number Diff line change
@@ -1,3 +1,23 @@
-- Notify playbook action created or status updated
CREATE OR REPLACE FUNCTION notify_playbook_action_update() RETURNS TRIGGER AS $$
BEGIN
IF TG_OP = 'INSERT' THEN
NOTIFY playbook_action_updates;
ELSEIF TG_OP = 'UPDATE' THEN
IF OLD.status != NEW.status AND NEW.status = 'scheduled' THEN
NOTIFY playbook_action_updates;
END IF;
END IF;

RETURN NULL;
END
$$ LANGUAGE plpgsql;

CREATE OR REPLACE TRIGGER playbook_action_updates
AFTER INSERT OR UPDATE ON playbook_run_actions
FOR EACH ROW
EXECUTE PROCEDURE notify_playbook_action_update();

-- Notify playbook run created or status updated
CREATE OR REPLACE FUNCTION notify_playbook_run_update() RETURNS TRIGGER AS $$
BEGIN
Expand Down

0 comments on commit 9333a63

Please sign in to comment.