Skip to content

Commit

Permalink
feat(deploy): add the terraform deployment
Browse files Browse the repository at this point in the history
  • Loading branch information
nfroidure committed Jul 16, 2024
1 parent 3b2e869 commit 522151c
Show file tree
Hide file tree
Showing 8 changed files with 427 additions and 22 deletions.
51 changes: 51 additions & 0 deletions packages/whook-example/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -92,6 +92,57 @@ Debug `knifecycle` internals (dependency injection issues):
DEBUG=knifecycle npm run dev
```

## Deploying with Google Cloud Functions

Create a project and save its credentials to `.credentials.json`.

Then install Terraform:
```sh
wget https://releases.hashicorp.com/terraform/0.12.24/terraform_0.12.24_linux_amd64.zip
mkdir .bin
unzip -d .bin terraform_0.12.24_linux_amd64.zip
rm terraform_0.12.24_linux_amd64.zip
```

Then initialize the Terraform configuration:
```sh
.bin/terraform init ./terraform;
```

Create a new workspace:
```sh
.bin/terraform workspace new staging
```

Build the functions:
```sh
NODE_ENV=staging npm run build
```

Build the Whook commands Terraform depends on:
```sh
npm run compile
```

Plan the deployment:
```sh
.bin/terraform plan -var="project_id=my-project-1664" \
-out=terraform.plan terraform
```

Apply changes:
```sh
# parallelism may be necessary to avoid hitting
# timeouts with a slow connection
.bin/terraform apply -parallelism=1 terraform.plan
```

Finally retrieve the API URL and enjoy!
```sh
.bin/terraform -var="project_id=my-project-1664" \
output api_url
```

## Testing the GCP Functions

```sh
Expand Down
3 changes: 2 additions & 1 deletion packages/whook-example/src/cli.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -17,8 +17,9 @@ describe('commands should work', () => {
",
"stdout": "
# Provided by "@whook/example": 1 commands
# Provided by "@whook/example": 2 commands
- printEnv: A command printing every env values
- terraformValues: A command printing functions informations for Terraform
# Provided by "@whook/whook": 8 commands
Expand Down
228 changes: 228 additions & 0 deletions packages/whook-example/src/commands/terraformValues.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,228 @@
import { extra, autoService } from 'knifecycle';
import { readArgs } from '@whook/whook';
import { getOpenAPIOperations } from '@whook/http-router';
import { YError } from 'yerror';
import { exec } from 'child_process';
import crypto from 'crypto';
import yaml from 'js-yaml';
import type { ExecException } from 'child_process';
import type { LogService } from 'common-services';
import type {
WhookCommandArgs,
WhookCommandDefinition,
WhookAPIHandlerDefinition,
} from '@whook/whook';
import type { OpenAPIV3_1 } from 'openapi-types';

export const definition: WhookCommandDefinition = {
description: 'A command printing functions informations for Terraform',
example: `whook terraformValues --type paths`,
arguments: {
type: 'object',
additionalProperties: false,
required: ['type'],
properties: {
type: {
description: 'Type of values to return',
type: 'string',
enum: ['globals', 'paths', 'functions', 'function'],
},
pretty: {
description: 'Pretty print JSON values',
type: 'boolean',
},
functionName: {
description: 'Name of the function',
type: 'string',
},
pathsIndex: {
description: 'Index of the paths to retrieve',
type: 'number',
},
functionType: {
description: 'Types of the functions to return',
type: 'string',
},
},
},
};

export default extra(definition, autoService(initTerraformValuesCommand));

async function initTerraformValuesCommand({
API,
BASE_PATH,
log,
args,
execAsync = _execAsync,
}: {
API: OpenAPIV3_1.Document;
BASE_PATH: string;
log: LogService;
args: WhookCommandArgs;
execAsync: typeof _execAsync;
}) {
return async () => {
const {
namedArguments: { type, pretty, functionName, functionType },
} = readArgs<{
type: string;
pretty: boolean;
functionName: string;
functionType: string;
}>(definition.arguments, args);
const operations =
getOpenAPIOperations<WhookAPIHandlerDefinition['operation']['x-whook']>(
API,
);
const configurations = operations.map((operation) => {
const whookConfiguration = (operation['x-whook'] || {
type: 'http',
}) as WhookAPIHandlerDefinition['operation']['x-whook'];
const configuration = {
type: 'http',
timeout: '10',
memory: '128',
description: operation.summary || '',
enabled: 'true',
sourceOperationId: operation.operationId,
suffix: '',
...Object.keys(whookConfiguration || {}).reduce(
(accConfigurations, key) => ({
...accConfigurations,
[key]: (
(
whookConfiguration as NonNullable<
WhookAPIHandlerDefinition['operation']['x-whook']
>
)[key] as string
).toString(),
}),
{},
),
};
const qualifiedOperationId =
(configuration.sourceOperationId || operation.operationId) +
(configuration.suffix || '');

return {
qualifiedOperationId,
method: operation.method.toUpperCase(),
path: operation.path,
...configuration,
};
});

if (type === 'globals') {
const commitHash = await execAsync(`git rev-parse HEAD`);
const commitMessage = (
await execAsync(`git rev-list --format=%B --max-count=1 HEAD`)
).split('\n')[1];
const openapi2 = yaml.safeDump({
swagger: '2.0',
info: {
title: API.info.title,
description: API.info.description,
version: API.info.version,
},
host: '${infos_host}',
basePath: BASE_PATH,
schemes: ['https'],
produces: ['application/json'],
paths: configurations.reduce((accPaths, configuration) => {
const operation = operations.find(
({ operationId }) =>
operationId === configuration.sourceOperationId,
);

return {
...accPaths,
[configuration.path]: {
...(accPaths[configuration.path] || {}),
[configuration.method.toLowerCase()]: {
summary: configuration.description || '',
operationId: configuration.qualifiedOperationId,
...((operation?.parameters || []).length
? {
parameters: (
operation?.parameters as OpenAPIV3_1.ParameterObject[]
).map(({ in: theIn, name, required }) => ({
in: theIn,
name,
type: 'string',
required: required || false,
})),
}
: undefined),
'x-google-backend': {
address: `\${function_${configuration.qualifiedOperationId}}`,
},
responses: {
'200': { description: 'x', schema: { type: 'string' } },
},
},
},
};
}, {}),
});
const openapiHash = crypto
.createHash('md5')
.update(JSON.stringify(API))
.digest('hex');
const infos = {
commitHash,
commitMessage,
openapi2,
openapiHash,
};
log('info', JSON.stringify(infos));
return;
}

if (type === 'functions') {
const functions = configurations
.filter((configuration) =>
functionType ? configuration.type === functionType : true,
)
.reduce(
(accLambdas, configuration) => ({
...accLambdas,
[configuration.qualifiedOperationId]:
configuration.qualifiedOperationId,
}),
{},
);

log('info', `${JSON.stringify(functions, null, pretty ? 2 : 0)}`);
return;
}

if (!functionName) {
throw new YError('E_FUNCTION_NAME_REQUIRED');
}

const functionConfiguration = configurations.find(
({ qualifiedOperationId }) => qualifiedOperationId === functionName,
);

log(
'info',
`${JSON.stringify(functionConfiguration, null, pretty ? 2 : 0)}`,
);
};
}

async function _execAsync(command: string): Promise<string> {
return await new Promise((resolve, reject) => {
exec(
command,
(err: ExecException | null, stdout: string, stderr: string) => {
if (err) {
reject(YError.wrap(err, 'E_EXEC_FAILURE', stderr));
return;
}
resolve(stdout.trim());
},
);
});
}
40 changes: 20 additions & 20 deletions packages/whook-example/src/index.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -333,64 +333,64 @@ describe('runServer', () => {
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-example/src/index.test.ts:78:59)",
],
[
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-gcp-functions/dist/handlers/optionsWithCORS.js'",
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-example/src/index.test.ts:78:59)",
],
[
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-gcp-functions/dist/services/BUFFER_LIMIT.js'",
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-example/src/index.test.ts:78:59)",
],
[
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-gcp-functions/dist/services/DECODERS.js'",
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-example/src/index.test.ts:78:59)",
],
[
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-gcp-functions/dist/services/DEFAULT_ERROR_CODE.js'",
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-example/src/index.test.ts:78:59)",
],
[
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-gcp-functions/dist/services/ENCODERS.js'",
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-example/src/index.test.ts:78:59)",
],
[
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-gcp-functions/dist/services/HTTP_SERVER_OPTIONS.js'",
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-example/src/index.test.ts:78:59)",
],
[
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-gcp-functions/dist/services/IGNORED_FILES_PREFIXES.js'",
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-example/src/index.test.ts:78:59)",
],
[
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-gcp-functions/dist/services/IGNORED_FILES_SUFFIXES.js'",
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-example/src/index.test.ts:78:59)",
],
[
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-gcp-functions/dist/services/JWT_SECRET_ENV_NAME.js'",
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-example/src/index.test.ts:78:59)",
],
[
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-gcp-functions/dist/services/MAX_CLEAR_RATIO.js'",
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-example/src/index.test.ts:78:59)",
],
[
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-gcp-functions/dist/services/PARSERS.js'",
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-example/src/index.test.ts:78:59)",
],
[
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-gcp-functions/dist/services/PROCESS_NAME.js'",
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-example/src/index.test.ts:78:59)",
],
[
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-gcp-functions/dist/services/REDUCED_FILES_SUFFIXES.js'",
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-example/src/index.test.ts:78:59)",
],
[
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-gcp-functions/dist/services/SHIELD_CHAR.js'",
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-example/src/index.test.ts:78:59)",
],
[
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-gcp-functions/dist/services/SIGNALS.js'",
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-example/src/index.test.ts:78:59)",
],
[
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-gcp-functions/dist/services/STRINGIFYERS.js'",
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-example/src/index.test.ts:78:59)",
],
[
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-gcp-functions/dist/services/TIMEOUT.js'",
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-example/src/index.test.ts:78:59)",
],
[
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-gcp-functions/dist/services/uniqueId.js'",
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-example/src/index.test.ts:78:59)",
],
[
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-gcp-functions/dist/wrappers/wrapHandlerWithAuthorization.js'",
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-example/src/index.test.ts:78:59)",
],
[
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-gcp-functions/dist/wrappers/wrapHandlerWithCORS.js'",
"Error: ENOENT: no such file or directory, access 'file:///home/whoiam/projects/whook/packages/whook-example/src/index.test.ts:78:59)",
],
[
"⌛ - Delay service initialized.",
Expand Down
3 changes: 2 additions & 1 deletion packages/whook-example/src/whook.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -72,7 +72,8 @@ Here we export a custom handler definition type in order
T &
WhookAPIOperationGCPFunctionConfig &
WhookAPIOperationSwaggerConfig &
WhookAPIOperationCORSConfig & { // TODO: Add those properties to Whook GCP Functions?
WhookAPIOperationCORSConfig & {
// TODO: Add those properties to Whook GCP Functions?
private?: boolean;
memory?: number;
timeout?: number;
Expand Down
Loading

0 comments on commit 522151c

Please sign in to comment.