@venturekit/core 0.0.0-dev.20260307234057

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,54 @@
1
+ /**
2
+ * VentureKit Security Configuration Types
3
+ *
4
+ * OAuth scopes, app clients, and authentication settings.
5
+ * Separated from base.ts to keep config files manageable.
6
+ */
7
+ /**
8
+ * OAuth scope definition
9
+ */
10
+ export interface OAuthScope {
11
+ /** Scope name (e.g., 'projects.read', 'users.write') */
12
+ name: string;
13
+ /** Scope description */
14
+ description: string;
15
+ }
16
+ /**
17
+ * App client configuration
18
+ */
19
+ export interface AppClient {
20
+ /** Client name (e.g., 'web-app', 'mobile-app', 'admin-dashboard') */
21
+ name: string;
22
+ /** Allowed OAuth scopes for this client */
23
+ allowedScopes: string[];
24
+ /** Whether this client supports refresh tokens */
25
+ supportsRefreshTokens: boolean;
26
+ /** Generate client secret (true for server-side, false for SPA/mobile) */
27
+ generateSecret?: boolean;
28
+ /** Access token validity in hours (default: 1) */
29
+ accessTokenValidityHours?: number;
30
+ /** Refresh token validity in days (default: 30) */
31
+ refreshTokenValidityDays?: number;
32
+ }
33
+ /**
34
+ * Security configuration - authentication and authorization settings
35
+ *
36
+ * Defined ONCE and shared across all environments.
37
+ * Keep separate from base.ts to avoid config file bloat.
38
+ */
39
+ export interface SecurityConfig {
40
+ /** All available OAuth scopes for this project */
41
+ scopes: OAuthScope[];
42
+ /** App clients and their allowed scopes */
43
+ appClients: AppClient[];
44
+ /** MFA setting (default: 'optional') */
45
+ mfa?: 'off' | 'optional' | 'required';
46
+ /** Password policy */
47
+ passwordPolicy?: {
48
+ minLength?: number;
49
+ requireUppercase?: boolean;
50
+ requireLowercase?: boolean;
51
+ requireNumbers?: boolean;
52
+ requireSymbols?: boolean;
53
+ };
54
+ }
@@ -0,0 +1,9 @@
1
+ "use strict";
2
+ /**
3
+ * VentureKit Security Configuration Types
4
+ *
5
+ * OAuth scopes, app clients, and authentication settings.
6
+ * Separated from base.ts to keep config files manageable.
7
+ */
8
+ Object.defineProperty(exports, "__esModule", { value: true });
9
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VjdXJpdHkuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdHlwZXMvc2VjdXJpdHkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7OztHQUtHIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBWZW50dXJlS2l0IFNlY3VyaXR5IENvbmZpZ3VyYXRpb24gVHlwZXNcbiAqIFxuICogT0F1dGggc2NvcGVzLCBhcHAgY2xpZW50cywgYW5kIGF1dGhlbnRpY2F0aW9uIHNldHRpbmdzLlxuICogU2VwYXJhdGVkIGZyb20gYmFzZS50cyB0byBrZWVwIGNvbmZpZyBmaWxlcyBtYW5hZ2VhYmxlLlxuICovXG5cbi8qKlxuICogT0F1dGggc2NvcGUgZGVmaW5pdGlvblxuICovXG5leHBvcnQgaW50ZXJmYWNlIE9BdXRoU2NvcGUge1xuICAvKiogU2NvcGUgbmFtZSAoZS5nLiwgJ3Byb2plY3RzLnJlYWQnLCAndXNlcnMud3JpdGUnKSAqL1xuICBuYW1lOiBzdHJpbmc7XG4gIC8qKiBTY29wZSBkZXNjcmlwdGlvbiAqL1xuICBkZXNjcmlwdGlvbjogc3RyaW5nO1xufVxuXG4vKipcbiAqIEFwcCBjbGllbnQgY29uZmlndXJhdGlvblxuICovXG5leHBvcnQgaW50ZXJmYWNlIEFwcENsaWVudCB7XG4gIC8qKiBDbGllbnQgbmFtZSAoZS5nLiwgJ3dlYi1hcHAnLCAnbW9iaWxlLWFwcCcsICdhZG1pbi1kYXNoYm9hcmQnKSAqL1xuICBuYW1lOiBzdHJpbmc7XG4gIC8qKiBBbGxvd2VkIE9BdXRoIHNjb3BlcyBmb3IgdGhpcyBjbGllbnQgKi9cbiAgYWxsb3dlZFNjb3Blczogc3RyaW5nW107XG4gIC8qKiBXaGV0aGVyIHRoaXMgY2xpZW50IHN1cHBvcnRzIHJlZnJlc2ggdG9rZW5zICovXG4gIHN1cHBvcnRzUmVmcmVzaFRva2VuczogYm9vbGVhbjtcbiAgLyoqIEdlbmVyYXRlIGNsaWVudCBzZWNyZXQgKHRydWUgZm9yIHNlcnZlci1zaWRlLCBmYWxzZSBmb3IgU1BBL21vYmlsZSkgKi9cbiAgZ2VuZXJhdGVTZWNyZXQ/OiBib29sZWFuO1xuICAvKiogQWNjZXNzIHRva2VuIHZhbGlkaXR5IGluIGhvdXJzIChkZWZhdWx0OiAxKSAqL1xuICBhY2Nlc3NUb2tlblZhbGlkaXR5SG91cnM/OiBudW1iZXI7XG4gIC8qKiBSZWZyZXNoIHRva2VuIHZhbGlkaXR5IGluIGRheXMgKGRlZmF1bHQ6IDMwKSAqL1xuICByZWZyZXNoVG9rZW5WYWxpZGl0eURheXM/OiBudW1iZXI7XG59XG5cbi8qKlxuICogU2VjdXJpdHkgY29uZmlndXJhdGlvbiAtIGF1dGhlbnRpY2F0aW9uIGFuZCBhdXRob3JpemF0aW9uIHNldHRpbmdzXG4gKiBcbiAqIERlZmluZWQgT05DRSBhbmQgc2hhcmVkIGFjcm9zcyBhbGwgZW52aXJvbm1lbnRzLlxuICogS2VlcCBzZXBhcmF0ZSBmcm9tIGJhc2UudHMgdG8gYXZvaWQgY29uZmlnIGZpbGUgYmxvYXQuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgU2VjdXJpdHlDb25maWcge1xuICAvKiogQWxsIGF2YWlsYWJsZSBPQXV0aCBzY29wZXMgZm9yIHRoaXMgcHJvamVjdCAqL1xuICBzY29wZXM6IE9BdXRoU2NvcGVbXTtcbiAgXG4gIC8qKiBBcHAgY2xpZW50cyBhbmQgdGhlaXIgYWxsb3dlZCBzY29wZXMgKi9cbiAgYXBwQ2xpZW50czogQXBwQ2xpZW50W107XG4gIFxuICAvKiogTUZBIHNldHRpbmcgKGRlZmF1bHQ6ICdvcHRpb25hbCcpICovXG4gIG1mYT86ICdvZmYnIHwgJ29wdGlvbmFsJyB8ICdyZXF1aXJlZCc7XG4gIFxuICAvKiogUGFzc3dvcmQgcG9saWN5ICovXG4gIHBhc3N3b3JkUG9saWN5Pzoge1xuICAgIG1pbkxlbmd0aD86IG51bWJlcjtcbiAgICByZXF1aXJlVXBwZXJjYXNlPzogYm9vbGVhbjtcbiAgICByZXF1aXJlTG93ZXJjYXNlPzogYm9vbGVhbjtcbiAgICByZXF1aXJlTnVtYmVycz86IGJvb2xlYW47XG4gICAgcmVxdWlyZVN5bWJvbHM/OiBib29sZWFuO1xuICB9O1xufVxuIl19
@@ -0,0 +1,65 @@
1
+ /**
2
+ * VPC Environment Configuration
3
+ *
4
+ * Controls networking per environment.
5
+ * All fields are REQUIRED - no implicit defaults at the type level.
6
+ */
7
+ /**
8
+ * Subnet configuration
9
+ */
10
+ export interface SubnetConfig {
11
+ /** CIDR mask for subnets (e.g., 24 for /24) */
12
+ cidrMask: number;
13
+ }
14
+ /**
15
+ * Security group rule
16
+ */
17
+ export interface SecurityGroupRule {
18
+ /** Protocol (tcp, udp, icmp, -1 for all) */
19
+ protocol: string;
20
+ /** Port range start */
21
+ fromPort: number;
22
+ /** Port range end */
23
+ toPort: number;
24
+ /** CIDR blocks to allow */
25
+ cidrBlocks?: string[];
26
+ /** Description */
27
+ description?: string;
28
+ }
29
+ /**
30
+ * VPC configuration
31
+ *
32
+ * All fields required - use presets or explicit values.
33
+ */
34
+ export interface VpcEnvConfig {
35
+ /** Number of availability zones (1-3) */
36
+ maxAzs: number;
37
+ /** Number of NAT gateways (0 = no private subnet internet access) */
38
+ natGateways: number;
39
+ /** CIDR block for the VPC */
40
+ cidr: string;
41
+ /** Enable VPC flow logs for network monitoring */
42
+ flowLogsEnabled: boolean;
43
+ /** Flow logs retention in days */
44
+ flowLogsRetentionDays: number;
45
+ /** Public subnet configuration */
46
+ publicSubnets: SubnetConfig;
47
+ /** Private subnet configuration (for Lambda, RDS) */
48
+ privateSubnets: SubnetConfig;
49
+ /** Isolated subnet configuration (no internet, for RDS) */
50
+ isolatedSubnets: SubnetConfig | undefined;
51
+ /** Enable VPC endpoints for AWS services (S3, DynamoDB, etc.) */
52
+ enableVpcEndpoints: boolean;
53
+ /** Additional ingress rules for Lambda security group */
54
+ lambdaSecurityGroupIngress: SecurityGroupRule[];
55
+ /** Additional egress rules for Lambda security group */
56
+ lambdaSecurityGroupEgress: SecurityGroupRule[];
57
+ }
58
+ /**
59
+ * VPC configuration input (for user overrides)
60
+ */
61
+ export type VpcEnvConfigInput = Partial<Omit<VpcEnvConfig, 'publicSubnets' | 'privateSubnets' | 'isolatedSubnets'>> & {
62
+ publicSubnets?: Partial<SubnetConfig>;
63
+ privateSubnets?: Partial<SubnetConfig>;
64
+ isolatedSubnets?: Partial<SubnetConfig>;
65
+ };
@@ -0,0 +1,9 @@
1
+ "use strict";
2
+ /**
3
+ * VPC Environment Configuration
4
+ *
5
+ * Controls networking per environment.
6
+ * All fields are REQUIRED - no implicit defaults at the type level.
7
+ */
8
+ Object.defineProperty(exports, "__esModule", { value: true });
9
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidnBjLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3R5cGVzL3ZwYy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7Ozs7O0dBS0ciLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFZQQyBFbnZpcm9ubWVudCBDb25maWd1cmF0aW9uXG4gKiBcbiAqIENvbnRyb2xzIG5ldHdvcmtpbmcgcGVyIGVudmlyb25tZW50LlxuICogQWxsIGZpZWxkcyBhcmUgUkVRVUlSRUQgLSBubyBpbXBsaWNpdCBkZWZhdWx0cyBhdCB0aGUgdHlwZSBsZXZlbC5cbiAqL1xuXG4vKipcbiAqIFN1Ym5ldCBjb25maWd1cmF0aW9uXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgU3VibmV0Q29uZmlnIHtcbiAgLyoqIENJRFIgbWFzayBmb3Igc3VibmV0cyAoZS5nLiwgMjQgZm9yIC8yNCkgKi9cbiAgY2lkck1hc2s6IG51bWJlcjtcbn1cblxuLyoqXG4gKiBTZWN1cml0eSBncm91cCBydWxlXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgU2VjdXJpdHlHcm91cFJ1bGUge1xuICAvKiogUHJvdG9jb2wgKHRjcCwgdWRwLCBpY21wLCAtMSBmb3IgYWxsKSAqL1xuICBwcm90b2NvbDogc3RyaW5nO1xuICAvKiogUG9ydCByYW5nZSBzdGFydCAqL1xuICBmcm9tUG9ydDogbnVtYmVyO1xuICAvKiogUG9ydCByYW5nZSBlbmQgKi9cbiAgdG9Qb3J0OiBudW1iZXI7XG4gIC8qKiBDSURSIGJsb2NrcyB0byBhbGxvdyAqL1xuICBjaWRyQmxvY2tzPzogc3RyaW5nW107XG4gIC8qKiBEZXNjcmlwdGlvbiAqL1xuICBkZXNjcmlwdGlvbj86IHN0cmluZztcbn1cblxuLyoqXG4gKiBWUEMgY29uZmlndXJhdGlvblxuICogXG4gKiBBbGwgZmllbGRzIHJlcXVpcmVkIC0gdXNlIHByZXNldHMgb3IgZXhwbGljaXQgdmFsdWVzLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFZwY0VudkNvbmZpZyB7XG4gIC8qKiBOdW1iZXIgb2YgYXZhaWxhYmlsaXR5IHpvbmVzICgxLTMpICovXG4gIG1heEF6czogbnVtYmVyO1xuICBcbiAgLyoqIE51bWJlciBvZiBOQVQgZ2F0ZXdheXMgKDAgPSBubyBwcml2YXRlIHN1Ym5ldCBpbnRlcm5ldCBhY2Nlc3MpICovXG4gIG5hdEdhdGV3YXlzOiBudW1iZXI7XG4gIFxuICAvKiogQ0lEUiBibG9jayBmb3IgdGhlIFZQQyAqL1xuICBjaWRyOiBzdHJpbmc7XG4gIFxuICAvKiogRW5hYmxlIFZQQyBmbG93IGxvZ3MgZm9yIG5ldHdvcmsgbW9uaXRvcmluZyAqL1xuICBmbG93TG9nc0VuYWJsZWQ6IGJvb2xlYW47XG4gIFxuICAvKiogRmxvdyBsb2dzIHJldGVudGlvbiBpbiBkYXlzICovXG4gIGZsb3dMb2dzUmV0ZW50aW9uRGF5czogbnVtYmVyO1xuICBcbiAgLyoqIFB1YmxpYyBzdWJuZXQgY29uZmlndXJhdGlvbiAqL1xuICBwdWJsaWNTdWJuZXRzOiBTdWJuZXRDb25maWc7XG4gIFxuICAvKiogUHJpdmF0ZSBzdWJuZXQgY29uZmlndXJhdGlvbiAoZm9yIExhbWJkYSwgUkRTKSAqL1xuICBwcml2YXRlU3VibmV0czogU3VibmV0Q29uZmlnO1xuICBcbiAgLyoqIElzb2xhdGVkIHN1Ym5ldCBjb25maWd1cmF0aW9uIChubyBpbnRlcm5ldCwgZm9yIFJEUykgKi9cbiAgaXNvbGF0ZWRTdWJuZXRzOiBTdWJuZXRDb25maWcgfCB1bmRlZmluZWQ7XG4gIFxuICAvKiogRW5hYmxlIFZQQyBlbmRwb2ludHMgZm9yIEFXUyBzZXJ2aWNlcyAoUzMsIER5bmFtb0RCLCBldGMuKSAqL1xuICBlbmFibGVWcGNFbmRwb2ludHM6IGJvb2xlYW47XG4gIFxuICAvKiogQWRkaXRpb25hbCBpbmdyZXNzIHJ1bGVzIGZvciBMYW1iZGEgc2VjdXJpdHkgZ3JvdXAgKi9cbiAgbGFtYmRhU2VjdXJpdHlHcm91cEluZ3Jlc3M6IFNlY3VyaXR5R3JvdXBSdWxlW107XG4gIFxuICAvKiogQWRkaXRpb25hbCBlZ3Jlc3MgcnVsZXMgZm9yIExhbWJkYSBzZWN1cml0eSBncm91cCAqL1xuICBsYW1iZGFTZWN1cml0eUdyb3VwRWdyZXNzOiBTZWN1cml0eUdyb3VwUnVsZVtdO1xufVxuXG4vKipcbiAqIFZQQyBjb25maWd1cmF0aW9uIGlucHV0IChmb3IgdXNlciBvdmVycmlkZXMpXG4gKi9cbmV4cG9ydCB0eXBlIFZwY0VudkNvbmZpZ0lucHV0ID0gUGFydGlhbDxPbWl0PFZwY0VudkNvbmZpZywgJ3B1YmxpY1N1Ym5ldHMnIHwgJ3ByaXZhdGVTdWJuZXRzJyB8ICdpc29sYXRlZFN1Ym5ldHMnPj4gJiB7XG4gIHB1YmxpY1N1Ym5ldHM/OiBQYXJ0aWFsPFN1Ym5ldENvbmZpZz47XG4gIHByaXZhdGVTdWJuZXRzPzogUGFydGlhbDxTdWJuZXRDb25maWc+O1xuICBpc29sYXRlZFN1Ym5ldHM/OiBQYXJ0aWFsPFN1Ym5ldENvbmZpZz47XG59O1xuIl19
@@ -0,0 +1,24 @@
1
+ /**
2
+ * VentureKit WebSocket Configuration Types
3
+ *
4
+ * WebSocket API Gateway configuration for real-time features.
5
+ */
6
+ /**
7
+ * WebSocket configuration
8
+ */
9
+ export interface WebSocketEnvConfig {
10
+ /** Enable WebSocket API (default: false) */
11
+ enabled: boolean;
12
+ /** Route selection expression (default: '$request.body.action') */
13
+ routeSelectionExpression: string;
14
+ /** Idle connection timeout in seconds (default: 600, max: 7200) */
15
+ idleTimeoutSec: number;
16
+ /** Throttle rate limit (messages per second) */
17
+ throttleRateLimit: number;
18
+ /** Throttle burst limit */
19
+ throttleBurstLimit: number;
20
+ }
21
+ /**
22
+ * WebSocket configuration input (for user overrides)
23
+ */
24
+ export type WebSocketEnvConfigInput = Partial<WebSocketEnvConfig>;
@@ -0,0 +1,8 @@
1
+ "use strict";
2
+ /**
3
+ * VentureKit WebSocket Configuration Types
4
+ *
5
+ * WebSocket API Gateway configuration for real-time features.
6
+ */
7
+ Object.defineProperty(exports, "__esModule", { value: true });
8
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid2Vic29ja2V0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3R5cGVzL3dlYnNvY2tldC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7Ozs7R0FJRyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogVmVudHVyZUtpdCBXZWJTb2NrZXQgQ29uZmlndXJhdGlvbiBUeXBlc1xuICogXG4gKiBXZWJTb2NrZXQgQVBJIEdhdGV3YXkgY29uZmlndXJhdGlvbiBmb3IgcmVhbC10aW1lIGZlYXR1cmVzLlxuICovXG5cbi8qKlxuICogV2ViU29ja2V0IGNvbmZpZ3VyYXRpb25cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBXZWJTb2NrZXRFbnZDb25maWcge1xuICAvKiogRW5hYmxlIFdlYlNvY2tldCBBUEkgKGRlZmF1bHQ6IGZhbHNlKSAqL1xuICBlbmFibGVkOiBib29sZWFuO1xuICBcbiAgLyoqIFJvdXRlIHNlbGVjdGlvbiBleHByZXNzaW9uIChkZWZhdWx0OiAnJHJlcXVlc3QuYm9keS5hY3Rpb24nKSAqL1xuICByb3V0ZVNlbGVjdGlvbkV4cHJlc3Npb246IHN0cmluZztcbiAgXG4gIC8qKiBJZGxlIGNvbm5lY3Rpb24gdGltZW91dCBpbiBzZWNvbmRzIChkZWZhdWx0OiA2MDAsIG1heDogNzIwMCkgKi9cbiAgaWRsZVRpbWVvdXRTZWM6IG51bWJlcjtcbiAgXG4gIC8qKiBUaHJvdHRsZSByYXRlIGxpbWl0IChtZXNzYWdlcyBwZXIgc2Vjb25kKSAqL1xuICB0aHJvdHRsZVJhdGVMaW1pdDogbnVtYmVyO1xuICBcbiAgLyoqIFRocm90dGxlIGJ1cnN0IGxpbWl0ICovXG4gIHRocm90dGxlQnVyc3RMaW1pdDogbnVtYmVyO1xufVxuXG4vKipcbiAqIFdlYlNvY2tldCBjb25maWd1cmF0aW9uIGlucHV0IChmb3IgdXNlciBvdmVycmlkZXMpXG4gKi9cbmV4cG9ydCB0eXBlIFdlYlNvY2tldEVudkNvbmZpZ0lucHV0ID0gUGFydGlhbDxXZWJTb2NrZXRFbnZDb25maWc+O1xuIl19
@@ -0,0 +1,41 @@
1
+ /**
2
+ * VentureKit Configuration Validation
3
+ *
4
+ * Validates configuration before it reaches infrastructure.
5
+ * Catches errors early with clear, actionable messages.
6
+ */
7
+ import type { BaseConfig } from './types/base';
8
+ import type { SecurityConfig } from './types/security';
9
+ import type { EnvConfigInput, Environment } from './types/env';
10
+ /**
11
+ * Validation error
12
+ */
13
+ export declare class ValidationError extends Error {
14
+ readonly field: string;
15
+ readonly message: string;
16
+ readonly value?: unknown | undefined;
17
+ constructor(field: string, message: string, value?: unknown | undefined);
18
+ }
19
+ /**
20
+ * Validation result
21
+ */
22
+ export interface ValidationResult {
23
+ valid: boolean;
24
+ errors: ValidationError[];
25
+ }
26
+ /**
27
+ * Validate base configuration
28
+ */
29
+ export declare function validateBaseConfig(config: BaseConfig): ValidationResult;
30
+ /**
31
+ * Validate security configuration
32
+ */
33
+ export declare function validateSecurityConfig(config: SecurityConfig): ValidationResult;
34
+ /**
35
+ * Validate environment configuration input
36
+ */
37
+ export declare function validateEnvConfigInput(env: Environment, config: EnvConfigInput): ValidationResult;
38
+ /**
39
+ * Assert validation passes, throw if not
40
+ */
41
+ export declare function assertValid(result: ValidationResult): void;
@@ -0,0 +1,162 @@
1
+ "use strict";
2
+ /**
3
+ * VentureKit Configuration Validation
4
+ *
5
+ * Validates configuration before it reaches infrastructure.
6
+ * Catches errors early with clear, actionable messages.
7
+ */
8
+ Object.defineProperty(exports, "__esModule", { value: true });
9
+ exports.ValidationError = void 0;
10
+ exports.validateBaseConfig = validateBaseConfig;
11
+ exports.validateSecurityConfig = validateSecurityConfig;
12
+ exports.validateEnvConfigInput = validateEnvConfigInput;
13
+ exports.assertValid = assertValid;
14
+ /**
15
+ * Validation error
16
+ */
17
+ class ValidationError extends Error {
18
+ field;
19
+ message;
20
+ value;
21
+ constructor(field, message, value) {
22
+ super(`${field}: ${message}`);
23
+ this.field = field;
24
+ this.message = message;
25
+ this.value = value;
26
+ this.name = 'ValidationError';
27
+ }
28
+ }
29
+ exports.ValidationError = ValidationError;
30
+ /**
31
+ * Create a successful validation result
32
+ */
33
+ function ok() {
34
+ return { valid: true, errors: [] };
35
+ }
36
+ /**
37
+ * Create a failed validation result
38
+ */
39
+ function fail(errors) {
40
+ return { valid: false, errors };
41
+ }
42
+ /**
43
+ * Merge validation results
44
+ */
45
+ function merge(...results) {
46
+ const errors = results.flatMap(r => r.errors);
47
+ return { valid: errors.length === 0, errors };
48
+ }
49
+ /**
50
+ * Validate base configuration
51
+ */
52
+ function validateBaseConfig(config) {
53
+ const errors = [];
54
+ // Name validation
55
+ if (!config.name) {
56
+ errors.push(new ValidationError('name', 'Project name is required'));
57
+ }
58
+ else if (!/^[a-z][a-z0-9-]*$/.test(config.name)) {
59
+ errors.push(new ValidationError('name', 'Project name must start with lowercase letter and contain only lowercase letters, numbers, and hyphens', config.name));
60
+ }
61
+ else if (config.name.length > 32) {
62
+ errors.push(new ValidationError('name', 'Project name must be 32 characters or less', config.name));
63
+ }
64
+ // Display name
65
+ if (!config.displayName) {
66
+ errors.push(new ValidationError('displayName', 'Display name is required'));
67
+ }
68
+ // Region validation
69
+ if (!config.region) {
70
+ errors.push(new ValidationError('region', 'AWS region is required'));
71
+ }
72
+ else if (!/^[a-z]{2}-[a-z]+-\d$/.test(config.region)) {
73
+ errors.push(new ValidationError('region', 'Invalid AWS region format (e.g., eu-west-1)', config.region));
74
+ }
75
+ return errors.length > 0 ? fail(errors) : ok();
76
+ }
77
+ /**
78
+ * Validate security configuration
79
+ */
80
+ function validateSecurityConfig(config) {
81
+ const errors = [];
82
+ // Scopes validation
83
+ if (!config.scopes || config.scopes.length === 0) {
84
+ errors.push(new ValidationError('scopes', 'At least one OAuth scope is required'));
85
+ }
86
+ else {
87
+ config.scopes.forEach((scope, i) => {
88
+ if (!scope.name) {
89
+ errors.push(new ValidationError(`scopes[${i}].name`, 'Scope name is required'));
90
+ }
91
+ else if (!/^[a-z][a-z0-9.]*$/.test(scope.name)) {
92
+ errors.push(new ValidationError(`scopes[${i}].name`, 'Scope name must be lowercase with dots (e.g., projects.read)', scope.name));
93
+ }
94
+ });
95
+ }
96
+ // App clients validation
97
+ if (!config.appClients || config.appClients.length === 0) {
98
+ errors.push(new ValidationError('appClients', 'At least one app client is required'));
99
+ }
100
+ else {
101
+ const scopeNames = new Set(config.scopes?.map(s => s.name) || []);
102
+ config.appClients.forEach((client, i) => {
103
+ if (!client.name) {
104
+ errors.push(new ValidationError(`appClients[${i}].name`, 'Client name is required'));
105
+ }
106
+ client.allowedScopes.forEach((scope, j) => {
107
+ if (!scopeNames.has(scope)) {
108
+ errors.push(new ValidationError(`appClients[${i}].allowedScopes[${j}]`, `Scope "${scope}" is not defined in scopes`, scope));
109
+ }
110
+ });
111
+ });
112
+ }
113
+ return errors.length > 0 ? fail(errors) : ok();
114
+ }
115
+ /**
116
+ * Validate environment configuration input
117
+ */
118
+ function validateEnvConfigInput(env, config) {
119
+ const errors = [];
120
+ // Lambda validation
121
+ if (config.lambda) {
122
+ if (config.lambda.memoryMb !== undefined) {
123
+ if (config.lambda.memoryMb < 128 || config.lambda.memoryMb > 10240) {
124
+ errors.push(new ValidationError(`${env}.lambda.memoryMb`, 'Lambda memory must be between 128 and 10240 MB', config.lambda.memoryMb));
125
+ }
126
+ }
127
+ if (config.lambda.timeoutSec !== undefined) {
128
+ if (config.lambda.timeoutSec < 1 || config.lambda.timeoutSec > 900) {
129
+ errors.push(new ValidationError(`${env}.lambda.timeoutSec`, 'Lambda timeout must be between 1 and 900 seconds', config.lambda.timeoutSec));
130
+ }
131
+ }
132
+ }
133
+ // API validation
134
+ if (config.api) {
135
+ if (config.api.throttleRateLimit !== undefined && config.api.throttleRateLimit < 1) {
136
+ errors.push(new ValidationError(`${env}.api.throttleRateLimit`, 'API rate limit must be at least 1', config.api.throttleRateLimit));
137
+ }
138
+ if (config.api.throttleBurstLimit !== undefined && config.api.throttleBurstLimit < 1) {
139
+ errors.push(new ValidationError(`${env}.api.throttleBurstLimit`, 'API burst limit must be at least 1', config.api.throttleBurstLimit));
140
+ }
141
+ }
142
+ // VPC validation
143
+ if (config.vpc) {
144
+ if (config.vpc.maxAzs !== undefined && (config.vpc.maxAzs < 1 || config.vpc.maxAzs > 3)) {
145
+ errors.push(new ValidationError(`${env}.vpc.maxAzs`, 'VPC must have 1-3 availability zones', config.vpc.maxAzs));
146
+ }
147
+ if (config.vpc.natGateways !== undefined && config.vpc.natGateways < 0) {
148
+ errors.push(new ValidationError(`${env}.vpc.natGateways`, 'NAT gateways cannot be negative', config.vpc.natGateways));
149
+ }
150
+ }
151
+ return errors.length > 0 ? fail(errors) : ok();
152
+ }
153
+ /**
154
+ * Assert validation passes, throw if not
155
+ */
156
+ function assertValid(result) {
157
+ if (!result.valid) {
158
+ const messages = result.errors.map(e => ` - ${e.field}: ${e.message}`).join('\n');
159
+ throw new Error(`Configuration validation failed:\n${messages}`);
160
+ }
161
+ }
162
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"validate.js","sourceRoot":"","sources":["../src/validate.ts"],"names":[],"mappings":";AAAA;;;;;GAKG;;;AAqDH,gDAqCC;AAKD,wDA4CC;AAKD,wDAiEC;AAKD,kCAKC;AArND;;GAEG;AACH,MAAa,eAAgB,SAAQ,KAAK;IAEtB;IACA;IACA;IAHlB,YACkB,KAAa,EACb,OAAe,EACf,KAAe;QAE/B,KAAK,CAAC,GAAG,KAAK,KAAK,OAAO,EAAE,CAAC,CAAC;QAJd,UAAK,GAAL,KAAK,CAAQ;QACb,YAAO,GAAP,OAAO,CAAQ;QACf,UAAK,GAAL,KAAK,CAAU;QAG/B,IAAI,CAAC,IAAI,GAAG,iBAAiB,CAAC;IAChC,CAAC;CACF;AATD,0CASC;AAUD;;GAEG;AACH,SAAS,EAAE;IACT,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,MAAM,EAAE,EAAE,EAAE,CAAC;AACrC,CAAC;AAED;;GAEG;AACH,SAAS,IAAI,CAAC,MAAyB;IACrC,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC;AAClC,CAAC;AAED;;GAEG;AACH,SAAS,KAAK,CAAC,GAAG,OAA2B;IAC3C,MAAM,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC;IAC9C,OAAO,EAAE,KAAK,EAAE,MAAM,CAAC,MAAM,KAAK,CAAC,EAAE,MAAM,EAAE,CAAC;AAChD,CAAC;AAED;;GAEG;AACH,SAAgB,kBAAkB,CAAC,MAAkB;IACnD,MAAM,MAAM,GAAsB,EAAE,CAAC;IAErC,kBAAkB;IAClB,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC;QACjB,MAAM,CAAC,IAAI,CAAC,IAAI,eAAe,CAAC,MAAM,EAAE,0BAA0B,CAAC,CAAC,CAAC;IACvE,CAAC;SAAM,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC;QAClD,MAAM,CAAC,IAAI,CAAC,IAAI,eAAe,CAC7B,MAAM,EACN,wGAAwG,EACxG,MAAM,CAAC,IAAI,CACZ,CAAC,CAAC;IACL,CAAC;SAAM,IAAI,MAAM,CAAC,IAAI,CAAC,MAAM,GAAG,EAAE,EAAE,CAAC;QACnC,MAAM,CAAC,IAAI,CAAC,IAAI,eAAe,CAC7B,MAAM,EACN,4CAA4C,EAC5C,MAAM,CAAC,IAAI,CACZ,CAAC,CAAC;IACL,CAAC;IAED,eAAe;IACf,IAAI,CAAC,MAAM,CAAC,WAAW,EAAE,CAAC;QACxB,MAAM,CAAC,IAAI,CAAC,IAAI,eAAe,CAAC,aAAa,EAAE,0BAA0B,CAAC,CAAC,CAAC;IAC9E,CAAC;IAED,oBAAoB;IACpB,IAAI,CAAC,MAAM,CAAC,MAAM,EAAE,CAAC;QACnB,MAAM,CAAC,IAAI,CAAC,IAAI,eAAe,CAAC,QAAQ,EAAE,wBAAwB,CAAC,CAAC,CAAC;IACvE,CAAC;SAAM,IAAI,CAAC,sBAAsB,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC;QACvD,MAAM,CAAC,IAAI,CAAC,IAAI,eAAe,CAC7B,QAAQ,EACR,6CAA6C,EAC7C,MAAM,CAAC,MAAM,CACd,CAAC,CAAC;IACL,CAAC;IAED,OAAO,MAAM,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC;AACjD,CAAC;AAED;;GAEG;AACH,SAAgB,sBAAsB,CAAC,MAAsB;IAC3D,MAAM,MAAM,GAAsB,EAAE,CAAC;IAErC,oBAAoB;IACpB,IAAI,CAAC,MAAM,CAAC,MAAM,IAAI,MAAM,CAAC,MAAM,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACjD,MAAM,CAAC,IAAI,CAAC,IAAI,eAAe,CAAC,QAAQ,EAAE,sCAAsC,CAAC,CAAC,CAAC;IACrF,CAAC;SAAM,CAAC;QACN,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,KAAK,EAAE,CAAC,EAAE,EAAE;YACjC,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC;gBAChB,MAAM,CAAC,IAAI,CAAC,IAAI,eAAe,CAAC,UAAU,CAAC,QAAQ,EAAE,wBAAwB,CAAC,CAAC,CAAC;YAClF,CAAC;iBAAM,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC;gBACjD,MAAM,CAAC,IAAI,CAAC,IAAI,eAAe,CAC7B,UAAU,CAAC,QAAQ,EACnB,8DAA8D,EAC9D,KAAK,CAAC,IAAI,CACX,CAAC,CAAC;YACL,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED,yBAAyB;IACzB,IAAI,CAAC,MAAM,CAAC,UAAU,IAAI,MAAM,CAAC,UAAU,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACzD,MAAM,CAAC,IAAI,CAAC,IAAI,eAAe,CAAC,YAAY,EAAE,qCAAqC,CAAC,CAAC,CAAC;IACxF,CAAC;SAAM,CAAC;QACN,MAAM,UAAU,GAAG,IAAI,GAAG,CAAC,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;QAElE,MAAM,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE;YACtC,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC;gBACjB,MAAM,CAAC,IAAI,CAAC,IAAI,eAAe,CAAC,cAAc,CAAC,QAAQ,EAAE,yBAAyB,CAAC,CAAC,CAAC;YACvF,CAAC;YAED,MAAM,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC,KAAK,EAAE,CAAC,EAAE,EAAE;gBACxC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC;oBAC3B,MAAM,CAAC,IAAI,CAAC,IAAI,eAAe,CAC7B,cAAc,CAAC,mBAAmB,CAAC,GAAG,EACtC,UAAU,KAAK,4BAA4B,EAC3C,KAAK,CACN,CAAC,CAAC;gBACL,CAAC;YACH,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;IAED,OAAO,MAAM,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC;AACjD,CAAC;AAED;;GAEG;AACH,SAAgB,sBAAsB,CAAC,GAAgB,EAAE,MAAsB;IAC7E,MAAM,MAAM,GAAsB,EAAE,CAAC;IAErC,oBAAoB;IACpB,IAAI,MAAM,CAAC,MAAM,EAAE,CAAC;QAClB,IAAI,MAAM,CAAC,MAAM,CAAC,QAAQ,KAAK,SAAS,EAAE,CAAC;YACzC,IAAI,MAAM,CAAC,MAAM,CAAC,QAAQ,GAAG,GAAG,IAAI,MAAM,CAAC,MAAM,CAAC,QAAQ,GAAG,KAAK,EAAE,CAAC;gBACnE,MAAM,CAAC,IAAI,CAAC,IAAI,eAAe,CAC7B,GAAG,GAAG,kBAAkB,EACxB,gDAAgD,EAChD,MAAM,CAAC,MAAM,CAAC,QAAQ,CACvB,CAAC,CAAC;YACL,CAAC;QACH,CAAC;QAED,IAAI,MAAM,CAAC,MAAM,CAAC,UAAU,KAAK,SAAS,EAAE,CAAC;YAC3C,IAAI,MAAM,CAAC,MAAM,CAAC,UAAU,GAAG,CAAC,IAAI,MAAM,CAAC,MAAM,CAAC,UAAU,GAAG,GAAG,EAAE,CAAC;gBACnE,MAAM,CAAC,IAAI,CAAC,IAAI,eAAe,CAC7B,GAAG,GAAG,oBAAoB,EAC1B,kDAAkD,EAClD,MAAM,CAAC,MAAM,CAAC,UAAU,CACzB,CAAC,CAAC;YACL,CAAC;QACH,CAAC;IACH,CAAC;IAED,iBAAiB;IACjB,IAAI,MAAM,CAAC,GAAG,EAAE,CAAC;QACf,IAAI,MAAM,CAAC,GAAG,CAAC,iBAAiB,KAAK,SAAS,IAAI,MAAM,CAAC,GAAG,CAAC,iBAAiB,GAAG,CAAC,EAAE,CAAC;YACnF,MAAM,CAAC,IAAI,CAAC,IAAI,eAAe,CAC7B,GAAG,GAAG,wBAAwB,EAC9B,mCAAmC,EACnC,MAAM,CAAC,GAAG,CAAC,iBAAiB,CAC7B,CAAC,CAAC;QACL,CAAC;QAED,IAAI,MAAM,CAAC,GAAG,CAAC,kBAAkB,KAAK,SAAS,IAAI,MAAM,CAAC,GAAG,CAAC,kBAAkB,GAAG,CAAC,EAAE,CAAC;YACrF,MAAM,CAAC,IAAI,CAAC,IAAI,eAAe,CAC7B,GAAG,GAAG,yBAAyB,EAC/B,oCAAoC,EACpC,MAAM,CAAC,GAAG,CAAC,kBAAkB,CAC9B,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED,iBAAiB;IACjB,IAAI,MAAM,CAAC,GAAG,EAAE,CAAC;QACf,IAAI,MAAM,CAAC,GAAG,CAAC,MAAM,KAAK,SAAS,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,MAAM,GAAG,CAAC,IAAI,MAAM,CAAC,GAAG,CAAC,MAAM,GAAG,CAAC,CAAC,EAAE,CAAC;YACxF,MAAM,CAAC,IAAI,CAAC,IAAI,eAAe,CAC7B,GAAG,GAAG,aAAa,EACnB,sCAAsC,EACtC,MAAM,CAAC,GAAG,CAAC,MAAM,CAClB,CAAC,CAAC;QACL,CAAC;QAED,IAAI,MAAM,CAAC,GAAG,CAAC,WAAW,KAAK,SAAS,IAAI,MAAM,CAAC,GAAG,CAAC,WAAW,GAAG,CAAC,EAAE,CAAC;YACvE,MAAM,CAAC,IAAI,CAAC,IAAI,eAAe,CAC7B,GAAG,GAAG,kBAAkB,EACxB,iCAAiC,EACjC,MAAM,CAAC,GAAG,CAAC,WAAW,CACvB,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED,OAAO,MAAM,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC;AACjD,CAAC;AAED;;GAEG;AACH,SAAgB,WAAW,CAAC,MAAwB;IAClD,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC;QAClB,MAAM,QAAQ,GAAG,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,OAAO,CAAC,CAAC,KAAK,KAAK,CAAC,CAAC,OAAO,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACnF,MAAM,IAAI,KAAK,CAAC,qCAAqC,QAAQ,EAAE,CAAC,CAAC;IACnE,CAAC;AACH,CAAC","sourcesContent":["/**\n * VentureKit Configuration Validation\n * \n * Validates configuration before it reaches infrastructure.\n * Catches errors early with clear, actionable messages.\n */\n\nimport type { BaseConfig } from './types/base';\nimport type { SecurityConfig, OAuthScope, AppClient } from './types/security';\nimport type { EnvConfigInput, Environment } from './types/env';\n\n/**\n * Validation error\n */\nexport class ValidationError extends Error {\n  constructor(\n    public readonly field: string,\n    public readonly message: string,\n    public readonly value?: unknown\n  ) {\n    super(`${field}: ${message}`);\n    this.name = 'ValidationError';\n  }\n}\n\n/**\n * Validation result\n */\nexport interface ValidationResult {\n  valid: boolean;\n  errors: ValidationError[];\n}\n\n/**\n * Create a successful validation result\n */\nfunction ok(): ValidationResult {\n  return { valid: true, errors: [] };\n}\n\n/**\n * Create a failed validation result\n */\nfunction fail(errors: ValidationError[]): ValidationResult {\n  return { valid: false, errors };\n}\n\n/**\n * Merge validation results\n */\nfunction merge(...results: ValidationResult[]): ValidationResult {\n  const errors = results.flatMap(r => r.errors);\n  return { valid: errors.length === 0, errors };\n}\n\n/**\n * Validate base configuration\n */\nexport function validateBaseConfig(config: BaseConfig): ValidationResult {\n  const errors: ValidationError[] = [];\n\n  // Name validation\n  if (!config.name) {\n    errors.push(new ValidationError('name', 'Project name is required'));\n  } else if (!/^[a-z][a-z0-9-]*$/.test(config.name)) {\n    errors.push(new ValidationError(\n      'name',\n      'Project name must start with lowercase letter and contain only lowercase letters, numbers, and hyphens',\n      config.name\n    ));\n  } else if (config.name.length > 32) {\n    errors.push(new ValidationError(\n      'name',\n      'Project name must be 32 characters or less',\n      config.name\n    ));\n  }\n\n  // Display name\n  if (!config.displayName) {\n    errors.push(new ValidationError('displayName', 'Display name is required'));\n  }\n\n  // Region validation\n  if (!config.region) {\n    errors.push(new ValidationError('region', 'AWS region is required'));\n  } else if (!/^[a-z]{2}-[a-z]+-\\d$/.test(config.region)) {\n    errors.push(new ValidationError(\n      'region',\n      'Invalid AWS region format (e.g., eu-west-1)',\n      config.region\n    ));\n  }\n\n  return errors.length > 0 ? fail(errors) : ok();\n}\n\n/**\n * Validate security configuration\n */\nexport function validateSecurityConfig(config: SecurityConfig): ValidationResult {\n  const errors: ValidationError[] = [];\n\n  // Scopes validation\n  if (!config.scopes || config.scopes.length === 0) {\n    errors.push(new ValidationError('scopes', 'At least one OAuth scope is required'));\n  } else {\n    config.scopes.forEach((scope, i) => {\n      if (!scope.name) {\n        errors.push(new ValidationError(`scopes[${i}].name`, 'Scope name is required'));\n      } else if (!/^[a-z][a-z0-9.]*$/.test(scope.name)) {\n        errors.push(new ValidationError(\n          `scopes[${i}].name`,\n          'Scope name must be lowercase with dots (e.g., projects.read)',\n          scope.name\n        ));\n      }\n    });\n  }\n\n  // App clients validation\n  if (!config.appClients || config.appClients.length === 0) {\n    errors.push(new ValidationError('appClients', 'At least one app client is required'));\n  } else {\n    const scopeNames = new Set(config.scopes?.map(s => s.name) || []);\n    \n    config.appClients.forEach((client, i) => {\n      if (!client.name) {\n        errors.push(new ValidationError(`appClients[${i}].name`, 'Client name is required'));\n      }\n      \n      client.allowedScopes.forEach((scope, j) => {\n        if (!scopeNames.has(scope)) {\n          errors.push(new ValidationError(\n            `appClients[${i}].allowedScopes[${j}]`,\n            `Scope \"${scope}\" is not defined in scopes`,\n            scope\n          ));\n        }\n      });\n    });\n  }\n\n  return errors.length > 0 ? fail(errors) : ok();\n}\n\n/**\n * Validate environment configuration input\n */\nexport function validateEnvConfigInput(env: Environment, config: EnvConfigInput): ValidationResult {\n  const errors: ValidationError[] = [];\n\n  // Lambda validation\n  if (config.lambda) {\n    if (config.lambda.memoryMb !== undefined) {\n      if (config.lambda.memoryMb < 128 || config.lambda.memoryMb > 10240) {\n        errors.push(new ValidationError(\n          `${env}.lambda.memoryMb`,\n          'Lambda memory must be between 128 and 10240 MB',\n          config.lambda.memoryMb\n        ));\n      }\n    }\n    \n    if (config.lambda.timeoutSec !== undefined) {\n      if (config.lambda.timeoutSec < 1 || config.lambda.timeoutSec > 900) {\n        errors.push(new ValidationError(\n          `${env}.lambda.timeoutSec`,\n          'Lambda timeout must be between 1 and 900 seconds',\n          config.lambda.timeoutSec\n        ));\n      }\n    }\n  }\n\n  // API validation\n  if (config.api) {\n    if (config.api.throttleRateLimit !== undefined && config.api.throttleRateLimit < 1) {\n      errors.push(new ValidationError(\n        `${env}.api.throttleRateLimit`,\n        'API rate limit must be at least 1',\n        config.api.throttleRateLimit\n      ));\n    }\n    \n    if (config.api.throttleBurstLimit !== undefined && config.api.throttleBurstLimit < 1) {\n      errors.push(new ValidationError(\n        `${env}.api.throttleBurstLimit`,\n        'API burst limit must be at least 1',\n        config.api.throttleBurstLimit\n      ));\n    }\n  }\n\n  // VPC validation\n  if (config.vpc) {\n    if (config.vpc.maxAzs !== undefined && (config.vpc.maxAzs < 1 || config.vpc.maxAzs > 3)) {\n      errors.push(new ValidationError(\n        `${env}.vpc.maxAzs`,\n        'VPC must have 1-3 availability zones',\n        config.vpc.maxAzs\n      ));\n    }\n    \n    if (config.vpc.natGateways !== undefined && config.vpc.natGateways < 0) {\n      errors.push(new ValidationError(\n        `${env}.vpc.natGateways`,\n        'NAT gateways cannot be negative',\n        config.vpc.natGateways\n      ));\n    }\n  }\n\n  return errors.length > 0 ? fail(errors) : ok();\n}\n\n/**\n * Assert validation passes, throw if not\n */\nexport function assertValid(result: ValidationResult): void {\n  if (!result.valid) {\n    const messages = result.errors.map(e => `  - ${e.field}: ${e.message}`).join('\\n');\n    throw new Error(`Configuration validation failed:\\n${messages}`);\n  }\n}\n"]}
package/package.json ADDED
@@ -0,0 +1,33 @@
1
+ {
2
+ "name": "@venturekit/core",
3
+ "version": "0.0.0-dev.20260307234057",
4
+ "description": "VentureKit core types, presets, and configuration resolution.",
5
+ "main": "dist/index.js",
6
+ "types": "dist/index.d.ts",
7
+ "files": [
8
+ "dist"
9
+ ],
10
+ "repository": {
11
+ "type": "git",
12
+ "url": "https://github.com/venturekit-dev/venturekit.private.git",
13
+ "directory": "packages/core"
14
+ },
15
+ "publishConfig": {
16
+ "registry": "https://registry.npmjs.org",
17
+ "access": "public"
18
+ },
19
+ "keywords": [
20
+ "venturekit",
21
+ "saas",
22
+ "config"
23
+ ],
24
+ "license": "Apache-2.0",
25
+ "devDependencies": {
26
+ "typescript": "^5.3.0"
27
+ },
28
+ "scripts": {
29
+ "build": "tsc",
30
+ "dev": "tsc --watch",
31
+ "clean": "rm -rf dist"
32
+ }
33
+ }