mirror of
https://github.com/openai/codex.git
synced 2026-04-24 06:35:50 +00:00
686 lines
25 KiB
Rust
686 lines
25 KiB
Rust
use crate::events::AppServerRpcTransport;
|
|
use crate::events::CodexAppMentionedEventRequest;
|
|
use crate::events::CodexAppServerClientMetadata;
|
|
use crate::events::CodexAppUsedEventRequest;
|
|
use crate::events::CodexPluginEventRequest;
|
|
use crate::events::CodexPluginUsedEventRequest;
|
|
use crate::events::CodexRuntimeMetadata;
|
|
use crate::events::CodexTurnEventParams;
|
|
use crate::events::CodexTurnEventRequest;
|
|
use crate::events::SkillInvocationEventParams;
|
|
use crate::events::SkillInvocationEventRequest;
|
|
use crate::events::ThreadInitializationMode;
|
|
use crate::events::ThreadInitializedEvent;
|
|
use crate::events::ThreadInitializedEventParams;
|
|
use crate::events::TrackEventRequest;
|
|
use crate::events::codex_app_metadata;
|
|
use crate::events::codex_plugin_metadata;
|
|
use crate::events::codex_plugin_used_metadata;
|
|
use crate::events::plugin_state_event_type;
|
|
use crate::events::subagent_thread_started_event_request;
|
|
use crate::events::thread_source_name;
|
|
use crate::facts::AnalyticsFact;
|
|
use crate::facts::AppMentionedInput;
|
|
use crate::facts::AppUsedInput;
|
|
use crate::facts::CustomAnalyticsFact;
|
|
use crate::facts::PluginState;
|
|
use crate::facts::PluginStateChangedInput;
|
|
use crate::facts::PluginUsedInput;
|
|
use crate::facts::SkillInvokedInput;
|
|
use crate::facts::SubAgentThreadStartedInput;
|
|
use crate::facts::TurnResolvedConfigFact;
|
|
use crate::facts::TurnStatus;
|
|
use codex_app_server_protocol::ClientRequest;
|
|
use codex_app_server_protocol::ClientResponse;
|
|
use codex_app_server_protocol::CodexErrorInfo;
|
|
use codex_app_server_protocol::InitializeParams;
|
|
use codex_app_server_protocol::RequestId;
|
|
use codex_app_server_protocol::ServerNotification;
|
|
use codex_app_server_protocol::TokenUsageBreakdown;
|
|
use codex_app_server_protocol::UserInput;
|
|
use codex_git_utils::collect_git_info;
|
|
use codex_git_utils::get_git_repo_root;
|
|
use codex_login::default_client::originator;
|
|
use codex_protocol::config_types::ModeKind;
|
|
use codex_protocol::config_types::Personality;
|
|
use codex_protocol::config_types::ReasoningSummary;
|
|
use codex_protocol::protocol::SandboxPolicy;
|
|
use codex_protocol::protocol::SessionSource;
|
|
use codex_protocol::protocol::SkillScope;
|
|
use sha1::Digest;
|
|
use std::collections::HashMap;
|
|
use std::path::Path;
|
|
|
|
#[derive(Default)]
|
|
pub(crate) struct AnalyticsReducer {
|
|
requests: HashMap<(u64, RequestId), RequestState>,
|
|
turns: HashMap<String, TurnState>,
|
|
connections: HashMap<u64, ConnectionState>,
|
|
}
|
|
|
|
struct ConnectionState {
|
|
app_server_client: CodexAppServerClientMetadata,
|
|
runtime: CodexRuntimeMetadata,
|
|
}
|
|
|
|
enum RequestState {
|
|
TurnStart(PendingTurnStartState),
|
|
}
|
|
|
|
struct PendingTurnStartState {
|
|
thread_id: String,
|
|
num_input_images: usize,
|
|
}
|
|
|
|
#[derive(Clone)]
|
|
struct CompletedTurnState {
|
|
status: Option<TurnStatus>,
|
|
turn_error: Option<CodexErrorInfo>,
|
|
completed_at: u64,
|
|
duration_ms: Option<u64>,
|
|
}
|
|
|
|
struct TurnState {
|
|
connection_id: Option<u64>,
|
|
thread_id: Option<String>,
|
|
num_input_images: Option<usize>,
|
|
resolved_config: Option<TurnResolvedConfigFact>,
|
|
created_at: Option<u64>,
|
|
token_usage: Option<TokenUsageBreakdown>,
|
|
completed: Option<CompletedTurnState>,
|
|
}
|
|
|
|
impl AnalyticsReducer {
|
|
pub(crate) async fn ingest(&mut self, input: AnalyticsFact, out: &mut Vec<TrackEventRequest>) {
|
|
match input {
|
|
AnalyticsFact::Initialize {
|
|
connection_id,
|
|
params,
|
|
product_client_id,
|
|
runtime,
|
|
rpc_transport,
|
|
} => {
|
|
self.ingest_initialize(
|
|
connection_id,
|
|
params,
|
|
product_client_id,
|
|
runtime,
|
|
rpc_transport,
|
|
);
|
|
}
|
|
AnalyticsFact::Request {
|
|
connection_id,
|
|
request_id,
|
|
request,
|
|
} => {
|
|
self.ingest_request(connection_id, request_id, *request);
|
|
}
|
|
AnalyticsFact::Response {
|
|
connection_id,
|
|
response,
|
|
} => {
|
|
self.ingest_response(connection_id, *response, out);
|
|
}
|
|
AnalyticsFact::Notification(notification) => {
|
|
self.ingest_notification(*notification, out);
|
|
}
|
|
AnalyticsFact::Custom(input) => match input {
|
|
CustomAnalyticsFact::SubAgentThreadStarted(input) => {
|
|
self.ingest_subagent_thread_started(input, out);
|
|
}
|
|
CustomAnalyticsFact::TurnResolvedConfig(input) => {
|
|
self.ingest_turn_resolved_config(*input, out);
|
|
}
|
|
CustomAnalyticsFact::SkillInvoked(input) => {
|
|
self.ingest_skill_invoked(input, out).await;
|
|
}
|
|
CustomAnalyticsFact::AppMentioned(input) => {
|
|
self.ingest_app_mentioned(input, out);
|
|
}
|
|
CustomAnalyticsFact::AppUsed(input) => {
|
|
self.ingest_app_used(input, out);
|
|
}
|
|
CustomAnalyticsFact::PluginUsed(input) => {
|
|
self.ingest_plugin_used(input, out);
|
|
}
|
|
CustomAnalyticsFact::PluginStateChanged(input) => {
|
|
self.ingest_plugin_state_changed(input, out);
|
|
}
|
|
},
|
|
}
|
|
}
|
|
|
|
fn ingest_initialize(
|
|
&mut self,
|
|
connection_id: u64,
|
|
params: InitializeParams,
|
|
product_client_id: String,
|
|
runtime: CodexRuntimeMetadata,
|
|
rpc_transport: AppServerRpcTransport,
|
|
) {
|
|
self.connections.insert(
|
|
connection_id,
|
|
ConnectionState {
|
|
app_server_client: CodexAppServerClientMetadata {
|
|
product_client_id,
|
|
client_name: Some(params.client_info.name),
|
|
client_version: Some(params.client_info.version),
|
|
rpc_transport,
|
|
experimental_api_enabled: params
|
|
.capabilities
|
|
.map(|capabilities| capabilities.experimental_api),
|
|
},
|
|
runtime,
|
|
},
|
|
);
|
|
}
|
|
|
|
fn ingest_subagent_thread_started(
|
|
&mut self,
|
|
input: SubAgentThreadStartedInput,
|
|
out: &mut Vec<TrackEventRequest>,
|
|
) {
|
|
out.push(TrackEventRequest::ThreadInitialized(
|
|
subagent_thread_started_event_request(input),
|
|
));
|
|
}
|
|
|
|
fn ingest_request(
|
|
&mut self,
|
|
connection_id: u64,
|
|
request_id: RequestId,
|
|
request: ClientRequest,
|
|
) {
|
|
let ClientRequest::TurnStart { params, .. } = request else {
|
|
return;
|
|
};
|
|
self.requests.insert(
|
|
(connection_id, request_id),
|
|
RequestState::TurnStart(PendingTurnStartState {
|
|
thread_id: params.thread_id,
|
|
num_input_images: params
|
|
.input
|
|
.iter()
|
|
.filter(|item| {
|
|
matches!(item, UserInput::Image { .. } | UserInput::LocalImage { .. })
|
|
})
|
|
.count(),
|
|
}),
|
|
);
|
|
}
|
|
|
|
fn ingest_turn_resolved_config(
|
|
&mut self,
|
|
input: TurnResolvedConfigFact,
|
|
out: &mut Vec<TrackEventRequest>,
|
|
) {
|
|
let turn_id = input.turn_id.clone();
|
|
let thread_id = input.thread_id.clone();
|
|
let num_input_images = input.num_input_images;
|
|
let turn_state = self.turns.entry(turn_id.clone()).or_insert(TurnState {
|
|
connection_id: None,
|
|
thread_id: None,
|
|
num_input_images: None,
|
|
resolved_config: None,
|
|
created_at: None,
|
|
token_usage: None,
|
|
completed: None,
|
|
});
|
|
turn_state.thread_id = Some(thread_id);
|
|
turn_state.num_input_images = Some(num_input_images);
|
|
turn_state.resolved_config = Some(input);
|
|
self.maybe_emit_turn_event(&turn_id, out);
|
|
}
|
|
|
|
async fn ingest_skill_invoked(
|
|
&mut self,
|
|
input: SkillInvokedInput,
|
|
out: &mut Vec<TrackEventRequest>,
|
|
) {
|
|
let SkillInvokedInput {
|
|
tracking,
|
|
invocations,
|
|
} = input;
|
|
for invocation in invocations {
|
|
let skill_scope = match invocation.skill_scope {
|
|
SkillScope::User => "user",
|
|
SkillScope::Repo => "repo",
|
|
SkillScope::System => "system",
|
|
SkillScope::Admin => "admin",
|
|
};
|
|
let repo_root = get_git_repo_root(invocation.skill_path.as_path());
|
|
let repo_url = if let Some(root) = repo_root.as_ref() {
|
|
collect_git_info(root)
|
|
.await
|
|
.and_then(|info| info.repository_url)
|
|
} else {
|
|
None
|
|
};
|
|
let skill_id = skill_id_for_local_skill(
|
|
repo_url.as_deref(),
|
|
repo_root.as_deref(),
|
|
invocation.skill_path.as_path(),
|
|
invocation.skill_name.as_str(),
|
|
);
|
|
out.push(TrackEventRequest::SkillInvocation(
|
|
SkillInvocationEventRequest {
|
|
event_type: "skill_invocation",
|
|
skill_id,
|
|
skill_name: invocation.skill_name.clone(),
|
|
event_params: SkillInvocationEventParams {
|
|
thread_id: Some(tracking.thread_id.clone()),
|
|
invoke_type: Some(invocation.invocation_type),
|
|
model_slug: Some(tracking.model_slug.clone()),
|
|
product_client_id: Some(originator().value),
|
|
repo_url,
|
|
skill_scope: Some(skill_scope.to_string()),
|
|
},
|
|
},
|
|
));
|
|
}
|
|
}
|
|
|
|
fn ingest_app_mentioned(&mut self, input: AppMentionedInput, out: &mut Vec<TrackEventRequest>) {
|
|
let AppMentionedInput { tracking, mentions } = input;
|
|
out.extend(mentions.into_iter().map(|mention| {
|
|
let event_params = codex_app_metadata(&tracking, mention);
|
|
TrackEventRequest::AppMentioned(CodexAppMentionedEventRequest {
|
|
event_type: "codex_app_mentioned",
|
|
event_params,
|
|
})
|
|
}));
|
|
}
|
|
|
|
fn ingest_app_used(&mut self, input: AppUsedInput, out: &mut Vec<TrackEventRequest>) {
|
|
let AppUsedInput { tracking, app } = input;
|
|
let event_params = codex_app_metadata(&tracking, app);
|
|
out.push(TrackEventRequest::AppUsed(CodexAppUsedEventRequest {
|
|
event_type: "codex_app_used",
|
|
event_params,
|
|
}));
|
|
}
|
|
|
|
fn ingest_plugin_used(&mut self, input: PluginUsedInput, out: &mut Vec<TrackEventRequest>) {
|
|
let PluginUsedInput { tracking, plugin } = input;
|
|
out.push(TrackEventRequest::PluginUsed(CodexPluginUsedEventRequest {
|
|
event_type: "codex_plugin_used",
|
|
event_params: codex_plugin_used_metadata(&tracking, plugin),
|
|
}));
|
|
}
|
|
|
|
fn ingest_plugin_state_changed(
|
|
&mut self,
|
|
input: PluginStateChangedInput,
|
|
out: &mut Vec<TrackEventRequest>,
|
|
) {
|
|
let PluginStateChangedInput { plugin, state } = input;
|
|
let event = CodexPluginEventRequest {
|
|
event_type: plugin_state_event_type(state),
|
|
event_params: codex_plugin_metadata(plugin),
|
|
};
|
|
out.push(match state {
|
|
PluginState::Installed => TrackEventRequest::PluginInstalled(event),
|
|
PluginState::Uninstalled => TrackEventRequest::PluginUninstalled(event),
|
|
PluginState::Enabled => TrackEventRequest::PluginEnabled(event),
|
|
PluginState::Disabled => TrackEventRequest::PluginDisabled(event),
|
|
});
|
|
}
|
|
|
|
fn ingest_response(
|
|
&mut self,
|
|
connection_id: u64,
|
|
response: ClientResponse,
|
|
out: &mut Vec<TrackEventRequest>,
|
|
) {
|
|
match response {
|
|
ClientResponse::ThreadStart { response, .. } => {
|
|
self.emit_thread_initialized(
|
|
connection_id,
|
|
response.thread,
|
|
response.model,
|
|
ThreadInitializationMode::New,
|
|
out,
|
|
);
|
|
}
|
|
ClientResponse::ThreadResume { response, .. } => {
|
|
self.emit_thread_initialized(
|
|
connection_id,
|
|
response.thread,
|
|
response.model,
|
|
ThreadInitializationMode::Resumed,
|
|
out,
|
|
);
|
|
}
|
|
ClientResponse::ThreadFork { response, .. } => {
|
|
self.emit_thread_initialized(
|
|
connection_id,
|
|
response.thread,
|
|
response.model,
|
|
ThreadInitializationMode::Forked,
|
|
out,
|
|
);
|
|
}
|
|
ClientResponse::TurnStart {
|
|
request_id,
|
|
response,
|
|
} => {
|
|
let turn_id = response.turn.id;
|
|
let Some(RequestState::TurnStart(pending_request)) =
|
|
self.requests.remove(&(connection_id, request_id))
|
|
else {
|
|
return;
|
|
};
|
|
let turn_state = self.turns.entry(turn_id.clone()).or_insert(TurnState {
|
|
connection_id: None,
|
|
thread_id: None,
|
|
num_input_images: None,
|
|
resolved_config: None,
|
|
created_at: None,
|
|
token_usage: None,
|
|
completed: None,
|
|
});
|
|
turn_state.connection_id = Some(connection_id);
|
|
turn_state.thread_id = Some(pending_request.thread_id);
|
|
turn_state.num_input_images = Some(pending_request.num_input_images);
|
|
self.maybe_emit_turn_event(&turn_id, out);
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
|
|
fn ingest_notification(
|
|
&mut self,
|
|
notification: ServerNotification,
|
|
out: &mut Vec<TrackEventRequest>,
|
|
) {
|
|
match notification {
|
|
ServerNotification::TurnStarted(notification) => {
|
|
let turn_state = self.turns.entry(notification.turn.id).or_insert(TurnState {
|
|
connection_id: None,
|
|
thread_id: None,
|
|
num_input_images: None,
|
|
resolved_config: None,
|
|
created_at: None,
|
|
token_usage: None,
|
|
completed: None,
|
|
});
|
|
turn_state.created_at = notification
|
|
.turn
|
|
.created_at
|
|
.and_then(|created_at| u64::try_from(created_at).ok());
|
|
}
|
|
ServerNotification::ThreadTokenUsageUpdated(notification) => {
|
|
let turn_state = self.turns.entry(notification.turn_id).or_insert(TurnState {
|
|
connection_id: None,
|
|
thread_id: None,
|
|
num_input_images: None,
|
|
resolved_config: None,
|
|
created_at: None,
|
|
token_usage: None,
|
|
completed: None,
|
|
});
|
|
turn_state.token_usage = Some(notification.token_usage.last);
|
|
}
|
|
ServerNotification::TurnCompleted(notification) => {
|
|
let turn_state =
|
|
self.turns
|
|
.entry(notification.turn.id.clone())
|
|
.or_insert(TurnState {
|
|
connection_id: None,
|
|
thread_id: None,
|
|
num_input_images: None,
|
|
resolved_config: None,
|
|
created_at: None,
|
|
token_usage: None,
|
|
completed: None,
|
|
});
|
|
turn_state.completed = Some(CompletedTurnState {
|
|
status: analytics_turn_status(notification.turn.status),
|
|
turn_error: notification
|
|
.turn
|
|
.error
|
|
.and_then(|error| error.codex_error_info),
|
|
completed_at: notification
|
|
.turn
|
|
.completed_at
|
|
.and_then(|completed_at| u64::try_from(completed_at).ok())
|
|
.unwrap_or_default(),
|
|
duration_ms: notification
|
|
.turn
|
|
.duration_ms
|
|
.and_then(|duration_ms| u64::try_from(duration_ms).ok()),
|
|
});
|
|
let turn_id = notification.turn.id;
|
|
self.maybe_emit_turn_event(&turn_id, out);
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
|
|
fn emit_thread_initialized(
|
|
&mut self,
|
|
connection_id: u64,
|
|
thread: codex_app_server_protocol::Thread,
|
|
model: String,
|
|
initialization_mode: ThreadInitializationMode,
|
|
out: &mut Vec<TrackEventRequest>,
|
|
) {
|
|
let thread_source: SessionSource = thread.source.into();
|
|
let Some(connection_state) = self.connections.get(&connection_id) else {
|
|
return;
|
|
};
|
|
out.push(TrackEventRequest::ThreadInitialized(
|
|
ThreadInitializedEvent {
|
|
event_type: "codex_thread_initialized",
|
|
event_params: ThreadInitializedEventParams {
|
|
thread_id: thread.id,
|
|
app_server_client: connection_state.app_server_client.clone(),
|
|
runtime: connection_state.runtime.clone(),
|
|
model,
|
|
ephemeral: thread.ephemeral,
|
|
thread_source: thread_source_name(&thread_source),
|
|
initialization_mode,
|
|
subagent_source: None,
|
|
parent_thread_id: None,
|
|
created_at: u64::try_from(thread.created_at).unwrap_or_default(),
|
|
},
|
|
},
|
|
));
|
|
}
|
|
|
|
fn maybe_emit_turn_event(&mut self, turn_id: &str, out: &mut Vec<TrackEventRequest>) {
|
|
let Some(turn_state) = self.turns.get(turn_id) else {
|
|
return;
|
|
};
|
|
if turn_state.thread_id.is_none()
|
|
|| turn_state.num_input_images.is_none()
|
|
|| turn_state.resolved_config.is_none()
|
|
|| turn_state.completed.is_none()
|
|
{
|
|
return;
|
|
}
|
|
let product_client_id = turn_state
|
|
.connection_id
|
|
.and_then(|connection_id| self.connections.get(&connection_id))
|
|
.map(|connection_state| connection_state.app_server_client.product_client_id.clone())
|
|
.unwrap_or_else(|| originator().value);
|
|
out.push(TrackEventRequest::TurnEvent(Box::new(
|
|
CodexTurnEventRequest {
|
|
event_type: "codex_turn_event",
|
|
event_params: codex_turn_event_params(
|
|
product_client_id,
|
|
turn_id.to_string(),
|
|
turn_state,
|
|
),
|
|
},
|
|
)));
|
|
self.turns.remove(turn_id);
|
|
}
|
|
}
|
|
|
|
fn codex_turn_event_params(
|
|
product_client_id: String,
|
|
turn_id: String,
|
|
turn_state: &TurnState,
|
|
) -> CodexTurnEventParams {
|
|
let (Some(thread_id), Some(num_input_images), Some(resolved_config), Some(completed)) = (
|
|
turn_state.thread_id.clone(),
|
|
turn_state.num_input_images,
|
|
turn_state.resolved_config.clone(),
|
|
turn_state.completed.clone(),
|
|
) else {
|
|
unreachable!("turn event params require a fully populated turn state");
|
|
};
|
|
let created_at = turn_state.created_at;
|
|
let TurnResolvedConfigFact {
|
|
turn_id: _resolved_turn_id,
|
|
thread_id: _resolved_thread_id,
|
|
num_input_images: _resolved_num_input_images,
|
|
submission_type,
|
|
model,
|
|
model_provider,
|
|
sandbox_policy,
|
|
reasoning_effort,
|
|
reasoning_summary,
|
|
service_tier,
|
|
approval_policy,
|
|
approvals_reviewer,
|
|
sandbox_network_access,
|
|
collaboration_mode,
|
|
personality,
|
|
is_first_turn,
|
|
} = resolved_config;
|
|
let token_usage = turn_state.token_usage.clone();
|
|
CodexTurnEventParams {
|
|
thread_id,
|
|
turn_id,
|
|
product_client_id,
|
|
submission_type,
|
|
model: Some(model),
|
|
model_provider,
|
|
sandbox_policy: Some(sandbox_policy_mode(&sandbox_policy)),
|
|
reasoning_effort: reasoning_effort.map(|value| value.to_string()),
|
|
reasoning_summary: reasoning_summary_mode(reasoning_summary),
|
|
service_tier: service_tier
|
|
.map(|value| value.to_string())
|
|
.unwrap_or_else(|| "default".to_string()),
|
|
approval_policy: approval_policy.to_string(),
|
|
approvals_reviewer: approvals_reviewer.to_string(),
|
|
sandbox_network_access,
|
|
collaboration_mode: Some(collaboration_mode_mode(collaboration_mode)),
|
|
personality: personality_mode(personality),
|
|
num_input_images,
|
|
is_first_turn,
|
|
status: completed.status,
|
|
turn_error: completed.turn_error,
|
|
steer_count: None,
|
|
total_tool_call_count: None,
|
|
shell_command_count: None,
|
|
file_change_count: None,
|
|
mcp_tool_call_count: None,
|
|
dynamic_tool_call_count: None,
|
|
subagent_tool_call_count: None,
|
|
web_search_count: None,
|
|
image_generation_count: None,
|
|
input_tokens: token_usage
|
|
.as_ref()
|
|
.map(|token_usage| token_usage.input_tokens),
|
|
cached_input_tokens: token_usage
|
|
.as_ref()
|
|
.map(|token_usage| token_usage.cached_input_tokens),
|
|
output_tokens: token_usage
|
|
.as_ref()
|
|
.map(|token_usage| token_usage.output_tokens),
|
|
reasoning_output_tokens: token_usage
|
|
.as_ref()
|
|
.map(|token_usage| token_usage.reasoning_output_tokens),
|
|
total_tokens: token_usage
|
|
.as_ref()
|
|
.map(|token_usage| token_usage.total_tokens),
|
|
duration_ms: completed.duration_ms,
|
|
created_at,
|
|
completed_at: Some(completed.completed_at),
|
|
}
|
|
}
|
|
|
|
fn sandbox_policy_mode(sandbox_policy: &SandboxPolicy) -> &'static str {
|
|
match sandbox_policy {
|
|
SandboxPolicy::DangerFullAccess => "full_access",
|
|
SandboxPolicy::ReadOnly { .. } => "read_only",
|
|
SandboxPolicy::WorkspaceWrite { .. } => "workspace_write",
|
|
SandboxPolicy::ExternalSandbox { .. } => "external_sandbox",
|
|
}
|
|
}
|
|
|
|
fn collaboration_mode_mode(mode: ModeKind) -> &'static str {
|
|
match mode {
|
|
ModeKind::Plan => "plan",
|
|
ModeKind::Default | ModeKind::PairProgramming | ModeKind::Execute => "default",
|
|
}
|
|
}
|
|
|
|
fn reasoning_summary_mode(summary: Option<ReasoningSummary>) -> Option<String> {
|
|
match summary {
|
|
Some(ReasoningSummary::None) | None => None,
|
|
Some(summary) => Some(summary.to_string()),
|
|
}
|
|
}
|
|
|
|
fn personality_mode(personality: Option<Personality>) -> Option<String> {
|
|
match personality {
|
|
Some(Personality::None) | None => None,
|
|
Some(personality) => Some(personality.to_string()),
|
|
}
|
|
}
|
|
|
|
fn analytics_turn_status(status: codex_app_server_protocol::TurnStatus) -> Option<TurnStatus> {
|
|
match status {
|
|
codex_app_server_protocol::TurnStatus::Completed => Some(TurnStatus::Completed),
|
|
codex_app_server_protocol::TurnStatus::Failed => Some(TurnStatus::Failed),
|
|
codex_app_server_protocol::TurnStatus::Interrupted => Some(TurnStatus::Interrupted),
|
|
codex_app_server_protocol::TurnStatus::InProgress => None,
|
|
}
|
|
}
|
|
|
|
pub(crate) fn skill_id_for_local_skill(
|
|
repo_url: Option<&str>,
|
|
repo_root: Option<&Path>,
|
|
skill_path: &Path,
|
|
skill_name: &str,
|
|
) -> String {
|
|
let path = normalize_path_for_skill_id(repo_url, repo_root, skill_path);
|
|
let prefix = if let Some(url) = repo_url {
|
|
format!("repo_{url}")
|
|
} else {
|
|
"personal".to_string()
|
|
};
|
|
let raw_id = format!("{prefix}_{path}_{skill_name}");
|
|
let mut hasher = sha1::Sha1::new();
|
|
sha1::Digest::update(&mut hasher, raw_id.as_bytes());
|
|
format!("{:x}", sha1::Digest::finalize(hasher))
|
|
}
|
|
|
|
/// Returns a normalized path for skill ID construction.
|
|
///
|
|
/// - Repo-scoped skills use a path relative to the repo root.
|
|
/// - User/admin/system skills use an absolute path.
|
|
pub(crate) fn normalize_path_for_skill_id(
|
|
repo_url: Option<&str>,
|
|
repo_root: Option<&Path>,
|
|
skill_path: &Path,
|
|
) -> String {
|
|
let resolved_path =
|
|
std::fs::canonicalize(skill_path).unwrap_or_else(|_| skill_path.to_path_buf());
|
|
match (repo_url, repo_root) {
|
|
(Some(_), Some(root)) => {
|
|
let resolved_root = std::fs::canonicalize(root).unwrap_or_else(|_| root.to_path_buf());
|
|
resolved_path
|
|
.strip_prefix(&resolved_root)
|
|
.unwrap_or(resolved_path.as_path())
|
|
.to_string_lossy()
|
|
.replace('\\', "/")
|
|
}
|
|
_ => resolved_path.to_string_lossy().replace('\\', "/"),
|
|
}
|
|
}
|