Class: AgentHarness::Providers::Codex
- Inherits:
-
Base
- Object
- Base
- AgentHarness::Providers::Codex
show all
- Defined in:
- lib/agent_harness/providers/codex.rb
Overview
OpenAI Codex CLI provider
Provides integration with the OpenAI Codex CLI tool.
Constant Summary
Constants inherited
from Base
Base::COMMON_ERROR_PATTERNS
Instance Attribute Summary
Attributes inherited from Base
#config, #executor, #logger
Class Method Summary
collapse
Instance Method Summary
collapse
Methods inherited from Base
#configure, #initialize, #sandboxed_environment?, #send_message
Methods included from Adapter
#auth_type, #build_mcp_flags, #fetch_mcp_servers, included, #parse_rate_limit_reset, #send_message, #supported_mcp_transports, #supports_dangerous_mode?, #supports_mcp?, #validate_mcp_servers!
Class Method Details
.binary_name ⇒ Object
16
17
18
|
# File 'lib/agent_harness/providers/codex.rb', line 16
def binary_name
"codex"
end
|
.discover_models ⇒ Object
45
46
47
48
49
50
51
|
# File 'lib/agent_harness/providers/codex.rb', line 45
def discover_models
return [] unless available?
[
{name: "codex", family: "codex", tier: "standard", provider: "codex"}
]
end
|
.firewall_requirements ⇒ Object
25
26
27
28
29
30
31
32
33
|
# File 'lib/agent_harness/providers/codex.rb', line 25
def firewall_requirements
{
domains: [
"api.openai.com",
"openai.com"
],
ip_ranges: []
}
end
|
.instruction_file_paths ⇒ Object
35
36
37
38
39
40
41
42
43
|
# File 'lib/agent_harness/providers/codex.rb', line 35
def instruction_file_paths
[
{
path: "AGENTS.md",
description: "OpenAI Codex agent instructions",
symlink: false
}
]
end
|
.provider_name ⇒ Object
12
13
14
|
# File 'lib/agent_harness/providers/codex.rb', line 12
def provider_name
:codex
end
|
Instance Method Details
#auth_status ⇒ Object
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
|
# File 'lib/agent_harness/providers/codex.rb', line 112
def auth_status
api_key = ENV["OPENAI_API_KEY"]
if api_key && !api_key.strip.empty?
if api_key.strip.start_with?("sk-")
return {valid: true, expires_at: nil, error: nil, auth_method: :api_key}
else
return {valid: false, expires_at: nil, error: "OPENAI_API_KEY is set but does not appear to be a valid OpenAI API key"}
end
end
credentials = read_codex_credentials
if credentials
key = credentials["api_key"] || credentials["apiKey"] || credentials["OPENAI_API_KEY"]
if key.is_a?(String) && !key.strip.empty?
if key.strip.start_with?("sk-")
return {valid: true, expires_at: nil, error: nil, auth_method: :config_file}
else
return {valid: false, expires_at: nil, error: "Config file API key is set but does not appear to be a valid OpenAI API key"}
end
end
end
{valid: false, expires_at: nil, error: "No OpenAI API key found. Set OPENAI_API_KEY or configure in #{codex_config_path}"}
rescue IOError, JSON::ParserError => e
{valid: false, expires_at: nil, error: e.message}
end
|
#capabilities ⇒ Object
62
63
64
65
66
67
68
69
70
71
72
|
# File 'lib/agent_harness/providers/codex.rb', line 62
def capabilities
{
streaming: false,
file_upload: false,
vision: false,
tool_use: true,
json_mode: false,
mcp: false,
dangerous_mode: true
}
end
|
#dangerous_mode_flags ⇒ Object
74
75
76
|
# File 'lib/agent_harness/providers/codex.rb', line 74
def dangerous_mode_flags
["--full-auto"]
end
|
#display_name ⇒ Object
58
59
60
|
# File 'lib/agent_harness/providers/codex.rb', line 58
def display_name
"OpenAI Codex CLI"
end
|
#error_patterns ⇒ Object
100
101
102
103
104
105
106
107
108
109
110
|
# File 'lib/agent_harness/providers/codex.rb', line 100
def error_patterns
COMMON_ERROR_PATTERNS.merge(
auth_expired: COMMON_ERROR_PATTERNS[:auth_expired] + [/401/, /incorrect.*api.*key/i],
transient: COMMON_ERROR_PATTERNS[:transient] + [/connection.*reset/i],
sandbox_failure: [
/bwrap.*no permissions/i,
/no permissions to create a new namespace/i,
/unprivileged.*namespace/i
]
)
end
|
#execution_semantics ⇒ Object
78
79
80
81
82
83
84
85
86
87
88
89
|
# File 'lib/agent_harness/providers/codex.rb', line 78
def execution_semantics
{
prompt_delivery: :arg,
output_format: :text,
sandbox_aware: true,
uses_subcommand: true,
non_interactive_flag: nil,
legitimate_exit_codes: [0],
stderr_is_diagnostic: true,
parses_rate_limit_reset: false
}
end
|
#health_status ⇒ Object
139
140
141
142
143
144
145
146
147
148
149
150
|
# File 'lib/agent_harness/providers/codex.rb', line 139
def health_status
unless self.class.available?
return {healthy: false, message: "Codex CLI not found in PATH. Install from https://github.com/openai/codex"}
end
auth = auth_status
unless auth[:valid]
return {healthy: false, message: auth[:error]}
end
{healthy: true, message: "Codex CLI available and authenticated"}
end
|
#name ⇒ Object
54
55
56
|
# File 'lib/agent_harness/providers/codex.rb', line 54
def name
"codex"
end
|
#session_flags(session_id) ⇒ Object
95
96
97
98
|
# File 'lib/agent_harness/providers/codex.rb', line 95
def session_flags(session_id)
return [] unless session_id && !session_id.empty?
["--session", session_id]
end
|
#supports_sessions? ⇒ Boolean
91
92
93
|
# File 'lib/agent_harness/providers/codex.rb', line 91
def supports_sessions?
true
end
|
#validate_config ⇒ Object
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
|
# File 'lib/agent_harness/providers/codex.rb', line 152
def validate_config
errors = []
flags = @config.default_flags
unless flags.nil?
if flags.is_a?(Array)
invalid = flags.reject { |f| f.is_a?(String) }
errors << "default_flags contains non-string values" if invalid.any?
else
errors << "default_flags must be an array of strings"
end
end
{valid: errors.empty?, errors: errors}
end
|