UNPKG

automation-deploy-template-iac

Version:

AWS CDK Infrastructure as Code templates for deploying full-stack applications with FastAPI backend, Vue.js frontend, and automated deployment

166 lines 26 kB
#!/usr/bin/env node "use strict"; var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { if (k2 === undefined) k2 = k; var desc = Object.getOwnPropertyDescriptor(m, k); if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { desc = { enumerable: true, get: function() { return m[k]; } }; } Object.defineProperty(o, k2, desc); }) : (function(o, m, k, k2) { if (k2 === undefined) k2 = k; o[k2] = m[k]; })); var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) { Object.defineProperty(o, "default", { enumerable: true, value: v }); }) : function(o, v) { o["default"] = v; }); var __importStar = (this && this.__importStar) || function (mod) { if (mod && mod.__esModule) return mod; var result = {}; if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k); __setModuleDefault(result, mod); return result; }; Object.defineProperty(exports, "__esModule", { value: true }); const cdk = __importStar(require("aws-cdk-lib")); const backend_stack_1 = require("../lib/backend-stack"); const env_1 = require("../lib/config/env"); const domain_utils_1 = require("../lib/utils/domain-utils"); const constants_1 = require("../lib/config/constants"); const cloudformation_sdk_utils_1 = require("../lib/utils/sdks/cloudformation-sdk-utils"); const rds_requests_1 = require("../lib/requests/rds-requests"); const container_config_requests_1 = require("../lib/requests/container-config-requests"); const custom_secrets_requests_1 = require("../lib/requests/custom-secrets-requests"); (async () => { const app = new cdk.App(); const projectName = (0, env_1.getProjectName)(); // Context から設定を取得 const environment = (0, env_1.getEnvironment)(app, 'dev'); const rootDomain = app.node.tryGetContext('rootDomain') || process.env.ROOT_DOMAIN; const domainName = rootDomain ? (0, domain_utils_1.getBackendDomain)(rootDomain, environment) : undefined; const containerPort = (0, env_1.getContainerPort)(app); const imageTag = (0, env_1.getImageTag)(); const cpu = (0, env_1.getCpu)(); const memory = (0, env_1.getMemory)(); const desiredCount = (0, env_1.getDesiredCount)(); //const databaseStackName = app.node.tryGetContext('databaseStackName') || `${projectName}-${environment}-DatabaseStack`; const stackName = (0, constants_1.createStackName)(projectName, environment, constants_1.STACK_TYPES.BACKEND); const databaseStackName = (0, constants_1.createStackName)(projectName, environment, constants_1.STACK_TYPES.DATABASE); // check the database stack is already deployed via sdk const databaseStack = await cloudformation_sdk_utils_1.CloudformationSdkUtils.create(databaseStackName); const isDatabaseStackDeployed = databaseStack.isDeployed; const rdsSecretArn = databaseStack.getOutputByKey('SecretArn'); const rdsRequests = rds_requests_1.RdsRequests.build(databaseStack.getOutputByKey('ClusterEndpoint'), databaseStack.getOutputByKey('ClusterPort'), databaseStack.getOutputByKey('DatabaseName'), databaseStack.getOutputByKey('SecretArn'), databaseStack.getOutputByKey('ClusterArn')); console.log('========================='); console.log(`Project Name: ${projectName}`); console.log(`Environment: ${environment}`); console.log(`Stack Name: ${stackName}`); console.log(`Domain Name: ${domainName}`); console.log(`Image Tag: ${imageTag}`); console.log(`Container Port: ${containerPort}`); console.log(`CPU: ${cpu}`); console.log(`Memory: ${memory}`); console.log(`Desired Count: ${desiredCount}`); console.log(`Database Stack Name: ${databaseStackName}`); console.log(`RDS Secret ARN: ${rdsSecretArn}`); // ============================================================================ // CUSTOM CONTAINER CONFIGURATION // ============================================================================ // Configure custom environment variables and secrets for ECS containers // Following layered architecture: pass only metadata, adapter will load secrets // To use: deploy the secrets stack first with `cdk deploy <SecretsStackName>` // Then pass the secret name via context: --context customSecretName=<secret-name> // ============================================================================ // ============================================================================ // AUTOMATIC SECRET DISCOVERY // ============================================================================ // Automatically discovers and loads secrets based on naming convention // Pattern: ${projectName}-${environment}-{custom-secrets, api-keys, third-party-secrets, encryption-secrets} // Secrets that don't exist will be gracefully skipped // ============================================================================ console.log(`\n📝 Custom Configuration: Auto-discovery mode`); console.log(`Looking for secrets with pattern: ${projectName}-${environment}-*`); // Define expected secrets and their key mappings const secretConfigurations = [ { // Main custom secrets (from SecretsStack) secretName: `${projectName}-${environment}-custom-secrets`, keyMappings: [ { envVarName: 'CUSTOM_API_KEY', secretKey: 'api-key' }, { envVarName: 'CUSTOM_API_TOKEN', secretKey: 'api-token' }, { envVarName: 'STRIPE_API_KEY', secretKey: 'stripe-api-key' }, { envVarName: 'SENDGRID_API_KEY', secretKey: 'sendgrid-api-key' }, { envVarName: 'ENCRYPTION_KEY', secretKey: 'encryption-key' }, { envVarName: 'JWT_SECRET', secretKey: 'jwt-secret' }, ], }, { // Optional: API keys secret secretName: `${projectName}-${environment}-api-keys`, keyMappings: [ { envVarName: 'EXTERNAL_API_KEY', secretKey: 'external-api-key' }, { envVarName: 'PAYMENT_API_KEY', secretKey: 'payment-api-key' }, ], }, { // Optional: Third-party service credentials secretName: `${projectName}-${environment}-third-party-secrets`, keyMappings: [ { envVarName: 'TWILIO_AUTH_TOKEN', secretKey: 'twilio-auth-token' }, { envVarName: 'AWS_S3_ACCESS_KEY', secretKey: 's3-access-key' }, ], }, ]; // Build custom secrets requests with all potential secrets // Adapter will gracefully handle secrets that don't exist const customSecretsRequests = custom_secrets_requests_1.CustomSecretsRequests.buildFromMultiple(secretConfigurations); // Build container configuration with custom environment variables and secrets metadata const containerConfigRequests = container_config_requests_1.ContainerConfigRequests.build({ // Custom environment variables CUSTOM_ENV_VAR: 'production-value', FEATURE_FLAG_ENABLED: 'true', LOG_LEVEL: environment === 'production' ? 'info' : 'debug', CUSTOM_API_ENDPOINT: 'https://api.example.com', }, customSecretsRequests); console.log(`✅ Custom container configuration prepared`); console.log(` - Custom environment variables: 4`); console.log(` - Secret configurations to check: ${customSecretsRequests.secretConfigurations.length}`); console.log(` - Total potential secret mappings: ${customSecretsRequests.getTotalMappingsCount()}`); console.log(` - Adapter will load secrets that exist and skip those that don't`); try { // BackendStackを作成 new backend_stack_1.BackendStack(app, stackName, { projectName, environment: environment, domainName, imageTag, containerPort, cpu, memory, desiredCount, databaseStackName, isDatabaseStackDeployed, rdsRequests, containerConfigRequests, /* If you don't specify 'env', this stack will be environment-agnostic. * Account/Region-dependent features and context lookups will not work, * but a single synthesized template can be deployed anywhere. */ /* Uncomment the next line to specialize this stack for the AWS Account * and Region that are implied by the current CLI configuration. */ env: { account: (0, env_1.getCdkDefaultAccount)(), region: (0, env_1.getCdkDefaultRegion)() }, /* Uncomment the next line if you know exactly what Account and Region you * want to deploy the stack to. */ // env: { account: '123456789012', region: 'us-east-1' }, /* For more information, see https://docs.aws.amazon.com/cdk/latest/guide/environments.html */ tags: (0, constants_1.createDefaultTags)(projectName, environment, constants_1.STACK_TYPES.BACKEND, 'backend') }); console.log(`Successfully created ${stackName}`); } catch (error) { console.error('Failed to create BackendStack:', error instanceof Error ? error.message : String(error)); process.exit(1); } })(); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"backend.js","sourceRoot":"","sources":["../../bin/backend.ts"],"names":[],"mappings":"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","sourcesContent":["#!/usr/bin/env node\nimport * as cdk from 'aws-cdk-lib';\nimport { BackendStack } from '../lib/backend-stack';\nimport { getProjectName, getEnvironment, getContainerPort, getImageTag, getCpu, getMemory, getDesiredCount, getCdkDefaultAccount, getCdkDefaultRegion } from '../lib/config/env';\nimport { getBackendDomain } from '../lib/utils/domain-utils';\nimport { STACK_TYPES, createStackName, createDefaultTags } from '../lib/config/constants';\nimport { CloudformationSdkUtils } from '../lib/utils/sdks/cloudformation-sdk-utils';\nimport { RdsRequests } from '../lib/requests/rds-requests';\nimport { ContainerConfigRequests } from '../lib/requests/container-config-requests';\nimport { CustomSecretsRequests } from '../lib/requests/custom-secrets-requests';\n\n(async () => {\n  const app = new cdk.App();\n  const projectName = getProjectName();\n\n  // Context から設定を取得\n  const environment = getEnvironment(app, 'dev');\n  const rootDomain = app.node.tryGetContext('rootDomain') || process.env.ROOT_DOMAIN;\n  const domainName = rootDomain ? getBackendDomain(rootDomain, environment) : undefined;\n  const containerPort = getContainerPort(app);\n  const imageTag = getImageTag();\n  const cpu = getCpu();\n  const memory = getMemory();\n  const desiredCount = getDesiredCount();\n  //const databaseStackName = app.node.tryGetContext('databaseStackName') || `${projectName}-${environment}-DatabaseStack`;\n\n  const stackName = createStackName(projectName, environment, STACK_TYPES.BACKEND);\n\n  const databaseStackName = createStackName(projectName, environment, STACK_TYPES.DATABASE);\n\n  // check the database stack is already deployed via sdk\n  const databaseStack = await CloudformationSdkUtils.create(databaseStackName);\n  const isDatabaseStackDeployed = databaseStack.isDeployed;\n  const rdsSecretArn = databaseStack.getOutputByKey('SecretArn');\n  const rdsRequests = RdsRequests.build(databaseStack.getOutputByKey('ClusterEndpoint'), databaseStack.getOutputByKey('ClusterPort'), databaseStack.getOutputByKey('DatabaseName'), databaseStack.getOutputByKey('SecretArn'), databaseStack.getOutputByKey('ClusterArn'));\n\n  console.log('=========================');\n  console.log(`Project Name: ${projectName}`);\n  console.log(`Environment: ${environment}`);\n  console.log(`Stack Name: ${stackName}`);\n  console.log(`Domain Name: ${domainName}`);\n  console.log(`Image Tag: ${imageTag}`);\n  console.log(`Container Port: ${containerPort}`);\n  console.log(`CPU: ${cpu}`);\n  console.log(`Memory: ${memory}`);\n  console.log(`Desired Count: ${desiredCount}`);\n  console.log(`Database Stack Name: ${databaseStackName}`);\n  console.log(`RDS Secret ARN: ${rdsSecretArn}`);\n\n  // ============================================================================\n  // CUSTOM CONTAINER CONFIGURATION\n  // ============================================================================\n  // Configure custom environment variables and secrets for ECS containers\n  // Following layered architecture: pass only metadata, adapter will load secrets\n  // To use: deploy the secrets stack first with `cdk deploy <SecretsStackName>`\n  // Then pass the secret name via context: --context customSecretName=<secret-name>\n  // ============================================================================\n\n  // ============================================================================\n  // AUTOMATIC SECRET DISCOVERY\n  // ============================================================================\n  // Automatically discovers and loads secrets based on naming convention\n  // Pattern: ${projectName}-${environment}-{custom-secrets, api-keys, third-party-secrets, encryption-secrets}\n  // Secrets that don't exist will be gracefully skipped\n  // ============================================================================\n  \n  console.log(`\\n📝 Custom Configuration: Auto-discovery mode`);\n  console.log(`Looking for secrets with pattern: ${projectName}-${environment}-*`);\n  \n  // Define expected secrets and their key mappings\n  const secretConfigurations = [\n    {\n      // Main custom secrets (from SecretsStack)\n      secretName: `${projectName}-${environment}-custom-secrets`,\n      keyMappings: [\n        { envVarName: 'CUSTOM_API_KEY', secretKey: 'api-key' },\n        { envVarName: 'CUSTOM_API_TOKEN', secretKey: 'api-token' },\n        { envVarName: 'STRIPE_API_KEY', secretKey: 'stripe-api-key' },\n        { envVarName: 'SENDGRID_API_KEY', secretKey: 'sendgrid-api-key' },\n        { envVarName: 'ENCRYPTION_KEY', secretKey: 'encryption-key' },\n        { envVarName: 'JWT_SECRET', secretKey: 'jwt-secret' },\n      ],\n    },\n    {\n      // Optional: API keys secret\n      secretName: `${projectName}-${environment}-api-keys`,\n      keyMappings: [\n        { envVarName: 'EXTERNAL_API_KEY', secretKey: 'external-api-key' },\n        { envVarName: 'PAYMENT_API_KEY', secretKey: 'payment-api-key' },\n      ],\n    },\n    {\n      // Optional: Third-party service credentials\n      secretName: `${projectName}-${environment}-third-party-secrets`,\n      keyMappings: [\n        { envVarName: 'TWILIO_AUTH_TOKEN', secretKey: 'twilio-auth-token' },\n        { envVarName: 'AWS_S3_ACCESS_KEY', secretKey: 's3-access-key' },\n      ],\n    },\n  ];\n\n  // Build custom secrets requests with all potential secrets\n  // Adapter will gracefully handle secrets that don't exist\n  const customSecretsRequests = CustomSecretsRequests.buildFromMultiple(secretConfigurations);\n\n  // Build container configuration with custom environment variables and secrets metadata\n  const containerConfigRequests = ContainerConfigRequests.build(\n    {\n      // Custom environment variables\n      CUSTOM_ENV_VAR: 'production-value',\n      FEATURE_FLAG_ENABLED: 'true',\n      LOG_LEVEL: environment === 'production' ? 'info' : 'debug',\n      CUSTOM_API_ENDPOINT: 'https://api.example.com',\n    },\n    customSecretsRequests\n  );\n\n  console.log(`✅ Custom container configuration prepared`);\n  console.log(`   - Custom environment variables: 4`);\n  console.log(`   - Secret configurations to check: ${customSecretsRequests.secretConfigurations.length}`);\n  console.log(`   - Total potential secret mappings: ${customSecretsRequests.getTotalMappingsCount()}`);\n  console.log(`   - Adapter will load secrets that exist and skip those that don't`);\n\n  try {\n    // BackendStackを作成\n    new BackendStack(app, stackName, {\n      projectName,\n      environment: environment,\n      domainName,\n      imageTag,\n      containerPort,\n      cpu,\n      memory,\n      desiredCount,\n      databaseStackName,\n      isDatabaseStackDeployed,\n      rdsRequests,\n      containerConfigRequests,\n      /* If you don't specify 'env', this stack will be environment-agnostic.\n       * Account/Region-dependent features and context lookups will not work,\n       * but a single synthesized template can be deployed anywhere. */\n\n      /* Uncomment the next line to specialize this stack for the AWS Account\n       * and Region that are implied by the current CLI configuration. */\n      env: { account: getCdkDefaultAccount(), region: getCdkDefaultRegion() },\n\n      /* Uncomment the next line if you know exactly what Account and Region you\n       * want to deploy the stack to. */\n      // env: { account: '123456789012', region: 'us-east-1' },\n\n      /* For more information, see https://docs.aws.amazon.com/cdk/latest/guide/environments.html */\n      tags: createDefaultTags(projectName, environment, STACK_TYPES.BACKEND, 'backend')\n    });\n\n    console.log(`Successfully created ${stackName}`);\n  } catch (error) {\n    console.error('Failed to create BackendStack:', error instanceof Error ? error.message : String(error));\n    process.exit(1);\n  }\n})(); "]}