systemprompt_models/profile/
from_env.rs1use super::{
2 ContentNegotiationConfig, DatabaseConfig, ExtensionsConfig, PathsConfig, Profile, ProfileError,
3 ProfileResult, ProfileType, RateLimitsConfig, RuntimeConfig, SecurityConfig,
4 SecurityHeadersConfig, ServerConfig, SiteConfig, TierMultipliers, default_agent_registry,
5 default_agents, default_artifacts, default_burst, default_content, default_contexts,
6 default_mcp, default_mcp_registry, default_oauth_auth, default_oauth_public, default_stream,
7 default_tasks,
8};
9
10impl Profile {
11 pub fn from_env(profile_name: &str, display_name: &str) -> ProfileResult<Self> {
12 let db_type = require_env("DATABASE_TYPE")?;
13
14 Ok(Self {
15 name: profile_name.to_string(),
16 display_name: display_name.to_string(),
17 target: ProfileType::Cloud,
18 site: site_config_from_env()?,
19 database: DatabaseConfig {
20 db_type,
21 external_db_access: false,
22 },
23 server: server_config_from_env()?,
24 paths: paths_config_from_env()?,
25 security: security_config_from_env()?,
26 rate_limits: rate_limits_from_env(),
27 runtime: runtime_config_from_env()?,
28 cloud: None,
29 secrets: None,
30 extensions: ExtensionsConfig::default(),
31 gateway: None,
32 })
33 }
34}
35
36fn get_env(key: &str) -> Option<String> {
37 std::env::var(key).ok()
38}
39
40fn require_env(name: &'static str) -> ProfileResult<String> {
41 std::env::var(name).map_err(|_| ProfileError::MissingEnvVar { name })
42}
43
44fn site_config_from_env() -> ProfileResult<SiteConfig> {
45 Ok(SiteConfig {
46 name: require_env("SITENAME")?,
47 github_link: get_env("GITHUB_LINK"),
48 })
49}
50
51fn server_config_from_env() -> ProfileResult<ServerConfig> {
52 let port = require_env("PORT")?
53 .parse()
54 .map_err(|e: std::num::ParseIntError| ProfileError::InvalidEnvVar {
55 name: "PORT",
56 message: e.to_string(),
57 })?;
58
59 Ok(ServerConfig {
60 host: require_env("HOST")?,
61 port,
62 api_server_url: require_env("API_SERVER_URL")?,
63 api_internal_url: require_env("API_INTERNAL_URL")?,
64 api_external_url: require_env("API_EXTERNAL_URL")?,
65 use_https: get_env("USE_HTTPS").is_some_and(|v| v.to_lowercase() == "true"),
66 cors_allowed_origins: get_env("CORS_ALLOWED_ORIGINS").map_or_else(Vec::new, |s| {
67 s.split(',').map(|s| s.trim().to_string()).collect()
68 }),
69 content_negotiation: ContentNegotiationConfig {
70 enabled: get_env("CONTENT_NEGOTIATION_ENABLED")
71 .is_some_and(|v| v.to_lowercase() == "true"),
72 ..Default::default()
73 },
74 security_headers: SecurityHeadersConfig::default(),
75 })
76}
77
78fn paths_config_from_env() -> ProfileResult<PathsConfig> {
79 Ok(PathsConfig {
80 system: require_env("SYSTEM_PATH")?,
81 services: require_env("SYSTEMPROMPT_SERVICES_PATH")?,
82 bin: require_env("BIN_PATH")?,
83 storage: get_env("STORAGE_PATH"),
84 geoip_database: get_env("GEOIP_DATABASE_PATH"),
85 web_path: get_env("SYSTEMPROMPT_WEB_PATH"),
86 })
87}
88
89fn security_config_from_env() -> ProfileResult<SecurityConfig> {
90 use crate::auth::JwtAudience;
91
92 let issuer = require_env("JWT_ISSUER")?;
93
94 let access_token_expiration = require_env("JWT_ACCESS_TOKEN_EXPIRATION")?
95 .parse()
96 .map_err(|e: std::num::ParseIntError| ProfileError::InvalidEnvVar {
97 name: "JWT_ACCESS_TOKEN_EXPIRATION",
98 message: e.to_string(),
99 })?;
100
101 let refresh_token_expiration = require_env("JWT_REFRESH_TOKEN_EXPIRATION")?
102 .parse()
103 .map_err(|e: std::num::ParseIntError| ProfileError::InvalidEnvVar {
104 name: "JWT_REFRESH_TOKEN_EXPIRATION",
105 message: e.to_string(),
106 })?;
107
108 let audiences_raw = require_env("JWT_AUDIENCES")?;
109 let audiences = audiences_raw
110 .split(',')
111 .map(|s| {
112 s.trim()
113 .parse::<JwtAudience>()
114 .map_err(|e| ProfileError::InvalidEnvVar {
115 name: "JWT_AUDIENCES",
116 message: e.to_string(),
117 })
118 })
119 .collect::<ProfileResult<Vec<_>>>()?;
120
121 let allow_registration =
122 get_env("ALLOW_REGISTRATION").is_none_or(|s| s.eq_ignore_ascii_case("true"));
123
124 Ok(SecurityConfig {
125 issuer,
126 access_token_expiration,
127 refresh_token_expiration,
128 audiences,
129 allow_registration,
130 })
131}
132
133fn rate_limits_from_env() -> RateLimitsConfig {
134 let parse_rate = |key: &str, default: fn() -> u64| -> u64 {
135 get_env(key)
136 .and_then(|s| {
137 s.parse()
138 .map_err(|e| {
139 tracing::warn!(key = %key, value = %s, error = %e, "Failed to parse rate limit value");
140 e
141 })
142 .ok()
143 })
144 .unwrap_or_else(default)
145 };
146
147 RateLimitsConfig {
148 disabled: get_env("RATE_LIMIT_DISABLED").is_some_and(|v| v.to_lowercase() == "true"),
149 oauth_public_per_second: parse_rate(
150 "RATE_LIMIT_OAUTH_PUBLIC_PER_SECOND",
151 default_oauth_public,
152 ),
153 oauth_auth_per_second: parse_rate("RATE_LIMIT_OAUTH_AUTH_PER_SECOND", default_oauth_auth),
154 contexts_per_second: parse_rate("RATE_LIMIT_CONTEXTS_PER_SECOND", default_contexts),
155 tasks_per_second: parse_rate("RATE_LIMIT_TASKS_PER_SECOND", default_tasks),
156 artifacts_per_second: parse_rate("RATE_LIMIT_ARTIFACTS_PER_SECOND", default_artifacts),
157 agent_registry_per_second: parse_rate(
158 "RATE_LIMIT_AGENT_REGISTRY_PER_SECOND",
159 default_agent_registry,
160 ),
161 agents_per_second: parse_rate("RATE_LIMIT_AGENTS_PER_SECOND", default_agents),
162 mcp_registry_per_second: parse_rate(
163 "RATE_LIMIT_MCP_REGISTRY_PER_SECOND",
164 default_mcp_registry,
165 ),
166 mcp_per_second: parse_rate("RATE_LIMIT_MCP_PER_SECOND", default_mcp),
167 stream_per_second: parse_rate("RATE_LIMIT_STREAM_PER_SECOND", default_stream),
168 content_per_second: parse_rate("RATE_LIMIT_CONTENT_PER_SECOND", default_content),
169 burst_multiplier: parse_rate("RATE_LIMIT_BURST_MULTIPLIER", default_burst),
170 tier_multipliers: TierMultipliers::default(),
171 }
172}
173
174fn runtime_config_from_env() -> ProfileResult<RuntimeConfig> {
175 let environment = get_env("SYSTEMPROMPT_ENV")
176 .unwrap_or_else(|| "development".to_string())
177 .parse()
178 .map_err(|e: String| ProfileError::InvalidEnvVar {
179 name: "SYSTEMPROMPT_ENV",
180 message: e,
181 })?;
182
183 let log_level = get_env("SYSTEMPROMPT_LOG_LEVEL")
184 .unwrap_or_else(|| "normal".to_string())
185 .parse()
186 .map_err(|e: String| ProfileError::InvalidEnvVar {
187 name: "SYSTEMPROMPT_LOG_LEVEL",
188 message: e,
189 })?;
190
191 let output_format = get_env("SYSTEMPROMPT_OUTPUT_FORMAT")
192 .unwrap_or_else(|| "text".to_string())
193 .parse()
194 .map_err(|e: String| ProfileError::InvalidEnvVar {
195 name: "SYSTEMPROMPT_OUTPUT_FORMAT",
196 message: e,
197 })?;
198
199 Ok(RuntimeConfig {
200 environment,
201 log_level,
202 output_format,
203 no_color: get_env("NO_COLOR").is_some(),
204 non_interactive: get_env("CI").is_some(),
205 })
206}