summaryrefslogblamecommitdiff
path: root/makima/src/daemon/api/directive.rs
blob: fcc2ca50f339acfd629cc47cef00c51ffd447a0b (plain) (tree)































                                                     















































                                                                                           

















































































                                                                                                         












                                                                        








                                                                                  










                                                                             























































































































































                                                                                                                                                                         








                                                     
 
//! Directive API methods.

use serde::Serialize;
use uuid::Uuid;

use super::client::{ApiClient, ApiError};
use super::supervisor::JsonValue;

#[derive(Serialize)]
#[serde(rename_all = "camelCase")]
pub struct CreateStepRequest {
    pub name: String,
    #[serde(skip_serializing_if = "Option::is_none")]
    pub description: Option<String>,
    #[serde(skip_serializing_if = "Option::is_none")]
    pub task_plan: Option<String>,
    pub depends_on: Vec<Uuid>,
    pub order_index: i32,
}

#[derive(Serialize)]
#[serde(rename_all = "camelCase")]
pub struct UpdateGoalRequest {
    pub goal: String,
}

#[derive(Serialize)]
#[serde(rename_all = "camelCase")]
pub struct UpdateStepDepsRequest {
    pub depends_on: Vec<Uuid>,
}

/// Percent-encode a string for use as a URL path segment.
///
/// Encodes all characters except unreserved characters (alphanumeric, `-`, `.`, `_`, `~`).
fn percent_encode_path(s: &str) -> String {
    let mut encoded = String::with_capacity(s.len());
    for byte in s.bytes() {
        match byte {
            b'A'..=b'Z' | b'a'..=b'z' | b'0'..=b'9' | b'-' | b'.' | b'_' | b'~' => {
                encoded.push(byte as char);
            }
            _ => {
                encoded.push_str(&format!("%{:02X}", byte));
            }
        }
    }
    encoded
}

/// Request body for setting a single memory entry.
#[derive(Serialize)]
#[serde(rename_all = "camelCase")]
pub struct SetMemoryRequest {
    pub key: String,
    pub value: String,
}

/// A single entry within a batch set request.
#[derive(Serialize)]
#[serde(rename_all = "camelCase")]
pub struct BatchMemoryEntry {
    pub key: String,
    pub value: String,
}

/// Request body for setting multiple memory entries at once.
#[derive(Serialize)]
#[serde(rename_all = "camelCase")]
pub struct BatchSetMemoryRequest {
    pub entries: Vec<BatchMemoryEntry>,
}

#[derive(Serialize)]
#[serde(rename_all = "camelCase")]
pub struct MemorySetRequest {
    pub value: String,
}


impl ApiClient {
    /// List all directives.
    pub async fn list_directives(&self) -> Result<JsonValue, ApiError> {
        self.get("/api/v1/directives").await
    }

    /// Get a directive with its steps.
    pub async fn get_directive(&self, directive_id: Uuid) -> Result<JsonValue, ApiError> {
        self.get(&format!("/api/v1/directives/{}", directive_id)).await
    }

    /// Add a step to a directive.
    pub async fn directive_add_step(
        &self,
        directive_id: Uuid,
        req: CreateStepRequest,
    ) -> Result<JsonValue, ApiError> {
        self.post(&format!("/api/v1/directives/{}/steps", directive_id), &req).await
    }

    /// Remove a step from a directive.
    pub async fn directive_remove_step(
        &self,
        directive_id: Uuid,
        step_id: Uuid,
    ) -> Result<(), ApiError> {
        self.delete(&format!("/api/v1/directives/{}/steps/{}", directive_id, step_id)).await
    }

    /// Set dependencies for a step.
    pub async fn directive_set_deps(
        &self,
        directive_id: Uuid,
        step_id: Uuid,
        depends_on: Vec<Uuid>,
    ) -> Result<JsonValue, ApiError> {
        let req = UpdateStepDepsRequest { depends_on };
        self.put(&format!("/api/v1/directives/{}/steps/{}", directive_id, step_id), &req).await
    }

    /// Start a directive.
    pub async fn directive_start(&self, directive_id: Uuid) -> Result<JsonValue, ApiError> {
        self.post_empty(&format!("/api/v1/directives/{}/start", directive_id)).await
    }

    /// Pause a directive.
    pub async fn directive_pause(&self, directive_id: Uuid) -> Result<JsonValue, ApiError> {
        self.post_empty(&format!("/api/v1/directives/{}/pause", directive_id)).await
    }

    /// Advance the directive DAG.
    pub async fn directive_advance(&self, directive_id: Uuid) -> Result<JsonValue, ApiError> {
        self.post_empty(&format!("/api/v1/directives/{}/advance", directive_id)).await
    }

    /// Mark a step as completed.
    pub async fn directive_complete_step(
        &self,
        directive_id: Uuid,
        step_id: Uuid,
    ) -> Result<JsonValue, ApiError> {
        self.post_empty(&format!("/api/v1/directives/{}/steps/{}/complete", directive_id, step_id)).await
    }

    /// Mark a step as failed.
    pub async fn directive_fail_step(
        &self,
        directive_id: Uuid,
        step_id: Uuid,
    ) -> Result<JsonValue, ApiError> {
        self.post_empty(&format!("/api/v1/directives/{}/steps/{}/fail", directive_id, step_id)).await
    }

    /// Mark a step as skipped.
    pub async fn directive_skip_step(
        &self,
        directive_id: Uuid,
        step_id: Uuid,
    ) -> Result<JsonValue, ApiError> {
        self.post_empty(&format!("/api/v1/directives/{}/steps/{}/skip", directive_id, step_id)).await
    }

