From c1085184b9de765ddf8c0120cc5a19e2e77854a0 Mon Sep 17 00:00:00 2001 From: Quan Nguyen Ba Date: Fri, 6 Dec 2024 11:11:50 +0700 Subject: [PATCH 1/2] isolate docker environment for each project --- packages/cli/src/actions/new.action.ts | 20 ++++++++++++++++++++ packages/cli/templates/.env.local | 4 +++- packages/cli/templates/package.json | 8 ++++---- 3 files changed, 27 insertions(+), 5 deletions(-) diff --git a/packages/cli/src/actions/new.action.ts b/packages/cli/src/actions/new.action.ts index c405e3f..736a0c5 100644 --- a/packages/cli/src/actions/new.action.ts +++ b/packages/cli/src/actions/new.action.ts @@ -63,6 +63,12 @@ export default async function newAction( const infraGitignore = path.join(destDir, 'infra/gitignore') copyFileSync(infraGitignore, path.join(destDir, 'infra/.gitignore')) unlinkSync(infraGitignore) + // replace project_name in .env.local + updateEnvLocal( + path.join(destDir, '.env.local'), + '%%projectName%%', + projectName, + ) // cp .env.local .env copyFileSync(path.join(destDir, '.env.local'), path.join(destDir, '.env')) @@ -139,10 +145,24 @@ function getPackageVersion(packageName: string, isLatest = false): string[] { return versions } +function updateEnvLocal( + envPath: string, + searchValue: string, + replaceValue: string, +): void { + const envLocalContent = readFileSync(envPath, 'utf8') + const newEnvLocalContent = envLocalContent.replaceAll( + searchValue, + replaceValue, + ) + writeFileSync(envPath, newEnvLocalContent) +} + export let exportsForTesting = { usePackageVersion, getPackageVersion, isLatestCli, + updateEnvLocal, } if (process.env.NODE_ENV !== 'test') { exportsForTesting = undefined diff --git a/packages/cli/templates/.env.local b/packages/cli/templates/.env.local index e0e6101..6ad904a 100644 --- a/packages/cli/templates/.env.local +++ b/packages/cli/templates/.env.local @@ -5,7 +5,9 @@ AWS_DEFAULT_REGION=ap-northeast-1 # running environment NODE_ENV=local # local, dev, stg, prod # name of application -APP_NAME=demo +APP_NAME=%%projectName%% +# name of docker compose +COMPOSE_PROJECT_NAME=%%projectName%% # set log levels LOG_LEVEL=verbose # debug, verbose, info, warn, error, fatal # disable event route for API GW integration diff --git a/packages/cli/templates/package.json b/packages/cli/templates/package.json index a5184c3..9473e5a 100644 --- a/packages/cli/templates/package.json +++ b/packages/cli/templates/package.json @@ -16,11 +16,11 @@ "start:repl": "nest start --watch --entryFile repl", "start:prod": "node dist/main", "offline:docker:build": "run-script-os", - "offline:docker:build:default": "cd infra-local && docker compose up --build --remove-orphans", - "offline:docker:build:win32": "powershell -Command \"Set-Location infra-local; docker compose up --build --remove-orphans\"", + "offline:docker:build:default": "ln -f .env $PWD/infra-local/.env && cd infra-local && docker compose up --build --remove-orphans", + "offline:docker:build:win32": "powershell -Command \"Copy-Item '.env' -Destination './infra-local/.env'; Set-Location infra-local; docker compose up --build --remove-orphans\"", "offline:docker": "run-script-os", - "offline:docker:default": "cd infra-local && mkdir -p docker-data/.cognito && cp -r cognito-local/db docker-data/.cognito && docker compose up --remove-orphans", - "offline:docker:win32": "powershell -Command \"Set-Location infra-local; New-Item -ItemType Directory -Force -Path 'docker-data/.cognito'; Copy-Item -Recurse -Force 'cognito-local/db' 'docker-data/.cognito'; docker compose up --remove-orphans\"", + "offline:docker:default": "ln -f .env $PWD/infra-local/.env && cd infra-local && mkdir -p docker-data/.cognito && cp -r cognito-local/db docker-data/.cognito && docker compose up --remove-orphans", + "offline:docker:win32": "powershell -Command \"Copy-Item '.env' -Destination './infra-local/.env'; Set-Location infra-local; New-Item -ItemType Directory -Force -Path 'docker-data/.cognito'; Copy-Item -Recurse -Force 'cognito-local/db' 'docker-data/.cognito'; docker compose up --remove-orphans\"", "offline:sls": "run-script-os", "offline:sls:default": "/bin/bash ./infra-local/scripts/resources.sh && /bin/bash ./infra-local/scripts/trigger_ddb_stream.sh & ln -f .env $PWD/infra-local/.env && cd infra-local && NODE_ENV=development AWS_ACCESS_KEY_ID=DUMMYIDEXAMPLE AWS_SECRET_ACCESS_KEY=DUMMYEXAMPLEKEY SLS_DEBUG=* serverless offline start", "offline:sls:win32": "npm run resources:win32 && concurrently \"npm run trigger-ddb:win32\" \"npm run sls:win32\"", From d18f37d728a5b213722bad361509b3671c877142 Mon Sep 17 00:00:00 2001 From: Quan Nguyen Ba Date: Fri, 6 Dec 2024 11:13:14 +0700 Subject: [PATCH 2/2] unit test for update env local function --- .../new.action.update-env-local.spec.ts | 49 +++++++++++++++++++ 1 file changed, 49 insertions(+) create mode 100644 packages/cli/src/actions/new.action.update-env-local.spec.ts diff --git a/packages/cli/src/actions/new.action.update-env-local.spec.ts b/packages/cli/src/actions/new.action.update-env-local.spec.ts new file mode 100644 index 0000000..9a8bab7 --- /dev/null +++ b/packages/cli/src/actions/new.action.update-env-local.spec.ts @@ -0,0 +1,49 @@ +import { readFileSync, writeFileSync } from 'fs' + +import { exportsForTesting } from './new.action' + +const { updateEnvLocal } = exportsForTesting + +jest.mock('fs') + +describe('updateEnvLocal', () => { + const mockEnvContent = ` + # name of application + APP_NAME=%%projectName%% + # name of docker compose + COMPOSE_PROJECT_NAME=%%projectName%% + ` + + const envPath = './.env.local' + + beforeEach(() => { + jest.clearAllMocks() + }) + + it('should update the specified value in the .env.local file', () => { + const searchValue = '%%projectName%%' + const replaceValue = 'new-project-name' + + ;(readFileSync as jest.Mock).mockReturnValue(mockEnvContent) + + updateEnvLocal(envPath, searchValue, replaceValue) + + expect(readFileSync).toHaveBeenCalledWith(envPath, 'utf8') + + const expectedContent = mockEnvContent.replaceAll(searchValue, replaceValue) + expect(writeFileSync).toHaveBeenCalledWith(envPath, expectedContent) + }) + + it('should not change the file content if searchValue is not found', () => { + const searchValue = 'non-existent-value' + const replaceValue = 'new-project-name' + + ;(readFileSync as jest.Mock).mockReturnValue(mockEnvContent) + + updateEnvLocal(envPath, searchValue, replaceValue) + + expect(readFileSync).toHaveBeenCalledWith(envPath, 'utf8') + + expect(writeFileSync).toHaveBeenCalledWith(envPath, mockEnvContent) + }) +})