@openhi/platform 0.0.0 → 0.0.2

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.
Files changed (40) hide show
  1. package/lib/{openhi.d.ts → index.d.mts} +42 -16
  2. package/lib/index.d.ts +195 -2
  3. package/lib/index.js +825 -17
  4. package/lib/index.js.map +1 -0
  5. package/lib/index.mjs +818 -0
  6. package/lib/index.mjs.map +1 -0
  7. package/package.json +33 -24
  8. package/lib/openhi.js +0 -108
  9. package/lib/service.d.ts +0 -36
  10. package/lib/service.js +0 -203
  11. package/lib/templates/generate-templates.d.ts +0 -5
  12. package/lib/templates/generate-templates.js +0 -45
  13. package/lib/templates/service-template.d.ts +0 -33
  14. package/lib/templates/service-template.js +0 -42
  15. package/lib/templates/src/README.md.d.ts +0 -5
  16. package/lib/templates/src/README.md.js +0 -19
  17. package/lib/templates/src/app-test.d.ts +0 -5
  18. package/lib/templates/src/app-test.js +0 -61
  19. package/lib/templates/src/app.d.ts +0 -5
  20. package/lib/templates/src/app.js +0 -27
  21. package/lib/templates/src/config.d.ts +0 -5
  22. package/lib/templates/src/config.js +0 -23
  23. package/lib/templates/src/data/README.md.d.ts +0 -5
  24. package/lib/templates/src/data/README.md.js +0 -19
  25. package/lib/templates/src/data/models/README.md.d.ts +0 -5
  26. package/lib/templates/src/data/models/README.md.js +0 -19
  27. package/lib/templates/src/infrastructure/README.md.d.ts +0 -5
  28. package/lib/templates/src/infrastructure/README.md.js +0 -19
  29. package/lib/templates/src/integrations/README.md.d.ts +0 -5
  30. package/lib/templates/src/integrations/README.md.js +0 -19
  31. package/lib/templates/src/main.d.ts +0 -5
  32. package/lib/templates/src/main.js +0 -15
  33. package/lib/templates/src/workflows/README.md.d.ts +0 -5
  34. package/lib/templates/src/workflows/README.md.js +0 -19
  35. package/lib/workflows/aws-teardown-workflow.d.ts +0 -13
  36. package/lib/workflows/aws-teardown-workflow.js +0 -222
  37. package/lib/workflows/build-dev-workflow.d.ts +0 -12
  38. package/lib/workflows/build-dev-workflow.js +0 -48
  39. package/lib/workflows/build-stage-workflow.d.ts +0 -12
  40. package/lib/workflows/build-stage-workflow.js +0 -60
