use std::time::Duration;
use reqwest::Client;
use serde::{de::DeserializeOwned, Serialize};
const DEFAULT_HTTP_URL: &str = "http://127.0.0.1:7878";
pub fn space_header_name() -> &'static str {
"x-origin-space"
}
pub fn discover_origin_url(cli_url: Option<String>) -> String {
if let Some(url) = cli_url {
return url;
}
DEFAULT_HTTP_URL.to_string()
}
#[derive(Clone)]
pub struct OriginClient {
client: Client,
base_url: String,
agent_name: Option<String>,
}
const MAX_RETRIES: u32 = 3;
const BACKOFF_BASE: Duration = Duration::from_secs(1);
impl OriginClient {
pub fn new(base_url: String) -> Self {
Self {
client: Client::new(),
base_url,
agent_name: None,
}
}
pub fn with_agent_name(mut self, name: String) -> Self {
self.agent_name = Some(name);
self
}
async fn send_with_retry(
&self,
build: impl Fn() -> reqwest::RequestBuilder,
) -> Result<reqwest::Response, OriginError> {
let mut last_err = None;
for attempt in 0..MAX_RETRIES {
if attempt > 0 {
tokio::time::sleep(BACKOFF_BASE * attempt).await;
}
match build().send().await {
Ok(resp) => return Ok(resp),
Err(e) if e.is_connect() => {
tracing::debug!(attempt, "daemon unreachable, retrying");
last_err = Some(e);
}
Err(e) => return Err(OriginError::Unreachable(e.to_string())),
}
}
Err(OriginError::Unreachable(last_err.map_or_else(
|| "connection failed".into(),
|e| e.to_string(),
)))
}
fn parse_response<R: DeserializeOwned>(bytes: &[u8]) -> Result<R, OriginError> {
serde_json::from_slice::<R>(bytes).map_err(|e| {
let preview = std::str::from_utf8(bytes)
.unwrap_or("<non-utf8>")
.chars()
.take(512)
.collect::<String>();
OriginError::Deserialize(format!("{e} (body preview: {preview})"))
})
}
async fn read_body(resp: reqwest::Response) -> Result<Vec<u8>, OriginError> {
if !resp.status().is_success() {
let status = resp.status().as_u16();
let body = resp.text().await.unwrap_or_default();
return Err(OriginError::Api { status, body });
}
resp.bytes()
.await
.map(|b| b.to_vec())
.map_err(|e| OriginError::Deserialize(format!("failed to read response body: {e:#}")))
}
fn attach_common_headers(
mut req: reqwest::RequestBuilder,
agent: Option<&str>,
) -> reqwest::RequestBuilder {
if let Some(a) = agent {
req = req.header("x-agent-name", a);
}
if let Some(space) = crate::lock_state::locked_space() {
req = req.header(space_header_name(), space);
}
req
}
pub async fn get<R: DeserializeOwned>(&self, path: &str) -> Result<R, OriginError> {
let url = format!("{}{}", self.base_url, path);
let agent = self.agent_name.clone();
let resp = self
.send_with_retry(|| {
let req = self.client.get(&url);
Self::attach_common_headers(req, agent.as_deref())
})
.await?;
let bytes = Self::read_body(resp).await?;
Self::parse_response(&bytes)
}
pub async fn post<B: Serialize, R: DeserializeOwned>(
&self,
path: &str,
body: &B,
) -> Result<R, OriginError> {
let url = format!("{}{}", self.base_url, path);
let agent = self.agent_name.clone();
let resp = self
.send_with_retry(|| {
let req = self.client.post(&url).json(body);
Self::attach_common_headers(req, agent.as_deref())
})
.await?;
let bytes = Self::read_body(resp).await?;
Self::parse_response(&bytes)
}
pub async fn post_empty<R: DeserializeOwned>(&self, path: &str) -> Result<R, OriginError> {
let url = format!("{}{}", self.base_url, path);
let agent = self.agent_name.clone();
let resp = self
.send_with_retry(|| {
let req = self.client.post(&url);
Self::attach_common_headers(req, agent.as_deref())
})
.await?;
let bytes = Self::read_body(resp).await?;
Self::parse_response(&bytes)
}
pub async fn delete<R: DeserializeOwned>(&self, path: &str) -> Result<R, OriginError> {
let url = format!("{}{}", self.base_url, path);
let agent = self.agent_name.clone();
let resp = self
.send_with_retry(|| {
let req = self.client.delete(&url);
Self::attach_common_headers(req, agent.as_deref())
})
.await?;
let bytes = Self::read_body(resp).await?;
Self::parse_response(&bytes)
}
pub async fn put<B: Serialize, R: DeserializeOwned>(
&self,
path: &str,
body: &B,
) -> Result<R, OriginError> {
let url = format!("{}{}", self.base_url, path);
let agent = self.agent_name.clone();
let resp = self
.send_with_retry(|| {
let req = self.client.put(&url).json(body);
Self::attach_common_headers(req, agent.as_deref())
})
.await?;
let bytes = Self::read_body(resp).await?;
Self::parse_response(&bytes)
}
pub async fn version_handshake(&self) -> Option<String> {
use crate::version_check::{compare, VersionStatus};
let url = format!("{}/api/health", self.base_url);
let resp = self
.client
.get(&url)
.timeout(Duration::from_secs(2))
.send()
.await
.ok()?;
let body: serde_json::Value = resp.json().await.ok()?;
let daemon_version = body["version"].as_str()?;
let mcp_version = env!("CARGO_PKG_VERSION");
match compare(mcp_version, daemon_version) {
VersionStatus::Compatible => None,
VersionStatus::McpOutdated { mcp, daemon } => Some(format!(
"Your origin-mcp v{mcp} is older than the daemon v{daemon}. \
Run `brew upgrade origin-mcp` (or `npm update -g origin-mcp`)."
)),
}
}
}
#[derive(Debug, thiserror::Error)]
pub enum OriginError {
#[error("Origin is not reachable: {0}")]
Unreachable(String),
#[error("Origin API error (HTTP {status}): {body}")]
Api { status: u16, body: String },
#[error("Failed to parse Origin response: {0}")]
Deserialize(String),
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_discover_url_prefers_cli_flag() {
let url = discover_origin_url(Some("http://localhost:9999".into()));
assert_eq!(url, "http://localhost:9999");
}
#[test]
fn test_discover_url_falls_back_to_http() {
let url = discover_origin_url(None);
assert_eq!(url, "http://127.0.0.1:7878");
}
#[test]
fn space_header_attached_when_locked() {
let _guard = crate::lock_state::ENV_LOCK.lock().unwrap();
std::env::set_var("ORIGIN_SPACE", "career");
crate::lock_state::init_from_env();
let client = Client::new();
let builder = OriginClient::attach_common_headers(
client.get("http://127.0.0.1:7878/api/health"),
None,
);
let req = builder.build().unwrap();
let header = req.headers().get(space_header_name()).unwrap();
assert_eq!(header.to_str().unwrap(), "career");
std::env::remove_var("ORIGIN_SPACE");
crate::lock_state::init_from_env();
}
#[test]
fn space_header_absent_when_unlocked() {
let _guard = crate::lock_state::ENV_LOCK.lock().unwrap();
std::env::remove_var("ORIGIN_SPACE");
crate::lock_state::init_from_env();
let client = Client::new();
let builder = OriginClient::attach_common_headers(
client.get("http://127.0.0.1:7878/api/health"),
None,
);
let req = builder.build().unwrap();
assert!(req.headers().get(space_header_name()).is_none());
}
}