mirror of
https://github.com/graphql-hive/console
synced 2026-05-24 09:38:26 +00:00
163 lines
4.9 KiB
TypeScript
163 lines
4.9 KiB
TypeScript
import zod from 'zod';
|
|
import { OpenTelemetryConfigurationModel } from '@hive/service-common';
|
|
|
|
const isNumberString = (input: unknown) => zod.string().regex(/^\d+$/).safeParse(input).success;
|
|
|
|
const numberFromNumberOrNumberString = (input: unknown): number | undefined => {
|
|
if (typeof input == 'number') return input;
|
|
if (isNumberString(input)) return Number(input);
|
|
};
|
|
|
|
const NumberFromString = zod.preprocess(numberFromNumberOrNumberString, zod.number().min(1));
|
|
|
|
// treat an empty string (`''`) as undefined
|
|
const emptyString = <T extends zod.ZodType>(input: T) => {
|
|
return zod.preprocess((value: unknown) => {
|
|
if (value === '') return undefined;
|
|
return value;
|
|
}, input);
|
|
};
|
|
|
|
const EnvironmentModel = zod.object({
|
|
PORT: emptyString(NumberFromString.optional()),
|
|
ENVIRONMENT: emptyString(zod.string().optional()),
|
|
RELEASE: emptyString(zod.string().optional()),
|
|
HEARTBEAT_ENDPOINT: emptyString(zod.string().url().optional()),
|
|
});
|
|
|
|
const RedisModel = zod.object({
|
|
REDIS_HOST: zod.string(),
|
|
REDIS_PORT: NumberFromString,
|
|
REDIS_PASSWORD: emptyString(zod.string().optional()),
|
|
});
|
|
|
|
const RequestBrokerModel = zod.union([
|
|
zod.object({
|
|
REQUEST_BROKER: emptyString(zod.literal('0').optional()),
|
|
}),
|
|
zod.object({
|
|
REQUEST_BROKER: zod.literal('1'),
|
|
REQUEST_BROKER_ENDPOINT: zod.string().min(1),
|
|
REQUEST_BROKER_SIGNATURE: zod.string().min(1),
|
|
}),
|
|
]);
|
|
|
|
const SentryModel = zod.union([
|
|
zod.object({
|
|
SENTRY: emptyString(zod.literal('0').optional()),
|
|
}),
|
|
zod.object({
|
|
SENTRY: zod.literal('1'),
|
|
SENTRY_DSN: zod.string(),
|
|
}),
|
|
]);
|
|
|
|
const PrometheusModel = zod.object({
|
|
PROMETHEUS_METRICS: emptyString(zod.union([zod.literal('0'), zod.literal('1')]).optional()),
|
|
PROMETHEUS_METRICS_LABEL_INSTANCE: emptyString(zod.string().optional()),
|
|
PROMETHEUS_METRICS_PORT: emptyString(NumberFromString.optional()),
|
|
});
|
|
|
|
const LogModel = zod.object({
|
|
LOG_LEVEL: emptyString(
|
|
zod
|
|
.union([
|
|
zod.literal('trace'),
|
|
zod.literal('debug'),
|
|
zod.literal('info'),
|
|
zod.literal('warn'),
|
|
zod.literal('error'),
|
|
zod.literal('fatal'),
|
|
zod.literal('silent'),
|
|
])
|
|
.optional(),
|
|
),
|
|
REQUEST_LOGGING: emptyString(zod.union([zod.literal('0'), zod.literal('1')]).optional()).default(
|
|
'1',
|
|
),
|
|
});
|
|
|
|
const configs = {
|
|
// eslint-disable-next-line no-process-env
|
|
base: EnvironmentModel.safeParse(process.env),
|
|
// eslint-disable-next-line no-process-env
|
|
redis: RedisModel.safeParse(process.env),
|
|
// eslint-disable-next-line no-process-env
|
|
sentry: SentryModel.safeParse(process.env),
|
|
// eslint-disable-next-line no-process-env
|
|
prometheus: PrometheusModel.safeParse(process.env),
|
|
// eslint-disable-next-line no-process-env
|
|
log: LogModel.safeParse(process.env),
|
|
// eslint-disable-next-line no-process-env
|
|
requestBroker: RequestBrokerModel.safeParse(process.env),
|
|
// eslint-disable-next-line no-process-env
|
|
tracing: OpenTelemetryConfigurationModel.safeParse(process.env),
|
|
};
|
|
|
|
const environmentErrors: Array<string> = [];
|
|
|
|
for (const config of Object.values(configs)) {
|
|
if (config.success === false) {
|
|
environmentErrors.push(JSON.stringify(config.error.format(), null, 4));
|
|
}
|
|
}
|
|
|
|
if (environmentErrors.length) {
|
|
const fullError = environmentErrors.join(`\n`);
|
|
console.error('❌ Invalid environment variables:', fullError);
|
|
process.exit(1);
|
|
}
|
|
|
|
function extractConfig<Input, Output>(config: zod.SafeParseReturnType<Input, Output>): Output {
|
|
if (!config.success) {
|
|
throw new Error('Something went wrong.');
|
|
}
|
|
return config.data;
|
|
}
|
|
|
|
const base = extractConfig(configs.base);
|
|
const redis = extractConfig(configs.redis);
|
|
const sentry = extractConfig(configs.sentry);
|
|
const prometheus = extractConfig(configs.prometheus);
|
|
const log = extractConfig(configs.log);
|
|
const requestBroker = extractConfig(configs.requestBroker);
|
|
const tracing = extractConfig(configs.tracing);
|
|
|
|
export const env = {
|
|
environment: base.ENVIRONMENT,
|
|
release: base.RELEASE ?? 'local',
|
|
http: {
|
|
port: base.PORT ?? 6250,
|
|
},
|
|
tracing: {
|
|
enabled: !!tracing.OPENTELEMETRY_COLLECTOR_ENDPOINT,
|
|
collectorEndpoint: tracing.OPENTELEMETRY_COLLECTOR_ENDPOINT,
|
|
},
|
|
redis: {
|
|
host: redis.REDIS_HOST,
|
|
port: redis.REDIS_PORT,
|
|
password: redis.REDIS_PASSWORD ?? '',
|
|
},
|
|
heartbeat: base.HEARTBEAT_ENDPOINT ? { endpoint: base.HEARTBEAT_ENDPOINT } : null,
|
|
sentry: sentry.SENTRY === '1' ? { dsn: sentry.SENTRY_DSN } : null,
|
|
log: {
|
|
level: log.LOG_LEVEL ?? 'info',
|
|
requests: log.REQUEST_LOGGING === '1',
|
|
},
|
|
prometheus:
|
|
prometheus.PROMETHEUS_METRICS === '1'
|
|
? {
|
|
labels: {
|
|
instance: prometheus.PROMETHEUS_METRICS_LABEL_INSTANCE ?? 'usage-service',
|
|
},
|
|
port: prometheus.PROMETHEUS_METRICS_PORT ?? 10_254,
|
|
}
|
|
: null,
|
|
requestBroker:
|
|
requestBroker.REQUEST_BROKER === '1'
|
|
? {
|
|
endpoint: requestBroker.REQUEST_BROKER_ENDPOINT,
|
|
signature: requestBroker.REQUEST_BROKER_SIGNATURE,
|
|
}
|
|
: null,
|
|
} as const;
|