@@ -1,222 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.AwsTeardownWorkflow = void 0;
4
- const configulator_1 = require("@codedrifters/configulator");
5
- const config_1 = require("@openhi/config");
6
- const projen_1 = require("projen");
7
- const github_1 = require("projen/lib/github");
8
- const workflows_model_1 = require("projen/lib/github/workflows-model");
9
- class AwsTeardownWorkflow extends projen_1.Component {
10
- constructor(rootProject, options) {
11
- super(rootProject);
12
- this.rootProject = rootProject;
13
- const { openhi } = options;
14
- /***************************************************************************
15
- *
16
- * Root project check
17
- *
18
- * Detect the root project and ensure it's of type MonorepoProject.
19
- *
20
- **************************************************************************/
21
- if (!(rootProject instanceof configulator_1.MonorepoProject)) {
22
- throw new Error("AwsTeardownWorkflow requires the root project to be a MonorepoProject");
23
- }
24
- //this.rootProject = project.root;
25
- /***************************************************************************
26
- *
27
- * GitHub Check
28
- *
29
- * Make sure github config is active in the project. This is to ensure all
30
- * workflows will be output properly during synth.
31
- *
32
- **************************************************************************/
33
- const github = github_1.GitHub.of(this.rootProject);
34
- if (!github) {
35
- throw new Error("AwsTeardownWorkflow requires a GitHub component in the root project");
36
- }
37
- /***************************************************************************
38
- *
39
- * Workflow Destroy Targets
40
- *
41
- * Find all the CI targets for the Auth and Core projects in the DEV stage
42
- * only. One job per (account, region) teardowns all orphan stacks (Auth,
43
- * Core, data) in that account/region.
44
- *
45
- **************************************************************************/
46
- const devTargetsFor = (service) => configulator_1.AwsDeploymentConfig.of(service.project)?.awsDeploymentTargets.filter((target) => target.awsStageType === config_1.OPEN_HI_STAGE.DEV && target.ciDeployment) ?? [];
47
- const coreTargets = devTargetsFor(openhi.core);
48
- const authTargets = devTargetsFor(openhi.auth);
49
- const byAccountRegion = new Map();
50
- [...authTargets, ...coreTargets].forEach((t) => {
51
- const key = `${t.account}-${t.region}`;
52
- if (!byAccountRegion.has(key))
53
- byAccountRegion.set(key, t);
54
- });
55
- const awsDestructionTargets = Array.from(byAccountRegion.values());
56
- /***************************************************************************
57
- *
58
- * Create Workflow
59
- *
60
- * Make a new workflow for AWS stack teardown. This runs each night and is
61
- * also triggered each time a branch is deleted.
62
- *
63
- **************************************************************************/
64
- const workflow = new github_1.GithubWorkflow(github, "teardown-dev");
65
- workflow.on({
66
- workflowDispatch: {},
67
- schedule: [
68
- {
69
- cron: "32 6 * * *", // Every Sunday at 6:32 AM UTC
70
- },
71
- ],
72
- delete: {
73
- branches: ["feature/*", "feat/*", "fix/*"],
74
- },
75
- /* for debugging
76
- push: {
77
- branches: ["feature/*"],
78
- },
79
- */
80
- });
81
- awsDestructionTargets.forEach((target) => {
82
- const { awsStageType, awsEnvironmentType, account, region, ciDeploymentConfig, } = target;
83
- const { roleArn } = ciDeploymentConfig ?? {};
84
- workflow.addJob(`teardown-${account}-${region}`.toLowerCase(), {
85
- name: `Teardown Stacks in ${target.account}/${target.region}`,
86
- //if: "github.event.ref_type == 'branch'",
87
- runsOn: ["ubuntu-latest"],
88
- permissions: {
89
- contents: workflows_model_1.JobPermission.READ,
90
- idToken: workflows_model_1.JobPermission.WRITE,
91
- },
92
- env: {
93
- REPO: "${{ github.repository }}",
94
- REGIONS: [region].join(" "),
95
- },
96
- steps: [
97
- /**
98
- * Configure AWS creds.
99
- */
100
- {
101
- name: `AWS Creds ${awsStageType}/${awsEnvironmentType}/${account}/${region}`,
102
- uses: "aws-actions/configure-aws-credentials@v4",
103
- with: {
104
- "role-to-assume": roleArn,
105
- "aws-region": region,
106
- "role-duration-seconds": 900, // 15 minutes
107
- },
108
- },
109
- /**
110
- * Fetch all branch names in the repo
111
- */
112
- {
113
- name: "Fetch All Branches",
114
- id: "fetch_branches",
115
- uses: "actions/github-script@v7",
116
- with: {
117
- script: [
118
- "const all = await github.paginate(github.rest.repos.listBranches, {",
119
- " owner: context.repo.owner,",
120
- " repo: context.repo.repo,",
121
- " per_page: 100",
122
- "});",
123
- "const names = all.map(b => b.name);",
124
- "console.log(`Found branches: ${names}`);",
125
- 'core.setOutput("json", JSON.stringify(names));',
126
- ].join("\n"),
127
- },
128
- },
129
- /**
130
- * Save branches to a file
131
- */
132
- {
133
- name: "Save Branches to File",
134
- run: [
135
- 'echo "Saving branches to file"',
136
- "echo '${{ steps.fetch_branches.outputs.json }}' | jq -r '.[]' | sort -u > branches.txt",
137
- 'echo "Branches:"',
138
- "cat branches.txt",
139
- ].join("\n"),
140
- },
141
- /**
142
- * Find all stacks tagged with a stage of dev. for this repo. return
143
- * tag and resource arn.
144
- */
145
- {
146
- name: "Find Stacks by Tag",
147
- id: "find_stacks",
148
- run: [
149
- "set -euo pipefail",
150
- ": > candidates.txt # columns: arn region branchTag",
151
- "# Build tag filters",
152
- 'TAG_FILTERS=( "Key=openhi:repo-name,Values=$REPO" )',
153
- `TAG_FILTERS+=( "Key=openhi:stage-type,Values=${config_1.OPEN_HI_STAGE.DEV}" )`,
154
- "for r in $REGIONS; do",
155
- ` echo "Scanning region: $r"`,
156
- " aws resourcegroupstaggingapi get-resources \\",
157
- ' --region "$r" \\',
158
- ' --resource-type-filters "cloudformation:stack" \\',
159
- ' --tag-filters "${TAG_FILTERS[@]}" \\',
160
- ` | jq -r --arg r "$r" '`,
161
- " .ResourceTagMappingList[]",
162
- " | . as $res",
163
- ' | ($res.Tags[] | select(.Key=="openhi:branch-name") | .Value) as $branch',
164
- ' | [$res.ResourceARN, $r, ($branch // "")]',
165
- " | @tsv",
166
- " ' >> candidates.txt",
167
- "done",
168
- "echo 'Tagged stacks:'",
169
- `(echo -e "ARN\\tREGION\\tBRANCH"; cat candidates.txt) | column -t -s $'\\t'`,
170
- ].join("\n"),
171
- },
172
- /**
173
- * Determine which stacks are orphans that no longer have a matching branch.
174
- * Save those to a file for the next step.
175
- */
176
- {
177
- name: "Determine Orphan Stacks (No Matching Branch)",
178
- run: [
179
- "set -euo pipefail",
180
- ": > orphans.txt # arn region branch",
181
- "while IFS=$'\\t' read -r arn region branch; do",
182
- ' [ -z "$arn" ] && continue',
183
- ' if [ -z "$branch" ]; then',
184
- " # If no Branch tag, treat as not-a-preview; skip (or flip to delete if you want)",
185
- " continue",
186
- " fi",
187
- ' if ! grep -Fxq "$branch" branches.txt; then',
188
- ' echo -e "$arn\\t$region\\t$branch" >> orphans.txt',
189
- " fi",
190
- "done < candidates.txt",
191
- "",
192
- "if [ -s orphans.txt ]; then",
193
- ' echo "Orphan stacks (no matching branch):"',
194
- " (echo -e \"ARN\\tREGION\\tBRANCH\"; cat orphans.txt) | column -t -s $'\\t'",
195
- "else",
196
- ' echo "No orphan stacks found."',
197
- "fi",
198
- ].join("\n"),
199
- },
200
- /**
201
- * Delete orphan stacks.
202
- */
203
- {
204
- name: "Delete Orphan Stacks",
205
- if: "hashFiles('orphans.txt') != ''",
206
- run: [
207
- "set -euo pipefail",
208
- "while IFS=$'\\t' read -r arn region branch; do",
209
- ' [ -z "$arn" ] && continue',
210
- " stack_name=$(cut -d'/' -f2 <<<\"$arn\")",
211
- ' echo "Deleting $stack_name (branch=$branch) in $region"',
212
- ' aws cloudformation delete-stack --region "$region" --stack-name "$stack_name" || true',
213
- "done < orphans.txt",
214
- ].join("\n"),
215
- },
216
- ],
217
- });
218
- });
219
- }
220
- }
221
- exports.AwsTeardownWorkflow = AwsTeardownWorkflow;
222
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"aws-teardown-workflow.js","sourceRoot":"","sources":["../../src/workflows/aws-teardown-workflow.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  AwsDeploymentConfig,\n  MonorepoProject,\n} from \"@codedrifters/configulator\";\nimport { OPEN_HI_STAGE } from \"@openhi/config\";\nimport { Component } from \"projen\";\nimport { GitHub, GithubWorkflow } from \"projen/lib/github\";\nimport { JobPermission } from \"projen/lib/github/workflows-model\";\nimport { OpenHi } from \"../openhi\";\nimport type { OpenHiService } from \"../service\";\n\nexport interface AwsTeardownWorkflowOptions {\n  /**\n   * openhi project\n   */\n  readonly openhi: OpenHi;\n}\n\nexport class AwsTeardownWorkflow extends Component {\n  constructor(\n    public rootProject: MonorepoProject,\n    options: AwsTeardownWorkflowOptions,\n  ) {\n    super(rootProject);\n\n    const { openhi } = options;\n\n    /***************************************************************************\n     *\n     * Root project check\n     *\n     * Detect the root project and ensure it's of type MonorepoProject.\n     *\n     **************************************************************************/\n\n    if (!(rootProject instanceof MonorepoProject)) {\n      throw new Error(\n        \"AwsTeardownWorkflow requires the root project to be a MonorepoProject\",\n      );\n    }\n\n    //this.rootProject = project.root;\n\n    /***************************************************************************\n     *\n     * GitHub Check\n     *\n     * Make sure github config is active in the project. This is to ensure all\n     * workflows will be output properly during synth.\n     *\n     **************************************************************************/\n\n    const github = GitHub.of(this.rootProject);\n\n    if (!github) {\n      throw new Error(\n        \"AwsTeardownWorkflow requires a GitHub component in the root project\",\n      );\n    }\n\n    /***************************************************************************\n     *\n     * Workflow Destroy Targets\n     *\n     * Find all the CI targets for the Auth and Core projects in the DEV stage\n     * only. One job per (account, region) teardowns all orphan stacks (Auth,\n     * Core, data) in that account/region.\n     *\n     **************************************************************************/\n\n    const devTargetsFor = (service: OpenHiService) =>\n      AwsDeploymentConfig.of(service.project)?.awsDeploymentTargets.filter(\n        (target) =>\n          target.awsStageType === OPEN_HI_STAGE.DEV && target.ciDeployment,\n      ) ?? [];\n\n    const coreTargets = devTargetsFor(openhi.core);\n    const authTargets = devTargetsFor(openhi.auth);\n    const byAccountRegion = new Map<string, (typeof coreTargets)[0]>();\n    [...authTargets, ...coreTargets].forEach((t) => {\n      const key = `${t.account}-${t.region}`;\n      if (!byAccountRegion.has(key)) byAccountRegion.set(key, t);\n    });\n    const awsDestructionTargets = Array.from(byAccountRegion.values());\n\n    /***************************************************************************\n     *\n     * Create Workflow\n     *\n     * Make a new workflow for AWS stack teardown. This runs each night and is\n     * also triggered each time a branch is deleted.\n     *\n     **************************************************************************/\n\n    const workflow = new GithubWorkflow(github, \"teardown-dev\");\n    workflow.on({\n      workflowDispatch: {},\n      schedule: [\n        {\n          cron: \"32 6 * * *\", // Every Sunday at 6:32 AM UTC\n        },\n      ],\n      delete: {\n        branches: [\"feature/*\", \"feat/*\", \"fix/*\"],\n      },\n      /* for debugging \n      push: {\n        branches: [\"feature/*\"],\n      },\n      */\n    });\n\n    awsDestructionTargets.forEach((target) => {\n      const {\n        awsStageType,\n        awsEnvironmentType,\n        account,\n        region,\n        ciDeploymentConfig,\n      } = target;\n      const { roleArn } = ciDeploymentConfig ?? {};\n\n      workflow.addJob(`teardown-${account}-${region}`.toLowerCase(), {\n        name: `Teardown Stacks in ${target.account}/${target.region}`,\n        //if: \"github.event.ref_type == 'branch'\",\n        runsOn: [\"ubuntu-latest\"],\n        permissions: {\n          contents: JobPermission.READ,\n          idToken: JobPermission.WRITE,\n        },\n        env: {\n          REPO: \"${{ github.repository }}\",\n          REGIONS: [region].join(\" \"),\n        },\n        steps: [\n          /**\n           * Configure AWS creds.\n           */\n          {\n            name: `AWS Creds ${awsStageType}/${awsEnvironmentType}/${account}/${region}`,\n            uses: \"aws-actions/configure-aws-credentials@v4\",\n            with: {\n              \"role-to-assume\": roleArn,\n              \"aws-region\": region,\n              \"role-duration-seconds\": 900, // 15 minutes\n            },\n          },\n\n          /**\n           * Fetch all branch names in the repo\n           */\n          {\n            name: \"Fetch All Branches\",\n            id: \"fetch_branches\",\n            uses: \"actions/github-script@v7\",\n            with: {\n              script: [\n                \"const all = await github.paginate(github.rest.repos.listBranches, {\",\n                \"  owner: context.repo.owner,\",\n                \"  repo: context.repo.repo,\",\n                \"  per_page: 100\",\n                \"});\",\n                \"const names = all.map(b => b.name);\",\n                \"console.log(`Found branches: ${names}`);\",\n                'core.setOutput(\"json\", JSON.stringify(names));',\n              ].join(\"\\n\"),\n            },\n          },\n\n          /**\n           * Save branches to a file\n           */\n          {\n            name: \"Save Branches to File\",\n            run: [\n              'echo \"Saving branches to file\"',\n              \"echo '${{ steps.fetch_branches.outputs.json }}' | jq -r '.[]' | sort -u > branches.txt\",\n              'echo \"Branches:\"',\n              \"cat branches.txt\",\n            ].join(\"\\n\"),\n          },\n\n          /**\n           * Find all stacks tagged with a stage of dev. for this repo. return\n           * tag and resource arn.\n           */\n          {\n            name: \"Find Stacks by Tag\",\n            id: \"find_stacks\",\n            run: [\n              \"set -euo pipefail\",\n              \": > candidates.txt  # columns: arn region branchTag\",\n\n              \"# Build tag filters\",\n              'TAG_FILTERS=( \"Key=openhi:repo-name,Values=$REPO\" )',\n              `TAG_FILTERS+=( \"Key=openhi:stage-type,Values=${OPEN_HI_STAGE.DEV}\" )`,\n\n              \"for r in $REGIONS; do\",\n              `  echo \"Scanning region: $r\"`,\n              \"  aws resourcegroupstaggingapi get-resources \\\\\",\n              '    --region \"$r\" \\\\',\n              '    --resource-type-filters \"cloudformation:stack\" \\\\',\n              '    --tag-filters \"${TAG_FILTERS[@]}\" \\\\',\n              ` | jq -r --arg r \"$r\" '`,\n              \"    .ResourceTagMappingList[]\",\n              \"    | . as $res\",\n              '    | ($res.Tags[] | select(.Key==\"openhi:branch-name\") | .Value) as $branch',\n              '    | [$res.ResourceARN, $r, ($branch // \"\")]',\n              \"    | @tsv\",\n              \"    ' >> candidates.txt\",\n              \"done\",\n\n              \"echo 'Tagged stacks:'\",\n              `(echo -e \"ARN\\\\tREGION\\\\tBRANCH\"; cat candidates.txt) | column -t -s $'\\\\t'`,\n            ].join(\"\\n\"),\n          },\n\n          /**\n           * Determine which stacks are orphans that no longer have a matching branch.\n           * Save those to a file for the next step.\n           */\n          {\n            name: \"Determine Orphan Stacks (No Matching Branch)\",\n            run: [\n              \"set -euo pipefail\",\n              \": > orphans.txt  # arn region branch\",\n              \"while IFS=$'\\\\t' read -r arn region branch; do\",\n              '  [ -z \"$arn\" ] && continue',\n              '  if [ -z \"$branch\" ]; then',\n              \"    # If no Branch tag, treat as not-a-preview; skip (or flip to delete if you want)\",\n              \"    continue\",\n              \"  fi\",\n              '  if ! grep -Fxq \"$branch\" branches.txt; then',\n              '    echo -e \"$arn\\\\t$region\\\\t$branch\" >> orphans.txt',\n              \"  fi\",\n              \"done < candidates.txt\",\n              \"\",\n              \"if [ -s orphans.txt ]; then\",\n              '  echo \"Orphan stacks (no matching branch):\"',\n              \"  (echo -e \\\"ARN\\\\tREGION\\\\tBRANCH\\\"; cat orphans.txt) | column -t -s $'\\\\t'\",\n              \"else\",\n              '  echo \"No orphan stacks found.\"',\n              \"fi\",\n            ].join(\"\\n\"),\n          },\n\n          /**\n           * Delete orphan stacks.\n           */\n          {\n            name: \"Delete Orphan Stacks\",\n            if: \"hashFiles('orphans.txt') != ''\",\n            run: [\n              \"set -euo pipefail\",\n              \"while IFS=$'\\\\t' read -r arn region branch; do\",\n              '  [ -z \"$arn\" ] && continue',\n              \"  stack_name=$(cut -d'/' -f2 <<<\\\"$arn\\\")\",\n              '  echo \"Deleting $stack_name (branch=$branch) in $region\"',\n              '  aws cloudformation delete-stack --region \"$region\" --stack-name \"$stack_name\" || true',\n              \"done < orphans.txt\",\n            ].join(\"\\n\"),\n          },\n        ],\n      });\n    });\n  }\n}\n"]}
@@ -1,12 +0,0 @@
1
- import { MonorepoProject } from "@codedrifters/configulator";
2
- import { Component } from "projen";
3
- import { OpenHi } from "../openhi";
4
- export interface BuildDevelopmentWorkflowOptions {
5
- /**
6
- * openhi project
7
- */
8
- readonly openhi: OpenHi;
9
- }
10
- export declare class BuildDevelopmentWorkflow extends Component {
11
- constructor(parent: MonorepoProject, options: BuildDevelopmentWorkflowOptions);
12
- }
@@ -1,48 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.BuildDevelopmentWorkflow = void 0;
4
- const configulator_1 = require("@codedrifters/configulator");
5
- const projen_1 = require("projen");
6
- class BuildDevelopmentWorkflow extends projen_1.Component {
7
- constructor(parent, options) {
8
- super(parent, "dev-workflow");
9
- const { openhi } = options;
10
- /**
11
- * Global Service Deploy
12
- *
13
- * Deploy the Global service first to configure DNS
14
- */
15
- const { awsDeploymentTargets: globalTargets } = new configulator_1.AwsDeployWorkflow(openhi.global.project, {
16
- buildWorkflow: parent.buildWorkflow,
17
- });
18
- /**
19
- * Auth Service Deploy
20
- *
21
- * Deploy the Auth service first (Cognito); Core consumes auth via SSM.
22
- */
23
- const { awsDeploymentTargets: authTargets } = new configulator_1.AwsDeployWorkflow(openhi.auth.project, {
24
- buildWorkflow: parent.buildWorkflow,
25
- deployAfterTargets: [...globalTargets],
26
- });
27
- /**
28
- * Data Service Deploy
29
- *
30
- * Deploy the Data service (DynamoDB FHIR store, S3) after Global.
31
- */
32
- const { awsDeploymentTargets: dataTargets } = new configulator_1.AwsDeployWorkflow(openhi.data.project, {
33
- buildWorkflow: parent.buildWorkflow,
34
- deployAfterTargets: [...globalTargets, ...authTargets],
35
- });
36
- /**
37
- * Rest API Service Deploy
38
- *
39
- * Deploy the Rest API service after Data (and Global).
40
- */
41
- new configulator_1.AwsDeployWorkflow(openhi.restApi.project, {
42
- buildWorkflow: parent.buildWorkflow,
43
- deployAfterTargets: [...globalTargets, ...authTargets, ...dataTargets],
44
- });
45
- }
46
- }
47
- exports.BuildDevelopmentWorkflow = BuildDevelopmentWorkflow;
48
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYnVpbGQtZGV2LXdvcmtmbG93LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3dvcmtmbG93cy9idWlsZC1kZXYtd29ya2Zsb3cudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsNkRBQWdGO0FBQ2hGLG1DQUFtQztBQVVuQyxNQUFhLHdCQUF5QixTQUFRLGtCQUFTO0lBQ3JELFlBQ0UsTUFBdUIsRUFDdkIsT0FBd0M7UUFFeEMsS0FBSyxDQUFDLE1BQU0sRUFBRSxjQUFjLENBQUMsQ0FBQztRQUU5QixNQUFNLEVBQUUsTUFBTSxFQUFFLEdBQUcsT0FBTyxDQUFDO1FBRTNCOzs7O1dBSUc7UUFDSCxNQUFNLEVBQUUsb0JBQW9CLEVBQUUsYUFBYSxFQUFFLEdBQUcsSUFBSSxnQ0FBaUIsQ0FDbkUsTUFBTSxDQUFDLE1BQU0sQ0FBQyxPQUFPLEVBQ3JCO1lBQ0UsYUFBYSxFQUFFLE1BQU0sQ0FBQyxhQUFhO1NBQ3BDLENBQ0YsQ0FBQztRQUVGOzs7O1dBSUc7UUFDSCxNQUFNLEVBQUUsb0JBQW9CLEVBQUUsV0FBVyxFQUFFLEdBQUcsSUFBSSxnQ0FBaUIsQ0FDakUsTUFBTSxDQUFDLElBQUksQ0FBQyxPQUFPLEVBQ25CO1lBQ0UsYUFBYSxFQUFFLE1BQU0sQ0FBQyxhQUFhO1lBQ25DLGtCQUFrQixFQUFFLENBQUMsR0FBRyxhQUFhLENBQUM7U0FDdkMsQ0FDRixDQUFDO1FBRUY7Ozs7V0FJRztRQUNILE1BQU0sRUFBRSxvQkFBb0IsRUFBRSxXQUFXLEVBQUUsR0FBRyxJQUFJLGdDQUFpQixDQUNqRSxNQUFNLENBQUMsSUFBSSxDQUFDLE9BQU8sRUFDbkI7WUFDRSxhQUFhLEVBQUUsTUFBTSxDQUFDLGFBQWE7WUFDbkMsa0JBQWtCLEVBQUUsQ0FBQyxHQUFHLGFBQWEsRUFBRSxHQUFHLFdBQVcsQ0FBQztTQUN2RCxDQUNGLENBQUM7UUFFRjs7OztXQUlHO1FBQ0gsSUFBSSxnQ0FBaUIsQ0FBQyxNQUFNLENBQUMsT0FBTyxDQUFDLE9BQU8sRUFBRTtZQUM1QyxhQUFhLEVBQUUsTUFBTSxDQUFDLGFBQWE7WUFDbkMsa0JBQWtCLEVBQUUsQ0FBQyxHQUFHLGFBQWEsRUFBRSxHQUFHLFdBQVcsRUFBRSxHQUFHLFdBQVcsQ0FBQztTQUN2RSxDQUFDLENBQUM7SUFDTCxDQUFDO0NBQ0Y7QUF6REQsNERBeURDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQXdzRGVwbG95V29ya2Zsb3csIE1vbm9yZXBvUHJvamVjdCB9IGZyb20gXCJAY29kZWRyaWZ0ZXJzL2NvbmZpZ3VsYXRvclwiO1xuaW1wb3J0IHsgQ29tcG9uZW50IH0gZnJvbSBcInByb2plblwiO1xuaW1wb3J0IHsgT3BlbkhpIH0gZnJvbSBcIi4uL29wZW5oaVwiO1xuXG5leHBvcnQgaW50ZXJmYWNlIEJ1aWxkRGV2ZWxvcG1lbnRXb3JrZmxvd09wdGlvbnMge1xuICAvKipcbiAgICogb3BlbmhpIHByb2plY3RcbiAgICovXG4gIHJlYWRvbmx5IG9wZW5oaTogT3BlbkhpO1xufVxuXG5leHBvcnQgY2xhc3MgQnVpbGREZXZlbG9wbWVudFdvcmtmbG93IGV4dGVuZHMgQ29tcG9uZW50IHtcbiAgY29uc3RydWN0b3IoXG4gICAgcGFyZW50OiBNb25vcmVwb1Byb2plY3QsXG4gICAgb3B0aW9uczogQnVpbGREZXZlbG9wbWVudFdvcmtmbG93T3B0aW9ucyxcbiAgKSB7XG4gICAgc3VwZXIocGFyZW50LCBcImRldi13b3JrZmxvd1wiKTtcblxuICAgIGNvbnN0IHsgb3BlbmhpIH0gPSBvcHRpb25zO1xuXG4gICAgLyoqXG4gICAgICogR2xvYmFsIFNlcnZpY2UgRGVwbG95XG4gICAgICpcbiAgICAgKiBEZXBsb3kgdGhlIEdsb2JhbCBzZXJ2aWNlIGZpcnN0IHRvIGNvbmZpZ3VyZSBETlNcbiAgICAgKi9cbiAgICBjb25zdCB7IGF3c0RlcGxveW1lbnRUYXJnZXRzOiBnbG9iYWxUYXJnZXRzIH0gPSBuZXcgQXdzRGVwbG95V29ya2Zsb3coXG4gICAgICBvcGVuaGkuZ2xvYmFsLnByb2plY3QsXG4gICAgICB7XG4gICAgICAgIGJ1aWxkV29ya2Zsb3c6IHBhcmVudC5idWlsZFdvcmtmbG93LFxuICAgICAgfSxcbiAgICApO1xuXG4gICAgLyoqXG4gICAgICogQXV0aCBTZXJ2aWNlIERlcGxveVxuICAgICAqXG4gICAgICogRGVwbG95IHRoZSBBdXRoIHNlcnZpY2UgZmlyc3QgKENvZ25pdG8pOyBDb3JlIGNvbnN1bWVzIGF1dGggdmlhIFNTTS5cbiAgICAgKi9cbiAgICBjb25zdCB7IGF3c0RlcGxveW1lbnRUYXJnZXRzOiBhdXRoVGFyZ2V0cyB9ID0gbmV3IEF3c0RlcGxveVdvcmtmbG93KFxuICAgICAgb3BlbmhpLmF1dGgucHJvamVjdCxcbiAgICAgIHtcbiAgICAgICAgYnVpbGRXb3JrZmxvdzogcGFyZW50LmJ1aWxkV29ya2Zsb3csXG4gICAgICAgIGRlcGxveUFmdGVyVGFyZ2V0czogWy4uLmdsb2JhbFRhcmdldHNdLFxuICAgICAgfSxcbiAgICApO1xuXG4gICAgLyoqXG4gICAgICogRGF0YSBTZXJ2aWNlIERlcGxveVxuICAgICAqXG4gICAgICogRGVwbG95IHRoZSBEYXRhIHNlcnZpY2UgKER5bmFtb0RCIEZISVIgc3RvcmUsIFMzKSBhZnRlciBHbG9iYWwuXG4gICAgICovXG4gICAgY29uc3QgeyBhd3NEZXBsb3ltZW50VGFyZ2V0czogZGF0YVRhcmdldHMgfSA9IG5ldyBBd3NEZXBsb3lXb3JrZmxvdyhcbiAgICAgIG9wZW5oaS5kYXRhLnByb2plY3QsXG4gICAgICB7XG4gICAgICAgIGJ1aWxkV29ya2Zsb3c6IHBhcmVudC5idWlsZFdvcmtmbG93LFxuICAgICAgICBkZXBsb3lBZnRlclRhcmdldHM6IFsuLi5nbG9iYWxUYXJnZXRzLCAuLi5hdXRoVGFyZ2V0c10sXG4gICAgICB9LFxuICAgICk7XG5cbiAgICAvKipcbiAgICAgKiBSZXN0IEFQSSBTZXJ2aWNlIERlcGxveVxuICAgICAqXG4gICAgICogRGVwbG95IHRoZSBSZXN0IEFQSSBzZXJ2aWNlIGFmdGVyIERhdGEgKGFuZCBHbG9iYWwpLlxuICAgICAqL1xuICAgIG5ldyBBd3NEZXBsb3lXb3JrZmxvdyhvcGVuaGkucmVzdEFwaS5wcm9qZWN0LCB7XG4gICAgICBidWlsZFdvcmtmbG93OiBwYXJlbnQuYnVpbGRXb3JrZmxvdyxcbiAgICAgIGRlcGxveUFmdGVyVGFyZ2V0czogWy4uLmdsb2JhbFRhcmdldHMsIC4uLmF1dGhUYXJnZXRzLCAuLi5kYXRhVGFyZ2V0c10sXG4gICAgfSk7XG4gIH1cbn1cbiJdfQ==
@@ -1,12 +0,0 @@
1
- import { MonorepoProject } from "@codedrifters/configulator";
2
- import { Component } from "projen";
3
- import { OpenHi } from "../openhi";
4
- export interface BuildStageWorkflowOptions {
5
- /**
6
- * openhi project
7
- */
8
- readonly openhi: OpenHi;
9
- }
10
- export declare class BuildStageWorkflow extends Component {
11
- constructor(parent: MonorepoProject, options: BuildStageWorkflowOptions);
12
- }
@@ -1,60 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.BuildStageWorkflow = void 0;
4
- const configulator_1 = require("@codedrifters/configulator");
5
- const config_1 = require("@openhi/config");
6
- const projen_1 = require("projen");
7
- class BuildStageWorkflow extends projen_1.Component {
8
- constructor(parent, options) {
9
- super(parent, "stage-workflow");
10
- const { openhi } = options;
11
- /**
12
- * Global Service Deploy
13
- *
14
- * Deploy the Global service first to configure DNS
15
- */
16
- const { awsDeploymentTargets: globalTargets, buildWorkflow } = new configulator_1.AwsDeployWorkflow(openhi.global.project, {
17
- awsStageType: config_1.OPEN_HI_STAGE.STAGE,
18
- buildWorkflowOptions: {
19
- name: "deploy-stage",
20
- workflowTriggers: {
21
- push: {
22
- branches: ["main"],
23
- },
24
- workflowDispatch: {},
25
- },
26
- },
27
- });
28
- /**
29
- * Auth Service Deploy
30
- *
31
- * Deploy the Auth service first (Cognito); Core consumes auth via SSM.
32
- * Auth creates the stage build workflow; Core and data services add to it.
33
- */
34
- const { awsDeploymentTargets: authTargets } = new configulator_1.AwsDeployWorkflow(openhi.auth.project, {
35
- awsStageType: config_1.OPEN_HI_STAGE.STAGE,
36
- buildWorkflow,
37
- deployAfterTargets: [...globalTargets],
38
- });
39
- /**
40
- * Data Service Deploy
41
- *
42
- * Deploy the Data service (DynamoDB FHIR store, S3) after Global.
43
- */
44
- const { awsDeploymentTargets: dataTargets } = new configulator_1.AwsDeployWorkflow(openhi.data.project, {
45
- awsStageType: config_1.OPEN_HI_STAGE.STAGE,
46
- buildWorkflow,
47
- deployAfterTargets: [...globalTargets, ...authTargets],
48
- });
49
- /**
50
- * Rest API Service Deploy
51
- */
52
- new configulator_1.AwsDeployWorkflow(openhi.restApi.project, {
53
- awsStageType: config_1.OPEN_HI_STAGE.STAGE,
54
- buildWorkflow,
55
- deployAfterTargets: [...globalTargets, ...authTargets, ...dataTargets],
56
- });
57
- }
58
- }
59
- exports.BuildStageWorkflow = BuildStageWorkflow;
60
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYnVpbGQtc3RhZ2Utd29ya2Zsb3cuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvd29ya2Zsb3dzL2J1aWxkLXN0YWdlLXdvcmtmbG93LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLDZEQUFnRjtBQUNoRiwyQ0FBK0M7QUFDL0MsbUNBQW1DO0FBVW5DLE1BQWEsa0JBQW1CLFNBQVEsa0JBQVM7SUFDL0MsWUFBWSxNQUF1QixFQUFFLE9BQWtDO1FBQ3JFLEtBQUssQ0FBQyxNQUFNLEVBQUUsZ0JBQWdCLENBQUMsQ0FBQztRQUVoQyxNQUFNLEVBQUUsTUFBTSxFQUFFLEdBQUcsT0FBTyxDQUFDO1FBRTNCOzs7O1dBSUc7UUFDSCxNQUFNLEVBQUUsb0JBQW9CLEVBQUUsYUFBYSxFQUFFLGFBQWEsRUFBRSxHQUMxRCxJQUFJLGdDQUFpQixDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsT0FBTyxFQUFFO1lBQzNDLFlBQVksRUFBRSxzQkFBYSxDQUFDLEtBQUs7WUFDakMsb0JBQW9CLEVBQUU7Z0JBQ3BCLElBQUksRUFBRSxjQUFjO2dCQUNwQixnQkFBZ0IsRUFBRTtvQkFDaEIsSUFBSSxFQUFFO3dCQUNKLFFBQVEsRUFBRSxDQUFDLE1BQU0sQ0FBQztxQkFDbkI7b0JBQ0QsZ0JBQWdCLEVBQUUsRUFBRTtpQkFDckI7YUFDRjtTQUNGLENBQUMsQ0FBQztRQUVMOzs7OztXQUtHO1FBQ0gsTUFBTSxFQUFFLG9CQUFvQixFQUFFLFdBQVcsRUFBRSxHQUFHLElBQUksZ0NBQWlCLENBQ2pFLE1BQU0sQ0FBQyxJQUFJLENBQUMsT0FBTyxFQUNuQjtZQUNFLFlBQVksRUFBRSxzQkFBYSxDQUFDLEtBQUs7WUFDakMsYUFBYTtZQUNiLGtCQUFrQixFQUFFLENBQUMsR0FBRyxhQUFhLENBQUM7U0FDdkMsQ0FDRixDQUFDO1FBRUY7Ozs7V0FJRztRQUNILE1BQU0sRUFBRSxvQkFBb0IsRUFBRSxXQUFXLEVBQUUsR0FBRyxJQUFJLGdDQUFpQixDQUNqRSxNQUFNLENBQUMsSUFBSSxDQUFDLE9BQU8sRUFDbkI7WUFDRSxZQUFZLEVBQUUsc0JBQWEsQ0FBQyxLQUFLO1lBQ2pDLGFBQWE7WUFDYixrQkFBa0IsRUFBRSxDQUFDLEdBQUcsYUFBYSxFQUFFLEdBQUcsV0FBVyxDQUFDO1NBQ3ZELENBQ0YsQ0FBQztRQUVGOztXQUVHO1FBQ0gsSUFBSSxnQ0FBaUIsQ0FBQyxNQUFNLENBQUMsT0FBTyxDQUFDLE9BQU8sRUFBRTtZQUM1QyxZQUFZLEVBQUUsc0JBQWEsQ0FBQyxLQUFLO1lBQ2pDLGFBQWE7WUFDYixrQkFBa0IsRUFBRSxDQUFDLEdBQUcsYUFBYSxFQUFFLEdBQUcsV0FBVyxFQUFFLEdBQUcsV0FBVyxDQUFDO1NBQ3ZFLENBQUMsQ0FBQztJQUNMLENBQUM7Q0FDRjtBQS9ERCxnREErREMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBBd3NEZXBsb3lXb3JrZmxvdywgTW9ub3JlcG9Qcm9qZWN0IH0gZnJvbSBcIkBjb2RlZHJpZnRlcnMvY29uZmlndWxhdG9yXCI7XG5pbXBvcnQgeyBPUEVOX0hJX1NUQUdFIH0gZnJvbSBcIkBvcGVuaGkvY29uZmlnXCI7XG5pbXBvcnQgeyBDb21wb25lbnQgfSBmcm9tIFwicHJvamVuXCI7XG5pbXBvcnQgeyBPcGVuSGkgfSBmcm9tIFwiLi4vb3BlbmhpXCI7XG5cbmV4cG9ydCBpbnRlcmZhY2UgQnVpbGRTdGFnZVdvcmtmbG93T3B0aW9ucyB7XG4gIC8qKlxuICAgKiBvcGVuaGkgcHJvamVjdFxuICAgKi9cbiAgcmVhZG9ubHkgb3BlbmhpOiBPcGVuSGk7XG59XG5cbmV4cG9ydCBjbGFzcyBCdWlsZFN0YWdlV29ya2Zsb3cgZXh0ZW5kcyBDb21wb25lbnQge1xuICBjb25zdHJ1Y3RvcihwYXJlbnQ6IE1vbm9yZXBvUHJvamVjdCwgb3B0aW9uczogQnVpbGRTdGFnZVdvcmtmbG93T3B0aW9ucykge1xuICAgIHN1cGVyKHBhcmVudCwgXCJzdGFnZS13b3JrZmxvd1wiKTtcblxuICAgIGNvbnN0IHsgb3BlbmhpIH0gPSBvcHRpb25zO1xuXG4gICAgLyoqXG4gICAgICogR2xvYmFsIFNlcnZpY2UgRGVwbG95XG4gICAgICpcbiAgICAgKiBEZXBsb3kgdGhlIEdsb2JhbCBzZXJ2aWNlIGZpcnN0IHRvIGNvbmZpZ3VyZSBETlNcbiAgICAgKi9cbiAgICBjb25zdCB7IGF3c0RlcGxveW1lbnRUYXJnZXRzOiBnbG9iYWxUYXJnZXRzLCBidWlsZFdvcmtmbG93IH0gPVxuICAgICAgbmV3IEF3c0RlcGxveVdvcmtmbG93KG9wZW5oaS5nbG9iYWwucHJvamVjdCwge1xuICAgICAgICBhd3NTdGFnZVR5cGU6IE9QRU5fSElfU1RBR0UuU1RBR0UsXG4gICAgICAgIGJ1aWxkV29ya2Zsb3dPcHRpb25zOiB7XG4gICAgICAgICAgbmFtZTogXCJkZXBsb3ktc3RhZ2VcIixcbiAgICAgICAgICB3b3JrZmxvd1RyaWdnZXJzOiB7XG4gICAgICAgICAgICBwdXNoOiB7XG4gICAgICAgICAgICAgIGJyYW5jaGVzOiBbXCJtYWluXCJdLFxuICAgICAgICAgICAgfSxcbiAgICAgICAgICAgIHdvcmtmbG93RGlzcGF0Y2g6IHt9LFxuICAgICAgICAgIH0sXG4gICAgICAgIH0sXG4gICAgICB9KTtcblxuICAgIC8qKlxuICAgICAqIEF1dGggU2VydmljZSBEZXBsb3lcbiAgICAgKlxuICAgICAqIERlcGxveSB0aGUgQXV0aCBzZXJ2aWNlIGZpcnN0IChDb2duaXRvKTsgQ29yZSBjb25zdW1lcyBhdXRoIHZpYSBTU00uXG4gICAgICogQXV0aCBjcmVhdGVzIHRoZSBzdGFnZSBidWlsZCB3b3JrZmxvdzsgQ29yZSBhbmQgZGF0YSBzZXJ2aWNlcyBhZGQgdG8gaXQuXG4gICAgICovXG4gICAgY29uc3QgeyBhd3NEZXBsb3ltZW50VGFyZ2V0czogYXV0aFRhcmdldHMgfSA9IG5ldyBBd3NEZXBsb3lXb3JrZmxvdyhcbiAgICAgIG9wZW5oaS5hdXRoLnByb2plY3QsXG4gICAgICB7XG4gICAgICAgIGF3c1N0YWdlVHlwZTogT1BFTl9ISV9TVEFHRS5TVEFHRSxcbiAgICAgICAgYnVpbGRXb3JrZmxvdyxcbiAgICAgICAgZGVwbG95QWZ0ZXJUYXJnZXRzOiBbLi4uZ2xvYmFsVGFyZ2V0c10sXG4gICAgICB9LFxuICAgICk7XG5cbiAgICAvKipcbiAgICAgKiBEYXRhIFNlcnZpY2UgRGVwbG95XG4gICAgICpcbiAgICAgKiBEZXBsb3kgdGhlIERhdGEgc2VydmljZSAoRHluYW1vREIgRkhJUiBzdG9yZSwgUzMpIGFmdGVyIEdsb2JhbC5cbiAgICAgKi9cbiAgICBjb25zdCB7IGF3c0RlcGxveW1lbnRUYXJnZXRzOiBkYXRhVGFyZ2V0cyB9ID0gbmV3IEF3c0RlcGxveVdvcmtmbG93KFxuICAgICAgb3BlbmhpLmRhdGEucHJvamVjdCxcbiAgICAgIHtcbiAgICAgICAgYXdzU3RhZ2VUeXBlOiBPUEVOX0hJX1NUQUdFLlNUQUdFLFxuICAgICAgICBidWlsZFdvcmtmbG93LFxuICAgICAgICBkZXBsb3lBZnRlclRhcmdldHM6IFsuLi5nbG9iYWxUYXJnZXRzLCAuLi5hdXRoVGFyZ2V0c10sXG4gICAgICB9LFxuICAgICk7XG5cbiAgICAvKipcbiAgICAgKiBSZXN0IEFQSSBTZXJ2aWNlIERlcGxveVxuICAgICAqL1xuICAgIG5ldyBBd3NEZXBsb3lXb3JrZmxvdyhvcGVuaGkucmVzdEFwaS5wcm9qZWN0LCB7XG4gICAgICBhd3NTdGFnZVR5cGU6IE9QRU5fSElfU1RBR0UuU1RBR0UsXG4gICAgICBidWlsZFdvcmtmbG93LFxuICAgICAgZGVwbG95QWZ0ZXJUYXJnZXRzOiBbLi4uZ2xvYmFsVGFyZ2V0cywgLi4uYXV0aFRhcmdldHMsIC4uLmRhdGFUYXJnZXRzXSxcbiAgICB9KTtcbiAgfVxufVxuIl19