mirror of
https://github.com/sourcebot-dev/sourcebot.git
synced 2025-12-12 04:15:30 +00:00
Some checks failed
Publish to ghcr / build (linux/amd64, blacksmith-4vcpu-ubuntu-2404) (push) Has been cancelled
Publish to ghcr / build (linux/arm64, blacksmith-8vcpu-ubuntu-2204-arm) (push) Has been cancelled
Update Roadmap Released / update (push) Has been cancelled
Publish to ghcr / merge (push) Has been cancelled
231 lines
No EOL
9.1 KiB
TypeScript
231 lines
No EOL
9.1 KiB
TypeScript
import { createEnv } from "@t3-oss/env-core";
|
|
import { z } from "zod";
|
|
import { loadConfig } from "./utils.js";
|
|
import { tenancyModeSchema } from "./types.js";
|
|
import { SourcebotConfig } from "@sourcebot/schemas/v3/index.type";
|
|
import { getTokenFromConfig } from "./crypto.js";
|
|
|
|
// Booleans are specified as 'true' or 'false' strings.
|
|
const booleanSchema = z.enum(["true", "false"]);
|
|
|
|
// Numbers are treated as strings in .env files.
|
|
// coerce helps us convert them to numbers.
|
|
// @see: https://zod.dev/?id=coercion-for-primitives
|
|
const numberSchema = z.coerce.number();
|
|
|
|
export const resolveEnvironmentVariableOverridesFromConfig = async (config: SourcebotConfig): Promise<Record<string, string>> => {
|
|
if (!config.environmentOverrides) {
|
|
return {};
|
|
}
|
|
|
|
const resolved: Record<string, string> = {};
|
|
|
|
const start = performance.now();
|
|
|
|
for (const [key, override] of Object.entries(config.environmentOverrides)) {
|
|
switch (override.type) {
|
|
case 'token':
|
|
resolved[key] = await getTokenFromConfig(override.value);
|
|
break;
|
|
case 'boolean':
|
|
resolved[key] = override.value ? 'true' : 'false';
|
|
break;
|
|
case 'number':
|
|
resolved[key] = override.value.toString();
|
|
break;
|
|
case 'string':
|
|
resolved[key] = override.value;
|
|
break;
|
|
}
|
|
}
|
|
|
|
const end = performance.now();
|
|
console.debug(`resolved environment variable overrides in ${end - start}ms`);
|
|
|
|
return resolved;
|
|
}
|
|
|
|
// Merge process.env with environment variables resolved from config.json
|
|
const runtimeEnv = await (async () => {
|
|
const configPath = process.env.CONFIG_PATH;
|
|
if (!configPath) {
|
|
return process.env;
|
|
}
|
|
|
|
const config = await loadConfig(configPath);
|
|
const overrides = await resolveEnvironmentVariableOverridesFromConfig(config);
|
|
return {
|
|
...process.env,
|
|
...overrides,
|
|
}
|
|
})();
|
|
|
|
export const env = createEnv({
|
|
server: {
|
|
// Zoekt
|
|
ZOEKT_WEBSERVER_URL: z.string().url().default("http://localhost:6070"),
|
|
|
|
// Auth
|
|
FORCE_ENABLE_ANONYMOUS_ACCESS: booleanSchema.default('false'),
|
|
AUTH_SECRET: z.string(),
|
|
AUTH_URL: z.string().url(),
|
|
AUTH_CREDENTIALS_LOGIN_ENABLED: booleanSchema.default('true'),
|
|
AUTH_EMAIL_CODE_LOGIN_ENABLED: booleanSchema.default('false'),
|
|
|
|
// Enterprise Auth
|
|
AUTH_EE_GITHUB_CLIENT_ID: z.string().optional(),
|
|
AUTH_EE_GITHUB_CLIENT_SECRET: z.string().optional(),
|
|
AUTH_EE_GITHUB_BASE_URL: z.string().optional(),
|
|
|
|
AUTH_EE_GITLAB_CLIENT_ID: z.string().optional(),
|
|
AUTH_EE_GITLAB_CLIENT_SECRET: z.string().optional(),
|
|
AUTH_EE_GITLAB_BASE_URL: z.string().default("https://gitlab.com"),
|
|
|
|
AUTH_EE_GOOGLE_CLIENT_ID: z.string().optional(),
|
|
AUTH_EE_GOOGLE_CLIENT_SECRET: z.string().optional(),
|
|
|
|
AUTH_EE_OKTA_CLIENT_ID: z.string().optional(),
|
|
AUTH_EE_OKTA_CLIENT_SECRET: z.string().optional(),
|
|
AUTH_EE_OKTA_ISSUER: z.string().optional(),
|
|
|
|
AUTH_EE_KEYCLOAK_CLIENT_ID: z.string().optional(),
|
|
AUTH_EE_KEYCLOAK_CLIENT_SECRET: z.string().optional(),
|
|
AUTH_EE_KEYCLOAK_ISSUER: z.string().optional(),
|
|
|
|
AUTH_EE_MICROSOFT_ENTRA_ID_CLIENT_ID: z.string().optional(),
|
|
AUTH_EE_MICROSOFT_ENTRA_ID_CLIENT_SECRET: z.string().optional(),
|
|
AUTH_EE_MICROSOFT_ENTRA_ID_ISSUER: z.string().optional(),
|
|
|
|
AUTH_EE_GCP_IAP_ENABLED: booleanSchema.default('false'),
|
|
AUTH_EE_GCP_IAP_AUDIENCE: z.string().optional(),
|
|
|
|
DATA_CACHE_DIR: z.string(),
|
|
|
|
SOURCEBOT_PUBLIC_KEY_PATH: z.string(),
|
|
|
|
// Email
|
|
SMTP_CONNECTION_URL: z.string().url().optional(),
|
|
EMAIL_FROM_ADDRESS: z.string().email().optional(),
|
|
|
|
// Stripe
|
|
STRIPE_SECRET_KEY: z.string().optional(),
|
|
STRIPE_PRODUCT_ID: z.string().optional(),
|
|
STRIPE_WEBHOOK_SECRET: z.string().optional(),
|
|
STRIPE_ENABLE_TEST_CLOCKS: booleanSchema.default('false'),
|
|
|
|
LOGTAIL_TOKEN: z.string().optional(),
|
|
LOGTAIL_HOST: z.string().url().optional(),
|
|
|
|
// Misc
|
|
CONFIG_MAX_REPOS_NO_TOKEN: numberSchema.default(Number.MAX_SAFE_INTEGER),
|
|
NODE_ENV: z.enum(["development", "test", "production"]),
|
|
SOURCEBOT_TELEMETRY_DISABLED: booleanSchema.default('false'),
|
|
// @note: this is also declared in the Dockerfile.
|
|
POSTHOG_PAPIK: z.string().default("phc_lLPuFFi5LH6c94eFJcqvYVFwiJffVcV6HD8U4a1OnRW"),
|
|
|
|
// Database variables
|
|
// Either DATABASE_URL or DATABASE_HOST, DATABASE_USERNAME, DATABASE_PASSWORD, and DATABASE_NAME must be set.
|
|
// @see: shared/src/db.ts for more details.
|
|
DATABASE_URL: z.string().url().optional(),
|
|
DATABASE_HOST: z.string().optional(),
|
|
DATABASE_USERNAME: z.string().optional(),
|
|
DATABASE_PASSWORD: z.string().optional(),
|
|
DATABASE_NAME: z.string().optional(),
|
|
DATABASE_ARGS: z.string().optional(),
|
|
|
|
SOURCEBOT_TENANCY_MODE: tenancyModeSchema.default("single"),
|
|
CONFIG_PATH: z.string(),
|
|
|
|
// Misc UI flags
|
|
SECURITY_CARD_ENABLED: booleanSchema.default('false'),
|
|
|
|
// EE License
|
|
SOURCEBOT_EE_LICENSE_KEY: z.string().optional(),
|
|
SOURCEBOT_EE_AUDIT_LOGGING_ENABLED: booleanSchema.default('true'),
|
|
|
|
// GitHub app for review agent
|
|
GITHUB_REVIEW_AGENT_APP_ID: z.string().optional(),
|
|
GITHUB_REVIEW_AGENT_APP_WEBHOOK_SECRET: z.string().optional(),
|
|
GITHUB_REVIEW_AGENT_APP_PRIVATE_KEY_PATH: z.string().optional(),
|
|
REVIEW_AGENT_API_KEY: z.string().optional(),
|
|
REVIEW_AGENT_LOGGING_ENABLED: booleanSchema.default('true'),
|
|
REVIEW_AGENT_AUTO_REVIEW_ENABLED: booleanSchema.default('false'),
|
|
REVIEW_AGENT_REVIEW_COMMAND: z.string().default('review'),
|
|
|
|
ANTHROPIC_API_KEY: z.string().optional(),
|
|
ANTHROPIC_THINKING_BUDGET_TOKENS: numberSchema.default(12000),
|
|
|
|
AZURE_API_KEY: z.string().optional(),
|
|
AZURE_RESOURCE_NAME: z.string().optional(),
|
|
|
|
DEEPSEEK_API_KEY: z.string().optional(),
|
|
|
|
OPENAI_API_KEY: z.string().optional(),
|
|
|
|
OPENROUTER_API_KEY: z.string().optional(),
|
|
|
|
XAI_API_KEY: z.string().optional(),
|
|
|
|
MISTRAL_API_KEY: z.string().optional(),
|
|
|
|
GOOGLE_GENERATIVE_AI_API_KEY: z.string().optional(),
|
|
GOOGLE_VERTEX_PROJECT: z.string().optional(),
|
|
GOOGLE_VERTEX_REGION: z.string().default('us-central1'),
|
|
GOOGLE_APPLICATION_CREDENTIALS: z.string().optional(),
|
|
GOOGLE_VERTEX_THINKING_BUDGET_TOKENS: numberSchema.default(-1),
|
|
GOOGLE_VERTEX_INCLUDE_THOUGHTS: booleanSchema.default('true'),
|
|
|
|
AWS_ACCESS_KEY_ID: z.string().optional(),
|
|
AWS_SECRET_ACCESS_KEY: z.string().optional(),
|
|
AWS_SESSION_TOKEN: z.string().optional(),
|
|
AWS_REGION: z.string().optional(),
|
|
|
|
SOURCEBOT_CHAT_MODEL_TEMPERATURE: numberSchema.default(0.3),
|
|
SOURCEBOT_CHAT_MAX_STEP_COUNT: numberSchema.default(20),
|
|
|
|
DEBUG_WRITE_CHAT_MESSAGES_TO_FILE: booleanSchema.default('false'),
|
|
|
|
LANGFUSE_SECRET_KEY: z.string().optional(),
|
|
|
|
SOURCEBOT_DEMO_EXAMPLES_PATH: z.string().optional(),
|
|
|
|
// Experimental Environment Variables
|
|
// @note: These environment variables are subject to change at any time and are not garunteed to be backwards compatible.
|
|
EXPERIMENT_DISABLE_API_KEY_CREATION_FOR_NON_ADMIN_USERS: booleanSchema.default('false'),
|
|
EXPERIMENT_SELF_SERVE_REPO_INDEXING_ENABLED: booleanSchema.default('false'),
|
|
// @NOTE: Take care to update actions.ts when changing the name of this.
|
|
EXPERIMENT_SELF_SERVE_REPO_INDEXING_GITHUB_TOKEN: z.string().optional(),
|
|
EXPERIMENT_EE_PERMISSION_SYNC_ENABLED: booleanSchema.default('false'),
|
|
|
|
SOURCEBOT_ENCRYPTION_KEY: z.string(),
|
|
SOURCEBOT_INSTALL_ID: z.string().default("unknown"),
|
|
|
|
FALLBACK_GITHUB_CLOUD_TOKEN: z.string().optional(),
|
|
FALLBACK_GITLAB_CLOUD_TOKEN: z.string().optional(),
|
|
FALLBACK_GITEA_CLOUD_TOKEN: z.string().optional(),
|
|
|
|
REDIS_URL: z.string().url().default("redis://localhost:6379"),
|
|
REDIS_REMOVE_ON_COMPLETE: numberSchema.default(0),
|
|
REDIS_REMOVE_ON_FAIL: numberSchema.default(100),
|
|
|
|
DEBUG_ENABLE_GROUPMQ_LOGGING: booleanSchema.default('false'),
|
|
|
|
CONNECTION_MANAGER_UPSERT_TIMEOUT_MS: numberSchema.default(300000),
|
|
REPO_SYNC_RETRY_BASE_SLEEP_SECONDS: numberSchema.default(60),
|
|
|
|
GITLAB_CLIENT_QUERY_TIMEOUT_SECONDS: numberSchema.default(60 * 10),
|
|
|
|
SOURCEBOT_LOG_LEVEL: z.enum(["info", "debug", "warn", "error"]).default("info"),
|
|
SOURCEBOT_STRUCTURED_LOGGING_ENABLED: booleanSchema.default("false"),
|
|
SOURCEBOT_STRUCTURED_LOGGING_FILE: z.string().optional(),
|
|
|
|
// Configure the default maximum number of search results to return by default.
|
|
DEFAULT_MAX_MATCH_COUNT: numberSchema.default(10_000),
|
|
|
|
// A comma separated list of glob patterns that shwould always be indexed regardless of their size.
|
|
ALWAYS_INDEX_FILE_PATTERNS: z.string().optional(),
|
|
},
|
|
runtimeEnv,
|
|
emptyStringAsUndefined: true,
|
|
skipValidation: process.env.SKIP_ENV_VALIDATION === "1",
|
|
}); |