hyperp 1.0.1 → 1.0.3

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.
package/cdk.json CHANGED
@@ -1,5 +1,5 @@
1
1
  {
2
- "app": "npx ts-node --prefer-ts-exts bin/hyperp.ts",
2
+ "app": "node dist/bin/hyperp.js",
3
3
  "watch": {
4
4
  "include": ["**"],
5
5
  "exclude": [
@@ -48,9 +48,39 @@ const iam = __importStar(require("aws-cdk-lib/aws-iam"));
48
48
  const events = __importStar(require("aws-cdk-lib/aws-events"));
49
49
  const targets = __importStar(require("aws-cdk-lib/aws-events-targets"));
50
50
  const path = __importStar(require("path"));
51
+ const fs = __importStar(require("fs"));
51
52
  class HyperpStack extends cdk.Stack {
52
53
  constructor(scope, id, props) {
53
54
  super(scope, id, props);
55
+ // Resolve Lambda entry paths - handle both development and npm package contexts
56
+ // When compiled, __dirname is dist/lib/, so we need to go up to package root
57
+ // When in npm package, we need to find the package root
58
+ const getLambdaPath = (lambdaPath) => {
59
+ // Try relative to __dirname first (for npm package - go up 2 levels from dist/lib/)
60
+ const npmPath = path.join(__dirname, "../../lambda", lambdaPath);
61
+ if (fs.existsSync(npmPath)) {
62
+ return npmPath;
63
+ }
64
+ // Try relative to __dirname with one less level (for development - from lib/)
65
+ const devPath = path.join(__dirname, "../lambda", lambdaPath);
66
+ if (fs.existsSync(devPath)) {
67
+ return devPath;
68
+ }
69
+ // Fallback: try to find package root by looking for package.json
70
+ let currentDir = __dirname;
71
+ while (currentDir !== path.dirname(currentDir)) {
72
+ const packageJsonPath = path.join(currentDir, "package.json");
73
+ if (fs.existsSync(packageJsonPath)) {
74
+ const lambdaFullPath = path.join(currentDir, "lambda", lambdaPath);
75
+ if (fs.existsSync(lambdaFullPath)) {
76
+ return lambdaFullPath;
77
+ }
78
+ }
79
+ currentDir = path.dirname(currentDir);
80
+ }
81
+ // Last resort: return the path relative to __dirname (will fail if file doesn't exist)
82
+ return path.join(__dirname, "../lambda", lambdaPath);
83
+ };
54
84
  // Get GitHub App configuration from CDK context
55
85
  const githubAppId = this.node.tryGetContext("githubAppId") || process.env.GITHUB_APP_ID;
56
86
  const githubAppWebhookSecret = this.node.tryGetContext("githubAppWebhookSecret") ||
@@ -325,7 +355,7 @@ class HyperpStack extends cdk.Stack {
325
355
  functionName: "hyperp-efs-controller",
326
356
  runtime: lambda.Runtime.NODEJS_20_X,
327
357
  handler: "handler",
328
- entry: path.join(__dirname, "../lambda/efs-controller/index.ts"),
358
+ entry: getLambdaPath("efs-controller/index.ts"),
329
359
  timeout: cdk.Duration.seconds(30),
330
360
  memorySize: 512,
331
361
  vpc: vpc,
@@ -356,7 +386,7 @@ class HyperpStack extends cdk.Stack {
356
386
  functionName: "hyperp-github-webhook-handler",
357
387
  runtime: lambda.Runtime.NODEJS_20_X,
358
388
  handler: "handler",
359
- entry: path.join(__dirname, "../lambda/github-webhook-handler/index.ts"),
389
+ entry: getLambdaPath("github-webhook-handler/index.ts"),
360
390
  timeout: cdk.Duration.minutes(5),
361
391
  memorySize: 1024,
362
392
  environment: {
@@ -399,7 +429,7 @@ class HyperpStack extends cdk.Stack {
399
429
  functionName: "hyperp-task-state-change-handler",
400
430
  runtime: lambda.Runtime.NODEJS_20_X,
401
431
  handler: "handler",
402
- entry: path.join(__dirname, "../lambda/task-state-change-handler/index.ts"),
432
+ entry: getLambdaPath("task-state-change-handler/index.ts"),
403
433
  timeout: cdk.Duration.minutes(2),
404
434
  memorySize: 512,
405
435
  environment: {
@@ -492,7 +522,7 @@ class HyperpStack extends cdk.Stack {
492
522
  functionName: "hyperp-cli-rest-api",
493
523
  runtime: lambda.Runtime.NODEJS_20_X,
494
524
  handler: "handler",
495
- entry: path.join(__dirname, "../lambda/cli-rest-api/index.ts"),
525
+ entry: getLambdaPath("cli-rest-api/index.ts"),
496
526
  timeout: cdk.Duration.seconds(30),
497
527
  memorySize: 512,
498
528
  environment: {
@@ -531,4 +561,4 @@ class HyperpStack extends cdk.Stack {
531
561
  }
532
562
  }
533
563
  exports.HyperpStack = HyperpStack;
534
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hyperp-stack.js","sourceRoot":"","sources":["../../lib/hyperp-stack.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from \"aws-cdk-lib\";\nimport { Construct } from \"constructs\";\nimport * as ec2 from \"aws-cdk-lib/aws-ec2\";\nimport * as ecs from \"aws-cdk-lib/aws-ecs\";\nimport * as ecr from \"aws-cdk-lib/aws-ecr\";\nimport * as efs from \"aws-cdk-lib/aws-efs\";\nimport * as dynamodb from \"aws-cdk-lib/aws-dynamodb\";\nimport * as s3 from \"aws-cdk-lib/aws-s3\";\nimport * as lambda from \"aws-cdk-lib/aws-lambda\";\nimport { NodejsFunction, OutputFormat } from \"aws-cdk-lib/aws-lambda-nodejs\";\nimport * as logs from \"aws-cdk-lib/aws-logs\";\nimport * as iam from \"aws-cdk-lib/aws-iam\";\nimport * as events from \"aws-cdk-lib/aws-events\";\nimport * as targets from \"aws-cdk-lib/aws-events-targets\";\nimport * as s3deploy from \"aws-cdk-lib/aws-s3-deployment\";\nimport * as path from \"path\";\n\nexport class HyperpStack extends cdk.Stack {\n  constructor(scope: Construct, id: string, props?: cdk.StackProps) {\n    super(scope, id, props);\n\n    // Get GitHub App configuration from CDK context\n    const githubAppId =\n      this.node.tryGetContext(\"githubAppId\") || process.env.GITHUB_APP_ID;\n    const githubAppWebhookSecret =\n      this.node.tryGetContext(\"githubAppWebhookSecret\") ||\n      process.env.GITHUB_APP_WEBHOOK_SECRET;\n\n    if (!githubAppId) {\n      throw new Error(\n        \"GitHub App ID is required. Provide it via --github-app-id or CDK context.\"\n      );\n    }\n\n    if (!githubAppWebhookSecret) {\n      throw new Error(\n        \"GitHub App webhook secret is required. Provide it via --github-app-webhook-secret or CDK context.\"\n      );\n    }\n\n    // ============================================\n    // VPC - Large addressable VPC (Public subnets only to avoid NAT gateway costs)\n    // ============================================\n    const vpc = new ec2.Vpc(this, \"HyperpVpc\", {\n      maxAzs: 3,\n      ipAddresses: ec2.IpAddresses.cidr(\"10.0.0.0/16\"),\n      natGateways: 0, // No NAT gateways - using public subnets only\n      subnetConfiguration: [\n        {\n          cidrMask: 20,\n          name: \"Public\",\n          subnetType: ec2.SubnetType.PUBLIC,\n        },\n      ],\n    });\n\n    // ============================================\n    // S3 Bucket for artifacts (GitHub App PEM key)\n    // ============================================\n    // Let CDK generate a unique bucket name to avoid global naming conflicts\n    // The bucket name will be available via stack outputs\n    const artifactsBucket = new s3.Bucket(this, \"HyperpArtifactsBucket\", {\n      // Removed fixed bucketName to avoid global naming conflicts\n      // CDK will generate: hyperpstack-hyperpartifactsbucket-<random>\n      versioned: true,\n      encryption: s3.BucketEncryption.S3_MANAGED,\n      blockPublicAccess: s3.BlockPublicAccess.BLOCK_ALL,\n      removalPolicy: cdk.RemovalPolicy.RETAIN,\n      autoDeleteObjects: false, // Keep objects when stack is deleted\n    });\n\n    // ============================================\n    // DynamoDB Table with GSI\n    // ============================================\n    const table = new dynamodb.Table(this, \"HyperpTable\", {\n      tableName: \"hyperp\",\n      partitionKey: { name: \"PK\", type: dynamodb.AttributeType.STRING },\n      sortKey: { name: \"SK\", type: dynamodb.AttributeType.STRING },\n      billingMode: dynamodb.BillingMode.PAY_PER_REQUEST,\n      removalPolicy: cdk.RemovalPolicy.RETAIN,\n      pointInTimeRecovery: true,\n    });\n\n    // GSI1 for querying workflows, workflow runs, etc.\n    table.addGlobalSecondaryIndex({\n      indexName: \"GSI1\",\n      partitionKey: { name: \"GSI1-PK\", type: dynamodb.AttributeType.STRING },\n      sortKey: { name: \"GSI1-SK\", type: dynamodb.AttributeType.STRING },\n    });\n\n    // GSI2 for querying workflow runs by commit hash\n    table.addGlobalSecondaryIndex({\n      indexName: \"GSI2\",\n      partitionKey: { name: \"GSI2-PK\", type: dynamodb.AttributeType.STRING },\n      sortKey: { name: \"GSI2-SK\", type: dynamodb.AttributeType.STRING },\n    });\n\n    // GSI3 for querying all workflow runs sorted by creation time\n    table.addGlobalSecondaryIndex({\n      indexName: \"GSI3\",\n      partitionKey: { name: \"GSI3-PK\", type: dynamodb.AttributeType.STRING },\n      sortKey: { name: \"GSI3-SK\", type: dynamodb.AttributeType.STRING },\n      projectionType: dynamodb.ProjectionType.INCLUDE,\n      nonKeyAttributes: [\n        \"workflowName\",\n        \"status\",\n        \"createdAt\",\n        \"updatedAt\",\n        \"entityId\",\n      ],\n    });\n\n    // ============================================\n    // ECR Repository\n    // ============================================\n    const ecrRepository = new ecr.Repository(this, \"HyperpRepository\", {\n      repositoryName: \"hyperp-1\",\n      removalPolicy: cdk.RemovalPolicy.RETAIN,\n      imageScanOnPush: false,\n      lifecycleRules: [\n        {\n          description: \"Keep last 100 images\",\n          maxImageCount: 100,\n        },\n      ],\n    });\n\n    // ============================================\n    // ECS Cluster\n    // ============================================\n    const cluster = new ecs.Cluster(this, \"HyperpCluster\", {\n      clusterName: \"hyperp-1\",\n      vpc: vpc,\n      containerInsights: true,\n    });\n\n    // ============================================\n    // EFS File System for artifact sharing\n    // ============================================\n    const fileSystem = new efs.FileSystem(this, \"HyperpEfs\", {\n      vpc: vpc,\n      encrypted: true,\n      lifecyclePolicy: efs.LifecyclePolicy.AFTER_14_DAYS,\n      performanceMode: efs.PerformanceMode.GENERAL_PURPOSE,\n      throughputMode: efs.ThroughputMode.ELASTIC,\n      removalPolicy: cdk.RemovalPolicy.RETAIN,\n      fileSystemPolicy: new iam.PolicyDocument({\n        statements: [\n          new iam.PolicyStatement({\n            actions: [\n              \"elasticfilesystem:ClientMount\",\n              \"elasticfilesystem:ClientWrite\",\n              \"elasticfilesystem:ClientRootAccess\",\n            ],\n            principals: [new iam.AnyPrincipal()],\n            resources: [\"*\"],\n            conditions: {\n              Bool: {\n                \"elasticfilesystem:AccessedViaMountTarget\": \"true\",\n              },\n            },\n          }),\n        ],\n      }),\n    });\n\n    // Security group for ECS tasks\n    const ecsTaskSecurityGroup = new ec2.SecurityGroup(\n      this,\n      \"EcsTaskSecurityGroup\",\n      {\n        vpc: vpc,\n        description: \"Security group for ECS tasks\",\n        allowAllOutbound: true,\n      }\n    );\n\n    // Allow ECS tasks to connect to EFS on NFS port\n    fileSystem.connections.allowFrom(\n      ecsTaskSecurityGroup,\n      ec2.Port.tcp(2049),\n      \"Allow NFS from ECS tasks\"\n    );\n\n    // ============================================\n    // CloudWatch Logs\n    // ============================================\n    const ecsLogGroup = new logs.LogGroup(this, \"HyperpEcsLogGroup\", {\n      logGroupName: \"/hyperp\",\n      retention: logs.RetentionDays.ONE_WEEK,\n      removalPolicy: cdk.RemovalPolicy.DESTROY,\n    });\n\n    // ============================================\n    // IAM Roles\n    // ============================================\n\n    // ECS Task Execution Role\n    const ecsTaskExecutionRole = new iam.Role(this, \"EcsTaskExecutionRole\", {\n      assumedBy: new iam.ServicePrincipal(\"ecs-tasks.amazonaws.com\"),\n      managedPolicies: [\n        iam.ManagedPolicy.fromAwsManagedPolicyName(\n          \"service-role/AmazonECSTaskExecutionRolePolicy\"\n        ),\n      ],\n    });\n\n    // Grant ECR permissions\n    ecrRepository.grantPull(ecsTaskExecutionRole);\n\n    // ECS Task Role (for containers)\n    const ecsTaskRole = new iam.Role(this, \"EcsTaskRole\", {\n      assumedBy: new iam.ServicePrincipal(\"ecs-tasks.amazonaws.com\"),\n    });\n\n    // Grant EFS permissions to task role\n    ecsTaskRole.addToPolicy(\n      new iam.PolicyStatement({\n        actions: [\n          \"elasticfilesystem:ClientMount\",\n          \"elasticfilesystem:ClientWrite\",\n          \"elasticfilesystem:ClientRootAccess\",\n          \"elasticfilesystem:DescribeMountTargets\",\n        ],\n        resources: [fileSystem.fileSystemArn],\n      })\n    );\n\n    // Grant ECR push/pull for image builds\n    ecrRepository.grantPullPush(ecsTaskRole);\n\n    // Grant S3 access for reading repository content and uploading artifacts\n    ecsTaskRole.addToPolicy(\n      new iam.PolicyStatement({\n        actions: [\"s3:GetObject\", \"s3:ListBucket\", \"s3:PutObject\"],\n        resources: [\"*\"],\n      })\n    );\n\n    // Grant DynamoDB permissions for usage calculator tasks\n    table.grantReadWriteData(ecsTaskRole);\n    // Explicitly grant UpdateItem permission to ensure usage calculator can update items\n    ecsTaskRole.addToPolicy(\n      new iam.PolicyStatement({\n        actions: [\n          \"dynamodb:GetItem\",\n          \"dynamodb:PutItem\",\n          \"dynamodb:UpdateItem\",\n          \"dynamodb:Query\",\n          \"dynamodb:Scan\",\n        ],\n        resources: [table.tableArn, `${table.tableArn}/index/*`],\n      })\n    );\n\n    // ============================================\n    // Lambda Execution Role\n    // ============================================\n    const lambdaRole = new iam.Role(this, \"HyperpLambdaRole\", {\n      assumedBy: new iam.ServicePrincipal(\"lambda.amazonaws.com\"),\n      managedPolicies: [\n        iam.ManagedPolicy.fromAwsManagedPolicyName(\n          \"service-role/AWSLambdaVPCAccessExecutionRole\"\n        ),\n        iam.ManagedPolicy.fromAwsManagedPolicyName(\n          \"service-role/AWSLambdaBasicExecutionRole\"\n        ),\n      ],\n    });\n\n    // Grant DynamoDB permissions\n    table.grantReadWriteData(lambdaRole);\n\n    // Grant S3 permissions (read and write for downloadable artifacts)\n    artifactsBucket.grantReadWrite(lambdaRole);\n\n    // Grant CloudWatch Logs permissions\n    lambdaRole.addToPolicy(\n      new iam.PolicyStatement({\n        actions: [\"logs:GetLogEvents\", \"logs:DescribeLogStreams\"],\n        resources: [\"*\"],\n      })\n    );\n\n    // Grant ECS permissions for downloadable creator tasks\n    lambdaRole.addToPolicy(\n      new iam.PolicyStatement({\n        actions: [\n          \"ecs:RegisterTaskDefinition\",\n          \"ecs:RunTask\",\n          \"ecs:DescribeTasks\",\n        ],\n        resources: [\"*\"],\n      })\n    );\n\n    // Grant ECS permissions\n    lambdaRole.addToPolicy(\n      new iam.PolicyStatement({\n        actions: [\n          \"ecs:RegisterTaskDefinition\",\n          \"ecs:RunTask\",\n          \"ecs:DescribeTasks\",\n          \"ecs:StopTask\",\n          \"ecs:ListTasks\",\n        ],\n        resources: [\"*\"],\n      })\n    );\n\n    // Grant IAM PassRole for ECS\n    lambdaRole.addToPolicy(\n      new iam.PolicyStatement({\n        actions: [\"iam:PassRole\"],\n        resources: [ecsTaskExecutionRole.roleArn, ecsTaskRole.roleArn],\n      })\n    );\n\n    // Grant ECR permissions\n    lambdaRole.addToPolicy(\n      new iam.PolicyStatement({\n        actions: [\n          \"ecr:GetAuthorizationToken\",\n          \"ecr:BatchCheckLayerAvailability\",\n          \"ecr:GetDownloadUrlForLayer\",\n          \"ecr:BatchGetImage\",\n          \"ecr:BatchDeleteImage\",\n        ],\n        resources: [\"*\"],\n      })\n    );\n\n    // Grant Lambda invoke permissions (for EFS controller)\n    lambdaRole.addToPolicy(\n      new iam.PolicyStatement({\n        actions: [\"lambda:InvokeFunction\"],\n        resources: [\n          `arn:aws:lambda:${this.region}:${this.account}:function:hyperp-efs-controller`,\n          `arn:aws:lambda:${this.region}:${this.account}:function:hyperp-efs-controller:*`,\n        ],\n      })\n    );\n\n    // ============================================\n    // EFS Controller Lambda\n    // ============================================\n    // Create an access point for Lambda to mount EFS\n    // Mount at /hyperp-artifacts so Lambda has write access to that directory\n    const lambdaAccessPoint = new efs.AccessPoint(\n      this,\n      \"LambdaEfsAccessPoint\",\n      {\n        fileSystem: fileSystem,\n        path: \"/hyperp-artifacts\",\n        createAcl: {\n          ownerGid: \"1000\",\n          ownerUid: \"1000\",\n          permissions: \"777\", // More permissive for Lambda to create subdirectories\n        },\n        posixUser: {\n          uid: \"1000\",\n          gid: \"1000\",\n        },\n      }\n    );\n\n    const efsControllerLambda = new NodejsFunction(\n      this,\n      \"HyperpEfsController\",\n      {\n        functionName: \"hyperp-efs-controller\",\n        runtime: lambda.Runtime.NODEJS_20_X,\n        handler: \"handler\",\n        entry: path.join(__dirname, \"../lambda/efs-controller/index.ts\"),\n        timeout: cdk.Duration.seconds(30),\n        memorySize: 512,\n        vpc: vpc,\n        vpcSubnets: { subnetType: ec2.SubnetType.PUBLIC },\n        securityGroups: [ecsTaskSecurityGroup],\n        allowPublicSubnet: true,\n        environment: {\n          EFS_FILE_SYSTEM_ID: fileSystem.fileSystemId,\n        },\n        role: lambdaRole,\n        filesystem: lambda.FileSystem.fromEfsAccessPoint(\n          lambdaAccessPoint,\n          \"/mnt/efs\"\n        ),\n        bundling: {\n          minify: true,\n          sourceMap: true,\n          format: OutputFormat.CJS,\n        },\n      }\n    );\n\n    // ============================================\n    // Collect subnet IDs for environment variables\n    // ============================================\n    const publicSubnetIds = vpc.publicSubnets\n      .map((subnet) => subnet.subnetId)\n      .join(\",\");\n\n    // ============================================\n    // GitHub Webhook Handler Lambda\n    // ============================================\n    const githubWebhookHandler = new NodejsFunction(\n      this,\n      \"HyperpGithubWebhookHandler\",\n      {\n        functionName: \"hyperp-github-webhook-handler\",\n        runtime: lambda.Runtime.NODEJS_20_X,\n        handler: \"handler\",\n        entry: path.join(\n          __dirname,\n          \"../lambda/github-webhook-handler/index.ts\"\n        ),\n        timeout: cdk.Duration.minutes(5),\n        memorySize: 1024,\n        environment: {\n          DYNAMODB_TABLE_NAME: table.tableName,\n          S3_BUCKET_NAME: artifactsBucket.bucketName,\n          ECS_CLUSTER_ARN: cluster.clusterArn,\n          ECS_CLUSTER_NAME: cluster.clusterName,\n          ECR_REPOSITORY_URI: ecrRepository.repositoryUri,\n          ECR_REPOSITORY_NAME: ecrRepository.repositoryName,\n          ECS_TASK_EXECUTION_ROLE_ARN: ecsTaskExecutionRole.roleArn,\n          ECS_TASK_ROLE_ARN: ecsTaskRole.roleArn,\n          EFS_FILE_SYSTEM_ID: fileSystem.fileSystemId,\n          ECS_TASK_SECURITY_GROUP_ID: ecsTaskSecurityGroup.securityGroupId,\n          PUBLIC_SUBNET_IDS: publicSubnetIds,\n          EFS_CONTROLLER_LAMBDA_ARN: efsControllerLambda.functionArn,\n          GITHUB_APP_ID: githubAppId,\n          GITHUB_APP_WEBHOOK_SECRET: githubAppWebhookSecret,\n          AWS_ACCOUNT_ID: this.account,\n        },\n        role: lambdaRole,\n        bundling: {\n          minify: true,\n          sourceMap: true,\n          format: OutputFormat.CJS,\n        },\n      }\n    );\n\n    // Create Function URL for GitHub Webhook Handler\n    const webhookFunctionUrl = githubWebhookHandler.addFunctionUrl({\n      authType: lambda.FunctionUrlAuthType.NONE, // Public access for GitHub webhooks\n      cors: {\n        allowedOrigins: [\"*\"],\n        allowedMethods: [lambda.HttpMethod.POST],\n        allowedHeaders: [\"*\"],\n      },\n    });\n\n    // ============================================\n    // Task State Change Handler Lambda\n    // ============================================\n    const taskStateChangeHandler = new NodejsFunction(\n      this,\n      \"HyperpTaskStateChangeHandler\",\n      {\n        functionName: \"hyperp-task-state-change-handler\",\n        runtime: lambda.Runtime.NODEJS_20_X,\n        handler: \"handler\",\n        entry: path.join(\n          __dirname,\n          \"../lambda/task-state-change-handler/index.ts\"\n        ),\n        timeout: cdk.Duration.minutes(2),\n        memorySize: 512,\n        environment: {\n          DYNAMODB_TABLE_NAME: table.tableName,\n          ECS_CLUSTER_ARN: cluster.clusterArn,\n          ECS_CLUSTER_NAME: cluster.clusterName,\n          ECR_REPOSITORY_URI: ecrRepository.repositoryUri,\n          ECS_TASK_EXECUTION_ROLE_ARN: ecsTaskExecutionRole.roleArn,\n          ECS_TASK_ROLE_ARN: ecsTaskRole.roleArn,\n          EFS_FILE_SYSTEM_ID: fileSystem.fileSystemId,\n          ECS_TASK_SECURITY_GROUP_ID: ecsTaskSecurityGroup.securityGroupId,\n          PUBLIC_SUBNET_IDS: publicSubnetIds,\n          EFS_CONTROLLER_LAMBDA_ARN: efsControllerLambda.functionArn,\n          AWS_ACCOUNT_ID: this.account,\n          FARGATE_VCPU_PRICE_PER_HOUR: \"0.04048\",\n          FARGATE_GB_PRICE_PER_HOUR: \"0.004445\",\n          S3_BUCKET_NAME: artifactsBucket.bucketName,\n          GITHUB_APP_ID: githubAppId,\n          DOWNLOADABLE_CREATOR_IMAGE:\n            \"public.ecr.aws/b6g9t8f1/downloadable-creator:latest\",\n          USAGE_CALCULATOR_IMAGE:\n            \"public.ecr.aws/b6g9t8f1/usage-calculator:latest\",\n        },\n        role: lambdaRole,\n        bundling: {\n          minify: true,\n          sourceMap: true,\n          format: OutputFormat.CJS,\n        },\n      }\n    );\n\n    // ============================================\n    // EventBridge Rule for ECS Task State Changes\n    // ============================================\n    const ecsTaskStateChangeRule = new events.Rule(\n      this,\n      \"EcsTaskStateChangeRule\",\n      {\n        eventPattern: {\n          source: [\"aws.ecs\"],\n          detailType: [\"ECS Task State Change\"],\n          detail: {\n            clusterArn: [cluster.clusterArn],\n            lastStatus: [\"STOPPED\"],\n          },\n        },\n      }\n    );\n\n    ecsTaskStateChangeRule.addTarget(\n      new targets.LambdaFunction(taskStateChangeHandler)\n    );\n\n    // ============================================\n    // Outputs\n    // ============================================\n    new cdk.CfnOutput(this, \"VpcId\", {\n      value: vpc.vpcId,\n      description: \"VPC ID\",\n    });\n\n    new cdk.CfnOutput(this, \"ClusterArn\", {\n      value: cluster.clusterArn,\n      description: \"ECS Cluster ARN\",\n    });\n\n    new cdk.CfnOutput(this, \"EcrRepositoryUri\", {\n      value: ecrRepository.repositoryUri,\n      description: \"ECR Repository URI\",\n    });\n\n    new cdk.CfnOutput(this, \"DynamoDbTableName\", {\n      value: table.tableName,\n      description: \"DynamoDB Table Name\",\n    });\n\n    new cdk.CfnOutput(this, \"S3BucketName\", {\n      value: artifactsBucket.bucketName,\n      description: \"S3 Artifacts Bucket Name\",\n    });\n\n    new cdk.CfnOutput(this, \"EfsFileSystemId\", {\n      value: fileSystem.fileSystemId,\n      description: \"EFS File System ID\",\n    });\n\n    new cdk.CfnOutput(this, \"GithubWebhookHandlerArn\", {\n      value: githubWebhookHandler.functionArn,\n      description: \"GitHub Webhook Handler Lambda ARN\",\n    });\n\n    new cdk.CfnOutput(this, \"GithubWebhookHandlerUrl\", {\n      value: webhookFunctionUrl.url,\n      description: \"GitHub Webhook Handler Function URL\",\n    });\n\n    new cdk.CfnOutput(this, \"TaskStateChangeHandlerArn\", {\n      value: taskStateChangeHandler.functionArn,\n      description: \"Task State Change Handler Lambda ARN\",\n    });\n\n    new cdk.CfnOutput(this, \"PublicSubnetIds\", {\n      value: publicSubnetIds,\n      description: \"Public Subnet IDs\",\n    });\n\n    // ============================================\n    // REST API Lambda for CLI\n    // ============================================\n    const cliRestApiLambda = new NodejsFunction(this, \"HyperpCliRestApi\", {\n      functionName: \"hyperp-cli-rest-api\",\n      runtime: lambda.Runtime.NODEJS_20_X,\n      handler: \"handler\",\n      entry: path.join(__dirname, \"../lambda/cli-rest-api/index.ts\"),\n      timeout: cdk.Duration.seconds(30),\n      memorySize: 512,\n      environment: {\n        DYNAMODB_TABLE_NAME: table.tableName,\n        ECS_CLUSTER_NAME: cluster.clusterName,\n        EFS_FILE_SYSTEM_ID: fileSystem.fileSystemId,\n        PUBLIC_SUBNET_IDS: vpc.publicSubnets.map((s) => s.subnetId).join(\",\"),\n        ECS_TASK_SECURITY_GROUP_ID: ecsTaskSecurityGroup.securityGroupId,\n        ECS_TASK_EXECUTION_ROLE_ARN: ecsTaskExecutionRole.roleArn,\n        ECS_TASK_ROLE_ARN: ecsTaskRole.roleArn,\n        S3_BUCKET_NAME: artifactsBucket.bucketName,\n        DOWNLOADABLE_CREATOR_IMAGE:\n          \"public.ecr.aws/b6g9t8f1/downloadable-creator:latest\",\n        GITHUB_APP_ID: githubAppId,\n        GITHUB_APP_WEBHOOK_SECRET: githubAppWebhookSecret,\n      },\n      role: lambdaRole,\n      bundling: {\n        minify: true,\n        sourceMap: true,\n        format: OutputFormat.CJS,\n      },\n    });\n\n    // Create Function URL\n    const functionUrl = cliRestApiLambda.addFunctionUrl({\n      authType: lambda.FunctionUrlAuthType.NONE, // Public access (can be secured later)\n      cors: {\n        allowedOrigins: [\"*\"],\n        allowedMethods: [lambda.HttpMethod.GET, lambda.HttpMethod.POST],\n        allowedHeaders: [\"*\"],\n      },\n    });\n\n    new cdk.CfnOutput(this, \"CliRestApiUrl\", {\n      value: functionUrl.url,\n      description: \"CLI REST API Function URL\",\n    });\n  }\n}\n"]}
564
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hyperp-stack.js","sourceRoot":"","sources":["../../lib/hyperp-stack.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from \"aws-cdk-lib\";\nimport { Construct } from \"constructs\";\nimport * as ec2 from \"aws-cdk-lib/aws-ec2\";\nimport * as ecs from \"aws-cdk-lib/aws-ecs\";\nimport * as ecr from \"aws-cdk-lib/aws-ecr\";\nimport * as efs from \"aws-cdk-lib/aws-efs\";\nimport * as dynamodb from \"aws-cdk-lib/aws-dynamodb\";\nimport * as s3 from \"aws-cdk-lib/aws-s3\";\nimport * as lambda from \"aws-cdk-lib/aws-lambda\";\nimport { NodejsFunction, OutputFormat } from \"aws-cdk-lib/aws-lambda-nodejs\";\nimport * as logs from \"aws-cdk-lib/aws-logs\";\nimport * as iam from \"aws-cdk-lib/aws-iam\";\nimport * as events from \"aws-cdk-lib/aws-events\";\nimport * as targets from \"aws-cdk-lib/aws-events-targets\";\nimport * as s3deploy from \"aws-cdk-lib/aws-s3-deployment\";\nimport * as path from \"path\";\nimport * as fs from \"fs\";\n\nexport class HyperpStack extends cdk.Stack {\n  constructor(scope: Construct, id: string, props?: cdk.StackProps) {\n    super(scope, id, props);\n\n    // Resolve Lambda entry paths - handle both development and npm package contexts\n    // When compiled, __dirname is dist/lib/, so we need to go up to package root\n    // When in npm package, we need to find the package root\n    const getLambdaPath = (lambdaPath: string): string => {\n      // Try relative to __dirname first (for npm package - go up 2 levels from dist/lib/)\n      const npmPath = path.join(__dirname, \"../../lambda\", lambdaPath);\n      if (fs.existsSync(npmPath)) {\n        return npmPath;\n      }\n      // Try relative to __dirname with one less level (for development - from lib/)\n      const devPath = path.join(__dirname, \"../lambda\", lambdaPath);\n      if (fs.existsSync(devPath)) {\n        return devPath;\n      }\n      // Fallback: try to find package root by looking for package.json\n      let currentDir = __dirname;\n      while (currentDir !== path.dirname(currentDir)) {\n        const packageJsonPath = path.join(currentDir, \"package.json\");\n        if (fs.existsSync(packageJsonPath)) {\n          const lambdaFullPath = path.join(currentDir, \"lambda\", lambdaPath);\n          if (fs.existsSync(lambdaFullPath)) {\n            return lambdaFullPath;\n          }\n        }\n        currentDir = path.dirname(currentDir);\n      }\n      // Last resort: return the path relative to __dirname (will fail if file doesn't exist)\n      return path.join(__dirname, \"../lambda\", lambdaPath);\n    };\n\n    // Get GitHub App configuration from CDK context\n    const githubAppId =\n      this.node.tryGetContext(\"githubAppId\") || process.env.GITHUB_APP_ID;\n    const githubAppWebhookSecret =\n      this.node.tryGetContext(\"githubAppWebhookSecret\") ||\n      process.env.GITHUB_APP_WEBHOOK_SECRET;\n\n    if (!githubAppId) {\n      throw new Error(\n        \"GitHub App ID is required. Provide it via --github-app-id or CDK context.\"\n      );\n    }\n\n    if (!githubAppWebhookSecret) {\n      throw new Error(\n        \"GitHub App webhook secret is required. Provide it via --github-app-webhook-secret or CDK context.\"\n      );\n    }\n\n    // ============================================\n    // VPC - Large addressable VPC (Public subnets only to avoid NAT gateway costs)\n    // ============================================\n    const vpc = new ec2.Vpc(this, \"HyperpVpc\", {\n      maxAzs: 3,\n      ipAddresses: ec2.IpAddresses.cidr(\"10.0.0.0/16\"),\n      natGateways: 0, // No NAT gateways - using public subnets only\n      subnetConfiguration: [\n        {\n          cidrMask: 20,\n          name: \"Public\",\n          subnetType: ec2.SubnetType.PUBLIC,\n        },\n      ],\n    });\n\n    // ============================================\n    // S3 Bucket for artifacts (GitHub App PEM key)\n    // ============================================\n    // Let CDK generate a unique bucket name to avoid global naming conflicts\n    // The bucket name will be available via stack outputs\n    const artifactsBucket = new s3.Bucket(this, \"HyperpArtifactsBucket\", {\n      // Removed fixed bucketName to avoid global naming conflicts\n      // CDK will generate: hyperpstack-hyperpartifactsbucket-<random>\n      versioned: true,\n      encryption: s3.BucketEncryption.S3_MANAGED,\n      blockPublicAccess: s3.BlockPublicAccess.BLOCK_ALL,\n      removalPolicy: cdk.RemovalPolicy.RETAIN,\n      autoDeleteObjects: false, // Keep objects when stack is deleted\n    });\n\n    // ============================================\n    // DynamoDB Table with GSI\n    // ============================================\n    const table = new dynamodb.Table(this, \"HyperpTable\", {\n      tableName: \"hyperp\",\n      partitionKey: { name: \"PK\", type: dynamodb.AttributeType.STRING },\n      sortKey: { name: \"SK\", type: dynamodb.AttributeType.STRING },\n      billingMode: dynamodb.BillingMode.PAY_PER_REQUEST,\n      removalPolicy: cdk.RemovalPolicy.RETAIN,\n      pointInTimeRecovery: true,\n    });\n\n    // GSI1 for querying workflows, workflow runs, etc.\n    table.addGlobalSecondaryIndex({\n      indexName: \"GSI1\",\n      partitionKey: { name: \"GSI1-PK\", type: dynamodb.AttributeType.STRING },\n      sortKey: { name: \"GSI1-SK\", type: dynamodb.AttributeType.STRING },\n    });\n\n    // GSI2 for querying workflow runs by commit hash\n    table.addGlobalSecondaryIndex({\n      indexName: \"GSI2\",\n      partitionKey: { name: \"GSI2-PK\", type: dynamodb.AttributeType.STRING },\n      sortKey: { name: \"GSI2-SK\", type: dynamodb.AttributeType.STRING },\n    });\n\n    // GSI3 for querying all workflow runs sorted by creation time\n    table.addGlobalSecondaryIndex({\n      indexName: \"GSI3\",\n      partitionKey: { name: \"GSI3-PK\", type: dynamodb.AttributeType.STRING },\n      sortKey: { name: \"GSI3-SK\", type: dynamodb.AttributeType.STRING },\n      projectionType: dynamodb.ProjectionType.INCLUDE,\n      nonKeyAttributes: [\n        \"workflowName\",\n        \"status\",\n        \"createdAt\",\n        \"updatedAt\",\n        \"entityId\",\n      ],\n    });\n\n    // ============================================\n    // ECR Repository\n    // ============================================\n    const ecrRepository = new ecr.Repository(this, \"HyperpRepository\", {\n      repositoryName: \"hyperp-1\",\n      removalPolicy: cdk.RemovalPolicy.RETAIN,\n      imageScanOnPush: false,\n      lifecycleRules: [\n        {\n          description: \"Keep last 100 images\",\n          maxImageCount: 100,\n        },\n      ],\n    });\n\n    // ============================================\n    // ECS Cluster\n    // ============================================\n    const cluster = new ecs.Cluster(this, \"HyperpCluster\", {\n      clusterName: \"hyperp-1\",\n      vpc: vpc,\n      containerInsights: true,\n    });\n\n    // ============================================\n    // EFS File System for artifact sharing\n    // ============================================\n    const fileSystem = new efs.FileSystem(this, \"HyperpEfs\", {\n      vpc: vpc,\n      encrypted: true,\n      lifecyclePolicy: efs.LifecyclePolicy.AFTER_14_DAYS,\n      performanceMode: efs.PerformanceMode.GENERAL_PURPOSE,\n      throughputMode: efs.ThroughputMode.ELASTIC,\n      removalPolicy: cdk.RemovalPolicy.RETAIN,\n      fileSystemPolicy: new iam.PolicyDocument({\n        statements: [\n          new iam.PolicyStatement({\n            actions: [\n              \"elasticfilesystem:ClientMount\",\n              \"elasticfilesystem:ClientWrite\",\n              \"elasticfilesystem:ClientRootAccess\",\n            ],\n            principals: [new iam.AnyPrincipal()],\n            resources: [\"*\"],\n            conditions: {\n              Bool: {\n                \"elasticfilesystem:AccessedViaMountTarget\": \"true\",\n              },\n            },\n          }),\n        ],\n      }),\n    });\n\n    // Security group for ECS tasks\n    const ecsTaskSecurityGroup = new ec2.SecurityGroup(\n      this,\n      \"EcsTaskSecurityGroup\",\n      {\n        vpc: vpc,\n        description: \"Security group for ECS tasks\",\n        allowAllOutbound: true,\n      }\n    );\n\n    // Allow ECS tasks to connect to EFS on NFS port\n    fileSystem.connections.allowFrom(\n      ecsTaskSecurityGroup,\n      ec2.Port.tcp(2049),\n      \"Allow NFS from ECS tasks\"\n    );\n\n    // ============================================\n    // CloudWatch Logs\n    // ============================================\n    const ecsLogGroup = new logs.LogGroup(this, \"HyperpEcsLogGroup\", {\n      logGroupName: \"/hyperp\",\n      retention: logs.RetentionDays.ONE_WEEK,\n      removalPolicy: cdk.RemovalPolicy.DESTROY,\n    });\n\n    // ============================================\n    // IAM Roles\n    // ============================================\n\n    // ECS Task Execution Role\n    const ecsTaskExecutionRole = new iam.Role(this, \"EcsTaskExecutionRole\", {\n      assumedBy: new iam.ServicePrincipal(\"ecs-tasks.amazonaws.com\"),\n      managedPolicies: [\n        iam.ManagedPolicy.fromAwsManagedPolicyName(\n          \"service-role/AmazonECSTaskExecutionRolePolicy\"\n        ),\n      ],\n    });\n\n    // Grant ECR permissions\n    ecrRepository.grantPull(ecsTaskExecutionRole);\n\n    // ECS Task Role (for containers)\n    const ecsTaskRole = new iam.Role(this, \"EcsTaskRole\", {\n      assumedBy: new iam.ServicePrincipal(\"ecs-tasks.amazonaws.com\"),\n    });\n\n    // Grant EFS permissions to task role\n    ecsTaskRole.addToPolicy(\n      new iam.PolicyStatement({\n        actions: [\n          \"elasticfilesystem:ClientMount\",\n          \"elasticfilesystem:ClientWrite\",\n          \"elasticfilesystem:ClientRootAccess\",\n          \"elasticfilesystem:DescribeMountTargets\",\n        ],\n        resources: [fileSystem.fileSystemArn],\n      })\n    );\n\n    // Grant ECR push/pull for image builds\n    ecrRepository.grantPullPush(ecsTaskRole);\n\n    // Grant S3 access for reading repository content and uploading artifacts\n    ecsTaskRole.addToPolicy(\n      new iam.PolicyStatement({\n        actions: [\"s3:GetObject\", \"s3:ListBucket\", \"s3:PutObject\"],\n        resources: [\"*\"],\n      })\n    );\n\n    // Grant DynamoDB permissions for usage calculator tasks\n    table.grantReadWriteData(ecsTaskRole);\n    // Explicitly grant UpdateItem permission to ensure usage calculator can update items\n    ecsTaskRole.addToPolicy(\n      new iam.PolicyStatement({\n        actions: [\n          \"dynamodb:GetItem\",\n          \"dynamodb:PutItem\",\n          \"dynamodb:UpdateItem\",\n          \"dynamodb:Query\",\n          \"dynamodb:Scan\",\n        ],\n        resources: [table.tableArn, `${table.tableArn}/index/*`],\n      })\n    );\n\n    // ============================================\n    // Lambda Execution Role\n    // ============================================\n    const lambdaRole = new iam.Role(this, \"HyperpLambdaRole\", {\n      assumedBy: new iam.ServicePrincipal(\"lambda.amazonaws.com\"),\n      managedPolicies: [\n        iam.ManagedPolicy.fromAwsManagedPolicyName(\n          \"service-role/AWSLambdaVPCAccessExecutionRole\"\n        ),\n        iam.ManagedPolicy.fromAwsManagedPolicyName(\n          \"service-role/AWSLambdaBasicExecutionRole\"\n        ),\n      ],\n    });\n\n    // Grant DynamoDB permissions\n    table.grantReadWriteData(lambdaRole);\n\n    // Grant S3 permissions (read and write for downloadable artifacts)\n    artifactsBucket.grantReadWrite(lambdaRole);\n\n    // Grant CloudWatch Logs permissions\n    lambdaRole.addToPolicy(\n      new iam.PolicyStatement({\n        actions: [\"logs:GetLogEvents\", \"logs:DescribeLogStreams\"],\n        resources: [\"*\"],\n      })\n    );\n\n    // Grant ECS permissions for downloadable creator tasks\n    lambdaRole.addToPolicy(\n      new iam.PolicyStatement({\n        actions: [\n          \"ecs:RegisterTaskDefinition\",\n          \"ecs:RunTask\",\n          \"ecs:DescribeTasks\",\n        ],\n        resources: [\"*\"],\n      })\n    );\n\n    // Grant ECS permissions\n    lambdaRole.addToPolicy(\n      new iam.PolicyStatement({\n        actions: [\n          \"ecs:RegisterTaskDefinition\",\n          \"ecs:RunTask\",\n          \"ecs:DescribeTasks\",\n          \"ecs:StopTask\",\n          \"ecs:ListTasks\",\n        ],\n        resources: [\"*\"],\n      })\n    );\n\n    // Grant IAM PassRole for ECS\n    lambdaRole.addToPolicy(\n      new iam.PolicyStatement({\n        actions: [\"iam:PassRole\"],\n        resources: [ecsTaskExecutionRole.roleArn, ecsTaskRole.roleArn],\n      })\n    );\n\n    // Grant ECR permissions\n    lambdaRole.addToPolicy(\n      new iam.PolicyStatement({\n        actions: [\n          \"ecr:GetAuthorizationToken\",\n          \"ecr:BatchCheckLayerAvailability\",\n          \"ecr:GetDownloadUrlForLayer\",\n          \"ecr:BatchGetImage\",\n          \"ecr:BatchDeleteImage\",\n        ],\n        resources: [\"*\"],\n      })\n    );\n\n    // Grant Lambda invoke permissions (for EFS controller)\n    lambdaRole.addToPolicy(\n      new iam.PolicyStatement({\n        actions: [\"lambda:InvokeFunction\"],\n        resources: [\n          `arn:aws:lambda:${this.region}:${this.account}:function:hyperp-efs-controller`,\n          `arn:aws:lambda:${this.region}:${this.account}:function:hyperp-efs-controller:*`,\n        ],\n      })\n    );\n\n    // ============================================\n    // EFS Controller Lambda\n    // ============================================\n    // Create an access point for Lambda to mount EFS\n    // Mount at /hyperp-artifacts so Lambda has write access to that directory\n    const lambdaAccessPoint = new efs.AccessPoint(\n      this,\n      \"LambdaEfsAccessPoint\",\n      {\n        fileSystem: fileSystem,\n        path: \"/hyperp-artifacts\",\n        createAcl: {\n          ownerGid: \"1000\",\n          ownerUid: \"1000\",\n          permissions: \"777\", // More permissive for Lambda to create subdirectories\n        },\n        posixUser: {\n          uid: \"1000\",\n          gid: \"1000\",\n        },\n      }\n    );\n\n    const efsControllerLambda = new NodejsFunction(\n      this,\n      \"HyperpEfsController\",\n      {\n        functionName: \"hyperp-efs-controller\",\n        runtime: lambda.Runtime.NODEJS_20_X,\n        handler: \"handler\",\n        entry: getLambdaPath(\"efs-controller/index.ts\"),\n        timeout: cdk.Duration.seconds(30),\n        memorySize: 512,\n        vpc: vpc,\n        vpcSubnets: { subnetType: ec2.SubnetType.PUBLIC },\n        securityGroups: [ecsTaskSecurityGroup],\n        allowPublicSubnet: true,\n        environment: {\n          EFS_FILE_SYSTEM_ID: fileSystem.fileSystemId,\n        },\n        role: lambdaRole,\n        filesystem: lambda.FileSystem.fromEfsAccessPoint(\n          lambdaAccessPoint,\n          \"/mnt/efs\"\n        ),\n        bundling: {\n          minify: true,\n          sourceMap: true,\n          format: OutputFormat.CJS,\n        },\n      }\n    );\n\n    // ============================================\n    // Collect subnet IDs for environment variables\n    // ============================================\n    const publicSubnetIds = vpc.publicSubnets\n      .map((subnet) => subnet.subnetId)\n      .join(\",\");\n\n    // ============================================\n    // GitHub Webhook Handler Lambda\n    // ============================================\n    const githubWebhookHandler = new NodejsFunction(\n      this,\n      \"HyperpGithubWebhookHandler\",\n      {\n        functionName: \"hyperp-github-webhook-handler\",\n        runtime: lambda.Runtime.NODEJS_20_X,\n        handler: \"handler\",\n        entry: getLambdaPath(\"github-webhook-handler/index.ts\"),\n        timeout: cdk.Duration.minutes(5),\n        memorySize: 1024,\n        environment: {\n          DYNAMODB_TABLE_NAME: table.tableName,\n          S3_BUCKET_NAME: artifactsBucket.bucketName,\n          ECS_CLUSTER_ARN: cluster.clusterArn,\n          ECS_CLUSTER_NAME: cluster.clusterName,\n          ECR_REPOSITORY_URI: ecrRepository.repositoryUri,\n          ECR_REPOSITORY_NAME: ecrRepository.repositoryName,\n          ECS_TASK_EXECUTION_ROLE_ARN: ecsTaskExecutionRole.roleArn,\n          ECS_TASK_ROLE_ARN: ecsTaskRole.roleArn,\n          EFS_FILE_SYSTEM_ID: fileSystem.fileSystemId,\n          ECS_TASK_SECURITY_GROUP_ID: ecsTaskSecurityGroup.securityGroupId,\n          PUBLIC_SUBNET_IDS: publicSubnetIds,\n          EFS_CONTROLLER_LAMBDA_ARN: efsControllerLambda.functionArn,\n          GITHUB_APP_ID: githubAppId,\n          GITHUB_APP_WEBHOOK_SECRET: githubAppWebhookSecret,\n          AWS_ACCOUNT_ID: this.account,\n        },\n        role: lambdaRole,\n        bundling: {\n          minify: true,\n          sourceMap: true,\n          format: OutputFormat.CJS,\n        },\n      }\n    );\n\n    // Create Function URL for GitHub Webhook Handler\n    const webhookFunctionUrl = githubWebhookHandler.addFunctionUrl({\n      authType: lambda.FunctionUrlAuthType.NONE, // Public access for GitHub webhooks\n      cors: {\n        allowedOrigins: [\"*\"],\n        allowedMethods: [lambda.HttpMethod.POST],\n        allowedHeaders: [\"*\"],\n      },\n    });\n\n    // ============================================\n    // Task State Change Handler Lambda\n    // ============================================\n    const taskStateChangeHandler = new NodejsFunction(\n      this,\n      \"HyperpTaskStateChangeHandler\",\n      {\n        functionName: \"hyperp-task-state-change-handler\",\n        runtime: lambda.Runtime.NODEJS_20_X,\n        handler: \"handler\",\n        entry: getLambdaPath(\"task-state-change-handler/index.ts\"),\n        timeout: cdk.Duration.minutes(2),\n        memorySize: 512,\n        environment: {\n          DYNAMODB_TABLE_NAME: table.tableName,\n          ECS_CLUSTER_ARN: cluster.clusterArn,\n          ECS_CLUSTER_NAME: cluster.clusterName,\n          ECR_REPOSITORY_URI: ecrRepository.repositoryUri,\n          ECS_TASK_EXECUTION_ROLE_ARN: ecsTaskExecutionRole.roleArn,\n          ECS_TASK_ROLE_ARN: ecsTaskRole.roleArn,\n          EFS_FILE_SYSTEM_ID: fileSystem.fileSystemId,\n          ECS_TASK_SECURITY_GROUP_ID: ecsTaskSecurityGroup.securityGroupId,\n          PUBLIC_SUBNET_IDS: publicSubnetIds,\n          EFS_CONTROLLER_LAMBDA_ARN: efsControllerLambda.functionArn,\n          AWS_ACCOUNT_ID: this.account,\n          FARGATE_VCPU_PRICE_PER_HOUR: \"0.04048\",\n          FARGATE_GB_PRICE_PER_HOUR: \"0.004445\",\n          S3_BUCKET_NAME: artifactsBucket.bucketName,\n          GITHUB_APP_ID: githubAppId,\n          DOWNLOADABLE_CREATOR_IMAGE:\n            \"public.ecr.aws/b6g9t8f1/downloadable-creator:latest\",\n          USAGE_CALCULATOR_IMAGE:\n            \"public.ecr.aws/b6g9t8f1/usage-calculator:latest\",\n        },\n        role: lambdaRole,\n        bundling: {\n          minify: true,\n          sourceMap: true,\n          format: OutputFormat.CJS,\n        },\n      }\n    );\n\n    // ============================================\n    // EventBridge Rule for ECS Task State Changes\n    // ============================================\n    const ecsTaskStateChangeRule = new events.Rule(\n      this,\n      \"EcsTaskStateChangeRule\",\n      {\n        eventPattern: {\n          source: [\"aws.ecs\"],\n          detailType: [\"ECS Task State Change\"],\n          detail: {\n            clusterArn: [cluster.clusterArn],\n            lastStatus: [\"STOPPED\"],\n          },\n        },\n      }\n    );\n\n    ecsTaskStateChangeRule.addTarget(\n      new targets.LambdaFunction(taskStateChangeHandler)\n    );\n\n    // ============================================\n    // Outputs\n    // ============================================\n    new cdk.CfnOutput(this, \"VpcId\", {\n      value: vpc.vpcId,\n      description: \"VPC ID\",\n    });\n\n    new cdk.CfnOutput(this, \"ClusterArn\", {\n      value: cluster.clusterArn,\n      description: \"ECS Cluster ARN\",\n    });\n\n    new cdk.CfnOutput(this, \"EcrRepositoryUri\", {\n      value: ecrRepository.repositoryUri,\n      description: \"ECR Repository URI\",\n    });\n\n    new cdk.CfnOutput(this, \"DynamoDbTableName\", {\n      value: table.tableName,\n      description: \"DynamoDB Table Name\",\n    });\n\n    new cdk.CfnOutput(this, \"S3BucketName\", {\n      value: artifactsBucket.bucketName,\n      description: \"S3 Artifacts Bucket Name\",\n    });\n\n    new cdk.CfnOutput(this, \"EfsFileSystemId\", {\n      value: fileSystem.fileSystemId,\n      description: \"EFS File System ID\",\n    });\n\n    new cdk.CfnOutput(this, \"GithubWebhookHandlerArn\", {\n      value: githubWebhookHandler.functionArn,\n      description: \"GitHub Webhook Handler Lambda ARN\",\n    });\n\n    new cdk.CfnOutput(this, \"GithubWebhookHandlerUrl\", {\n      value: webhookFunctionUrl.url,\n      description: \"GitHub Webhook Handler Function URL\",\n    });\n\n    new cdk.CfnOutput(this, \"TaskStateChangeHandlerArn\", {\n      value: taskStateChangeHandler.functionArn,\n      description: \"Task State Change Handler Lambda ARN\",\n    });\n\n    new cdk.CfnOutput(this, \"PublicSubnetIds\", {\n      value: publicSubnetIds,\n      description: \"Public Subnet IDs\",\n    });\n\n    // ============================================\n    // REST API Lambda for CLI\n    // ============================================\n    const cliRestApiLambda = new NodejsFunction(this, \"HyperpCliRestApi\", {\n      functionName: \"hyperp-cli-rest-api\",\n      runtime: lambda.Runtime.NODEJS_20_X,\n      handler: \"handler\",\n      entry: getLambdaPath(\"cli-rest-api/index.ts\"),\n      timeout: cdk.Duration.seconds(30),\n      memorySize: 512,\n      environment: {\n        DYNAMODB_TABLE_NAME: table.tableName,\n        ECS_CLUSTER_NAME: cluster.clusterName,\n        EFS_FILE_SYSTEM_ID: fileSystem.fileSystemId,\n        PUBLIC_SUBNET_IDS: vpc.publicSubnets.map((s) => s.subnetId).join(\",\"),\n        ECS_TASK_SECURITY_GROUP_ID: ecsTaskSecurityGroup.securityGroupId,\n        ECS_TASK_EXECUTION_ROLE_ARN: ecsTaskExecutionRole.roleArn,\n        ECS_TASK_ROLE_ARN: ecsTaskRole.roleArn,\n        S3_BUCKET_NAME: artifactsBucket.bucketName,\n        DOWNLOADABLE_CREATOR_IMAGE:\n          \"public.ecr.aws/b6g9t8f1/downloadable-creator:latest\",\n        GITHUB_APP_ID: githubAppId,\n        GITHUB_APP_WEBHOOK_SECRET: githubAppWebhookSecret,\n      },\n      role: lambdaRole,\n      bundling: {\n        minify: true,\n        sourceMap: true,\n        format: OutputFormat.CJS,\n      },\n    });\n\n    // Create Function URL\n    const functionUrl = cliRestApiLambda.addFunctionUrl({\n      authType: lambda.FunctionUrlAuthType.NONE, // Public access (can be secured later)\n      cors: {\n        allowedOrigins: [\"*\"],\n        allowedMethods: [lambda.HttpMethod.GET, lambda.HttpMethod.POST],\n        allowedHeaders: [\"*\"],\n      },\n    });\n\n    new cdk.CfnOutput(this, \"CliRestApiUrl\", {\n      value: functionUrl.url,\n      description: \"CLI REST API Function URL\",\n    });\n  }\n}\n"]}
@@ -14,11 +14,42 @@ import * as events from "aws-cdk-lib/aws-events";
14
14
  import * as targets from "aws-cdk-lib/aws-events-targets";
15
15
  import * as s3deploy from "aws-cdk-lib/aws-s3-deployment";
16
16
  import * as path from "path";
17
+ import * as fs from "fs";
17
18
 
18
19
  export class HyperpStack extends cdk.Stack {
19
20
  constructor(scope: Construct, id: string, props?: cdk.StackProps) {
20
21
  super(scope, id, props);
21
22
 
23
+ // Resolve Lambda entry paths - handle both development and npm package contexts
24
+ // When compiled, __dirname is dist/lib/, so we need to go up to package root
25
+ // When in npm package, we need to find the package root
26
+ const getLambdaPath = (lambdaPath: string): string => {
27
+ // Try relative to __dirname first (for npm package - go up 2 levels from dist/lib/)
28
+ const npmPath = path.join(__dirname, "../../lambda", lambdaPath);
29
+ if (fs.existsSync(npmPath)) {
30
+ return npmPath;
31
+ }
32
+ // Try relative to __dirname with one less level (for development - from lib/)
33
+ const devPath = path.join(__dirname, "../lambda", lambdaPath);
34
+ if (fs.existsSync(devPath)) {
35
+ return devPath;
36
+ }
37
+ // Fallback: try to find package root by looking for package.json
38
+ let currentDir = __dirname;
39
+ while (currentDir !== path.dirname(currentDir)) {
40
+ const packageJsonPath = path.join(currentDir, "package.json");
41
+ if (fs.existsSync(packageJsonPath)) {
42
+ const lambdaFullPath = path.join(currentDir, "lambda", lambdaPath);
43
+ if (fs.existsSync(lambdaFullPath)) {
44
+ return lambdaFullPath;
45
+ }
46
+ }
47
+ currentDir = path.dirname(currentDir);
48
+ }
49
+ // Last resort: return the path relative to __dirname (will fail if file doesn't exist)
50
+ return path.join(__dirname, "../lambda", lambdaPath);
51
+ };
52
+
22
53
  // Get GitHub App configuration from CDK context
23
54
  const githubAppId =
24
55
  this.node.tryGetContext("githubAppId") || process.env.GITHUB_APP_ID;
@@ -371,7 +402,7 @@ export class HyperpStack extends cdk.Stack {
371
402
  functionName: "hyperp-efs-controller",
372
403
  runtime: lambda.Runtime.NODEJS_20_X,
373
404
  handler: "handler",
374
- entry: path.join(__dirname, "../lambda/efs-controller/index.ts"),
405
+ entry: getLambdaPath("efs-controller/index.ts"),
375
406
  timeout: cdk.Duration.seconds(30),
376
407
  memorySize: 512,
377
408
  vpc: vpc,
@@ -411,10 +442,7 @@ export class HyperpStack extends cdk.Stack {
411
442
  functionName: "hyperp-github-webhook-handler",
412
443
  runtime: lambda.Runtime.NODEJS_20_X,
413
444
  handler: "handler",
414
- entry: path.join(
415
- __dirname,
416
- "../lambda/github-webhook-handler/index.ts"
417
- ),
445
+ entry: getLambdaPath("github-webhook-handler/index.ts"),
418
446
  timeout: cdk.Duration.minutes(5),
419
447
  memorySize: 1024,
420
448
  environment: {
@@ -463,10 +491,7 @@ export class HyperpStack extends cdk.Stack {
463
491
  functionName: "hyperp-task-state-change-handler",
464
492
  runtime: lambda.Runtime.NODEJS_20_X,
465
493
  handler: "handler",
466
- entry: path.join(
467
- __dirname,
468
- "../lambda/task-state-change-handler/index.ts"
469
- ),
494
+ entry: getLambdaPath("task-state-change-handler/index.ts"),
470
495
  timeout: cdk.Duration.minutes(2),
471
496
  memorySize: 512,
472
497
  environment: {
@@ -581,7 +606,7 @@ export class HyperpStack extends cdk.Stack {
581
606
  functionName: "hyperp-cli-rest-api",
582
607
  runtime: lambda.Runtime.NODEJS_20_X,
583
608
  handler: "handler",
584
- entry: path.join(__dirname, "../lambda/cli-rest-api/index.ts"),
609
+ entry: getLambdaPath("cli-rest-api/index.ts"),
585
610
  timeout: cdk.Duration.seconds(30),
586
611
  memorySize: 512,
587
612
  environment: {
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "hyperp",
3
- "version": "1.0.1",
3
+ "version": "1.0.3",
4
4
  "description": "Fully automated servereless compute platform on AWS",
5
5
  "main": "index.js",
6
6
  "bin": {
@@ -52,6 +52,7 @@
52
52
  "@aws-sdk/client-s3": "^3.700.0",
53
53
  "@aws-sdk/lib-dynamodb": "^3.700.0",
54
54
  "@aws-sdk/s3-request-presigner": "^3.700.0",
55
+ "@types/node": "22.19.3",
55
56
  "aws-cdk-lib": "^2.170.0",
56
57
  "axios": "^1.7.9",
57
58
  "commander": "^12.0.0",
@@ -64,7 +65,6 @@
64
65
  "devDependencies": {
65
66
  "@types/aws-lambda": "^8.10.145",
66
67
  "@types/jsonwebtoken": "^9.0.7",
67
- "@types/node": "22.19.3",
68
68
  "aws-cdk": "^2.170.0",
69
69
  "typescript": "^5.7.2"
70
70
  }