This commit is contained in:
2026-04-16 17:21:48 +03:00
parent c8fa4c442d
commit c8e7e4e927
116 changed files with 3720 additions and 4197 deletions
+29 -29
View File
@@ -1,58 +1,58 @@
import { registerAs } from '@nestjs/config';
import { registerAs } from "@nestjs/config";
export const appConfig = registerAs('app', () => ({
env: process.env.NODE_ENV || 'development',
port: parseInt(process.env.PORT || '3005', 10),
isDevelopment: process.env.NODE_ENV === 'development',
isProduction: process.env.NODE_ENV === 'production',
export const appConfig = registerAs("app", () => ({
env: process.env.NODE_ENV || "development",
port: parseInt(process.env.PORT || "3005", 10),
isDevelopment: process.env.NODE_ENV === "development",
isProduction: process.env.NODE_ENV === "production",
}));
export const databaseConfig = registerAs('database', () => ({
export const databaseConfig = registerAs("database", () => ({
url: process.env.DATABASE_URL,
}));
export const jwtConfig = registerAs('jwt', () => ({
export const jwtConfig = registerAs("jwt", () => ({
secret: process.env.JWT_SECRET,
accessExpiration: process.env.JWT_ACCESS_EXPIRATION || '15m',
refreshExpiration: process.env.JWT_REFRESH_EXPIRATION || '7d',
accessExpiration: process.env.JWT_ACCESS_EXPIRATION || "15m",
refreshExpiration: process.env.JWT_REFRESH_EXPIRATION || "7d",
}));
export const redisConfig = registerAs('redis', () => ({
enabled: process.env.REDIS_ENABLED === 'true',
host: process.env.REDIS_HOST || 'localhost',
port: parseInt(process.env.REDIS_PORT || '6379', 10),
export const redisConfig = registerAs("redis", () => ({
enabled: process.env.REDIS_ENABLED === "true",
host: process.env.REDIS_HOST || "localhost",
port: parseInt(process.env.REDIS_PORT || "6379", 10),
password: process.env.REDIS_PASSWORD || undefined,
}));
export const i18nConfig = registerAs('i18n', () => ({
defaultLanguage: process.env.DEFAULT_LANGUAGE || 'en',
fallbackLanguage: process.env.FALLBACK_LANGUAGE || 'en',
export const i18nConfig = registerAs("i18n", () => ({
defaultLanguage: process.env.DEFAULT_LANGUAGE || "en",
fallbackLanguage: process.env.FALLBACK_LANGUAGE || "en",
}));
export const featuresConfig = registerAs('features', () => ({
mail: process.env.ENABLE_MAIL === 'true',
s3: process.env.ENABLE_S3 === 'true',
websocket: process.env.ENABLE_WEBSOCKET === 'true',
multiTenancy: process.env.ENABLE_MULTI_TENANCY === 'true',
export const featuresConfig = registerAs("features", () => ({
mail: process.env.ENABLE_MAIL === "true",
s3: process.env.ENABLE_S3 === "true",
websocket: process.env.ENABLE_WEBSOCKET === "true",
multiTenancy: process.env.ENABLE_MULTI_TENANCY === "true",
}));
export const mailConfig = registerAs('mail', () => ({
export const mailConfig = registerAs("mail", () => ({
host: process.env.MAIL_HOST,
port: parseInt(process.env.MAIL_PORT || '587', 10),
port: parseInt(process.env.MAIL_PORT || "587", 10),
user: process.env.MAIL_USER,
password: process.env.MAIL_PASSWORD,
from: process.env.MAIL_FROM,
}));
export const s3Config = registerAs('s3', () => ({
export const s3Config = registerAs("s3", () => ({
endpoint: process.env.S3_ENDPOINT,
accessKey: process.env.S3_ACCESS_KEY,
secretKey: process.env.S3_SECRET_KEY,
bucket: process.env.S3_BUCKET,
region: process.env.S3_REGION || 'us-east-1',
region: process.env.S3_REGION || "us-east-1",
}));
export const throttleConfig = registerAs('throttle', () => ({
ttl: parseInt(process.env.THROTTLE_TTL || '60000', 10),
limit: parseInt(process.env.THROTTLE_LIMIT || '100', 10),
export const throttleConfig = registerAs("throttle", () => ({
ttl: parseInt(process.env.THROTTLE_TTL || "60000", 10),
limit: parseInt(process.env.THROTTLE_LIMIT || "100", 10),
}));
+20 -20
View File
@@ -1,4 +1,4 @@
import { z } from 'zod';
import { z } from "zod";
/**
* Helper to parse boolean from string
@@ -6,8 +6,8 @@ import { z } from 'zod';
const booleanString = z
.string()
.optional()
.default('false')
.transform((val) => val === 'true');
.default("false")
.transform((val) => val === "true");
/**
* Environment variables schema validation using Zod
@@ -15,46 +15,46 @@ const booleanString = z
export const envSchema = z.object({
// Environment
NODE_ENV: z
.enum(['development', 'production', 'test'])
.default('development'),
.enum(["development", "production", "test"])
.default("development"),
PORT: z.coerce.number().default(3005),
// Database
DATABASE_URL: z.string().url(),
// AI Engine
AI_ENGINE_URL: z.string().url().default('http://localhost:8000'),
AI_ENGINE_URL: z.string().url().default("http://localhost:8000"),
// JWT
JWT_SECRET: z.string().min(32),
JWT_ACCESS_EXPIRATION: z.string().default('15m'),
JWT_REFRESH_EXPIRATION: z.string().default('7d'),
JWT_ACCESS_EXPIRATION: z.string().default("15m"),
JWT_REFRESH_EXPIRATION: z.string().default("7d"),
// Redis
REDIS_ENABLED: z
.string()
.transform((val) => val === 'true')
.default('false' as any),
REDIS_HOST: z.string().default('localhost'),
.transform((val) => val === "true")
.default("false" as any),
REDIS_HOST: z.string().default("localhost"),
REDIS_PORT: z.coerce.number().default(6379),
REDIS_PASSWORD: z.string().optional(),
// i18n
DEFAULT_LANGUAGE: z.string().default('en'),
FALLBACK_LANGUAGE: z.string().default('en'),
DEFAULT_LANGUAGE: z.string().default("en"),
FALLBACK_LANGUAGE: z.string().default("en"),
// Gemini AI
ENABLE_GEMINI: z
.string()
.transform((val) => val === 'true')
.default('false' as any),
.transform((val) => val === "true")
.default("false" as any),
GOOGLE_API_KEY: z.string().optional(),
GEMINI_DEFAULT_MODEL: z.string().default('gemini-2.5-flash'),
GEMINI_DEFAULT_MODEL: z.string().default("gemini-2.5-flash"),
// Social Poster
SOCIAL_POSTER_ENABLED: z
.string()
.transform((val) => val === 'true')
.default('false' as any),
.transform((val) => val === "true")
.default("false" as any),
TWITTER_API_KEY: z.string().optional(),
TWITTER_API_SECRET: z.string().optional(),
TWITTER_ACCESS_TOKEN: z.string().optional(),
@@ -98,9 +98,9 @@ export function validateEnv(config: Record<string, unknown>): EnvConfig {
if (!result.success) {
const errors = result.error.issues.map(
(err) => `${err.path.join('.')}: ${err.message}`,
(err) => `${err.path.join(".")}: ${err.message}`,
);
throw new Error(`Environment validation failed:\n${errors.join('\n')}`);
throw new Error(`Environment validation failed:\n${errors.join("\n")}`);
}
return result.data;