    /// Batch add steps to a directive.
    pub async fn directive_batch_add_steps(
        &self,
        directive_id: Uuid,
        steps: serde_json::Value,
    ) -> Result<JsonValue, ApiError> {
        self.post(
            &format!("/api/v1/directives/{}/steps/batch", directive_id),
            &steps,
        )
        .await
    }

    /// Update the directive's goal.
    pub async fn directive_update_goal(
        &self,
        directive_id: Uuid,
        goal: &str,
    ) -> Result<JsonValue, ApiError> {
        let req = UpdateGoalRequest { goal: goal.to_string() };
        self.put(&format!("/api/v1/directives/{}/goal", directive_id), &req).await
    }

    /// Update directive metadata (PR URL, PR branch, etc.)
    pub async fn directive_update(
        &self,
        directive_id: Uuid,
        pr_url: Option<String>,
        pr_branch: Option<String>,
    ) -> Result<JsonValue, ApiError> {
        let req = UpdateDirectiveMetadataRequest { pr_url, pr_branch };
        self.put(&format!("/api/v1/directives/{}", directive_id), &req).await
    }

    // ── Directive Memory ──────────────────────────────────────────────

    /// List all memory entries for a directive.
    pub async fn list_memories(&self, directive_id: Uuid) -> Result<JsonValue, ApiError> {
        self.get(&format!("/api/v1/directives/{}/memory", directive_id))
            .await
    }

    /// Get a single memory entry by key.
    pub async fn get_memory(
        &self,
        directive_id: Uuid,
        key: &str,
    ) -> Result<JsonValue, ApiError> {
        self.get(&format!(
            "/api/v1/directives/{}/memory/{}",
            directive_id,
            percent_encode_path(key)
        ))
        .await
    }

    /// Set (create or update) a single memory entry.
    pub async fn set_memory(
        &self,
        directive_id: Uuid,
        key: &str,
        value: &str,
    ) -> Result<JsonValue, ApiError> {
        let req = SetMemoryRequest {
            key: key.to_string(),
            value: value.to_string(),
        };
        self.put(&format!("/api/v1/directives/{}/memory", directive_id), &req)
            .await
    }

    /// Set multiple memory entries in a single request.
    pub async fn batch_set_memories(
        &self,
        directive_id: Uuid,
        entries: Vec<(String, String)>,
    ) -> Result<JsonValue, ApiError> {
        let req = BatchSetMemoryRequest {
            entries: entries
                .into_iter()
                .map(|(key, value)| BatchMemoryEntry { key, value })
                .collect(),
        };
        self.post(
            &format!("/api/v1/directives/{}/memory/batch", directive_id),
            &req,
        )
        .await
    }

    /// Delete a single memory entry by key.
    pub async fn delete_memory(
        &self,
        directive_id: Uuid,
        key: &str,
    ) -> Result<(), ApiError> {
        self.delete(&format!(
            "/api/v1/directives/{}/memory/{}",
            directive_id,
            percent_encode_path(key)
        ))
        .await
    }

    /// Clear all memory entries for a directive.
    pub async fn clear_memories(&self, directive_id: Uuid) -> Result<(), ApiError> {
        self.delete(&format!("/api/v1/directives/{}/memory", directive_id))
            .await
    }

    // ── CLI-facing Directive Memory aliases ──────────────────────────

    /// Set a memory key-value pair for a directive (CLI-facing).
    pub async fn directive_memory_set(
        &self,
        directive_id: Uuid,
        key: &str,
        value: &str,
    ) -> Result<JsonValue, ApiError> {
        let req = MemorySetRequest {
            value: value.to_string(),
        };
        self.put(
            &format!("/api/v1/directives/{}/memory/{}", directive_id, key),
            &req,
        )
        .await
    }

    /// Get a memory value by key for a directive (CLI-facing).
    pub async fn directive_memory_get(
        &self,
        directive_id: Uuid,
        key: &str,
    ) -> Result<JsonValue, ApiError> {
        self.get(&format!(
            "/api/v1/directives/{}/memory/{}",
            directive_id, key
        ))
        .await
    }

    /// List all memory key-value pairs for a directive (CLI-facing).
    pub async fn directive_memory_list(
        &self,
        directive_id: Uuid,
    ) -> Result<JsonValue, ApiError> {
        self.get(&format!("/api/v1/directives/{}/memory", directive_id))
            .await
    }

    /// Delete a memory key for a directive (CLI-facing).
    pub async fn directive_memory_delete(
        &self,
        directive_id: Uuid,
        key: &str,
    ) -> Result<(), ApiError> {
        self.delete(&format!(
            "/api/v1/directives/{}/memory/{}",
            directive_id, key
        ))
        .await
    }

    /// Clear all memory for a directive (CLI-facing).
    pub async fn directive_memory_clear(
        &self,
        directive_id: Uuid,
    ) -> Result<(), ApiError> {
        self.delete(&format!("/api/v1/directives/{}/memory", directive_id))
            .await
    }

    /// Batch set multiple memory key-value pairs for a directive (CLI-facing).
    pub async fn directive_memory_batch_set(
        &self,
        directive_id: Uuid,
        entries: serde_json::Value,
    ) -> Result<JsonValue, ApiError> {
        self.post(
            &format!("/api/v1/directives/{}/memory/batch", directive_id),
            &entries,
        )
        .await
    }
}

#[derive(Serialize)]
#[serde(rename_all = "camelCase")]
pub struct UpdateDirectiveMetadataRequest {
    #[serde(skip_serializing_if = "Option::is_none")]
    pub pr_url: Option<String>,
    #[serde(skip_serializing_if = "Option::is_none")]
    pub pr_branch: Option<String>,
}