Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: support custom codec files in host repositories #870

Merged
merged 10 commits into from
Aug 8, 2024
150 changes: 120 additions & 30 deletions packages/openapi-generator/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ API specification into an OpenAPI specification.

## Install

```
```shell
npm install --save-dev @api-ts/openapi-generator
```

Expand All @@ -15,7 +15,7 @@ The **openapi-generator** assumes the io-ts-http `apiSpec` is exported in the to
of the Typescript file passed as an input parameter. The OpenAPI specification will be
written to stdout.

```
```shell
ARGUMENTS:
<file> - API route definition file

Expand All @@ -35,32 +35,122 @@ For example:
npx openapi-generator src/index.ts
```

## Custom codec file

`openapi-generator` only reads files in the specified package, and stops at the module
boundary. This allows it to work even without `node_modules` installed. It has built-in
support for `io-ts`, `io-ts-types`, and `@api-ts/io-ts-http` imports. If your package
imports codecs from another external library, then you will have to define them in a
custom configuration file so that `openapi-generator` will understand them. To do so,
create a JS file with the following format:

```typescript
module.exports = (E) => {
return {
'io-ts-bigint': {
BigIntFromString: () => E.right({ type: 'string' }),
NonZeroBigInt: () => E.right({ type: 'number' }),
NonZeroBigIntFromString: () => E.right({ type: 'string' }),
NegativeBigIntFromString: () => E.right({ type: 'string' }),
NonNegativeBigIntFromString: () => E.right({ type: 'string' }),
PositiveBigIntFromString: () => E.right({ type: 'string' }),
},
// ... and so on for other packages
};
};
```
## Preparing a types package for reusable codecs

In order to use types from external `io-ts` types packages, you must ensure two things
are done.

1. The package source code must be included in the bundle, as the generator is built to
generate specs based from the Typescript AST. It is not set up to work with
transpiled js code. You can do this by modifying your `package.json` to include your
source code in the bundle. For example, if the source code is present in the `src/`
directory, then add `src/` to the files array in the `package.json` of your project.
2. After Step 1, change the `types` field in the `package.json` to be the entry point of
the types in the source code. For example, if the entrypoint is `src/index.ts`, then
set `"types": "src/index.ts"` in the `package.json`

## Defining Custom Codecs

When working with `openapi-generator`, you may encounter challenges with handling custom
codecs that require JavaScript interpretation or aren't natively supported by the
generator. These issues typically arise with codecs such as `new t.Type(...)` and other
primitives that aren't directly supported. However, there are two solutions to address
these challenges effectively. Click [here](#list-of-supported-io-ts-primitives) for the
list of supported primitives.

### Solution 1: Using a Custom Codec Configuration File

`openapi-generator` supports importing codecs from other packages in `node_modules`, but
it struggles with primitives that need JavaScript interpretation, such as
`new t.Type(...)`. To work around this, you can define schemas for these codecs in a
configuration file within your downstream types package (where you generate the API
docs). This allows the generator to understand and use these schemas where necessary.
Follow these steps to create and use a custom codec configuration file:

1. Create a JavaScript file with the following format:

```javascript
module.exports = (E) => {
return {
'io-ts-bigint': {
BigIntFromString: () => E.right({ type: 'string' }),
NonZeroBigInt: () => E.right({ type: 'number' }),
NonZeroBigIntFromString: () => E.right({ type: 'string' }),
NegativeBigIntFromString: () => E.right({ type: 'string' }),
NonNegativeBigIntFromString: () => E.right({ type: 'string' }),
PositiveBigIntFromString: () => E.right({ type: 'string' }),
},
// ... and so on for other packages
};
};
```

2. The input parameter `E` is the namespace import of `fp-ts/Either`, which avoids
issues with `require`. The return type should be a `Record` containing AST
definitions for external libraries. For more information on the structure, refer to
[KNOWN_IMPORTS](./src/knownImports.ts).

### Solution 2: Defining Custom Codec Schemas in the Types Package (recommended)
ad-world marked this conversation as resolved.
Show resolved Hide resolved

`openapi-generator` now offers the ability to define the schema of custom codecs
directly within the types package that defines them, rather than the downstream package
that uses them. This approach is particularly useful for codecs that are used in many
different types packages. Here’s how you can define schemas for your custom codecs in
the upstream repository:

1. Create a file named `openapi-gen.config.js` in the root of your repository.

2. Add the following line to the `package.json` of the types package:

```json
"customCodecFile": "openapi-gen.config.js"
```

3. In the `openapi-gen.config.js` file, define your custom codecs:

```javascript
module.exports = (E) => {
return {
SampleCodecDefinition: () =>
E.right({
type: 'string',
default: 'defaultString',
minLength: 1,
}),
// ... rest of your custom codec definitions
};
};
```

By following these steps, the schemas for your custom codecs will be included in the
generated API docs for any endpoints that use the respective codecs. The input parameter
`E` is the namespace import of `fp-ts/Either`, and the return type should be a `Record`
containing AST definitions for external libraries. For more details, see
[KNOWN_IMPORTS](./src/knownImports.ts).

## List of supported io-ts primitives

The input parameter `E` is the namespace import of `fp-ts/Either` (so that trying to
`require` it from the config file isn't an issue), and the return type is a `Record`
containing AST definitions for external libraries.
[Refer to KNOWN_IMPORTS here for info on the structure](./src/knownImports.ts)
- string
- number
- bigint
- boolean
- null
- nullType
- undefined
- unknown
- any
- array
- readonlyArray
- object
- type
- partial
- exact
- strict
- record
- union
- intersection
- literal
- keyof
- brand
- UnknownRecord
- void
8 changes: 4 additions & 4 deletions packages/openapi-generator/src/cli.ts
ad-world marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
Expand Up @@ -171,17 +171,17 @@ const app = command({

const initE = findSymbolInitializer(project.right, sourceFile, ref.name);
if (E.isLeft(initE)) {
console.error(
`[ERROR] Could not find symbol '${ref.name}' in '${ref.location}': ${initE.left}`,
logError(
`Could not find symbol '${ref.name}' in '${ref.location}': ${initE.left}`,
);
process.exit(1);
}
const [newSourceFile, init, comment] = initE.right;

const codecE = parseCodecInitializer(project.right, newSourceFile, init);
if (E.isLeft(codecE)) {
console.error(
`[ERROR] Could not parse codec '${ref.name}' in '${ref.location}': ${codecE.left}`,
logError(
`Could not parse codec '${ref.name}' in '${ref.location}': ${codecE.left}`,
);
process.exit(1);
}
Expand Down
47 changes: 46 additions & 1 deletion packages/openapi-generator/src/project.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ import resolve from 'resolve';

import { KNOWN_IMPORTS, type KnownCodec } from './knownImports';
import { parseSource, type SourceFile } from './sourceFile';
import { errorLeft } from './error';
import { errorLeft, logError, logInfo } from './error';

const readFile = promisify(fs.readFile);

Expand Down Expand Up @@ -37,6 +37,7 @@ export class Project {
async parseEntryPoint(entryPoint: string): Promise<E.Either<string, Project>> {
const queue: string[] = [entryPoint];
let path: string | undefined;
const visitedPackages = new Set<string>();
while (((path = queue.pop()), path !== undefined)) {
if (!['.ts', '.js'].includes(p.extname(path))) {
continue;
Expand All @@ -59,6 +60,14 @@ export class Project {
// If we are not resolving a relative path, we need to resolve the entry point
const baseDir = p.dirname(sourceFile.path);
let entryPoint = this.resolveEntryPoint(baseDir, sym.from);

if (!visitedPackages.has(sym.from)) {
// This is a step that checks if this import has custom codecs, and loads them into known imports
await this.populateCustomCodecs(baseDir, sym.from);
}

visitedPackages.add(sym.from);

if (E.isLeft(entryPoint)) {
continue;
} else if (!this.has(entryPoint.right)) {
Expand Down Expand Up @@ -148,4 +157,40 @@ export class Project {
getTypes() {
return this.types;
}

private async populateCustomCodecs(basedir: string, packageName: string) {
try {
const packageJson = resolve.sync(`${packageName}/package.json`, {
basedir,
extensions: ['.json'],
});
const packageInfo = JSON.parse(fs.readFileSync(packageJson, 'utf8'));

if (packageInfo['customCodecFile']) {
// The package defines their own custom codecs
const customCodecPath = resolve.sync(
`${packageName}/${packageInfo['customCodecFile']}`,
{
basedir,
extensions: ['.ts', '.js'],
},
);
const module = await import(customCodecPath);
if (module.default === undefined) {
logError(`Could not find default export in ${customCodecPath}`);
anshchaturvedi marked this conversation as resolved.
Show resolved Hide resolved
return;
}

const customCodecs = module.default(E);
this.knownImports[packageName] = {
...customCodecs,
...this.knownImports[packageName],
};
Copy link
Contributor Author

@ad-world ad-world Aug 8, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

question for reviewers: do you think the order should be switched around here? i.e. ...customCodecs comes first, so that if a downstream repo wants to redefine a schema for whatever reason, they can still do that rather than having it overwritten by the main schema definition

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

went ahead and made the change, let me know if you think it should be reverted. I think it makes sense to give the end user more priority.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I agree with this approach 👍🏽


logInfo(`Loaded custom codecs for ${packageName}`);
}
} catch (e) {
return;
}
}
}
3 changes: 2 additions & 1 deletion packages/openapi-generator/src/sourceFile.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import * as swc from '@swc/core';

import { parseTopLevelSymbols, type SymbolTable } from './symbol';
import { logError } from './error';

export type SourceFile = {
path: string;
Expand Down Expand Up @@ -41,7 +42,7 @@ export async function parseSource(
span: module.span,
};
} catch (e: unknown) {
console.error(`Error parsing source file: ${path}`, e);
logError(`Error parsing source file: ${path}, ${e}`);
ad-world marked this conversation as resolved.
Show resolved Hide resolved
return undefined;
}
}
49 changes: 47 additions & 2 deletions packages/openapi-generator/test/externalModuleApiSpec.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -126,8 +126,8 @@ async function testCase(
components,
);

assert.deepEqual(errors, expectedErrors);
assert.deepEqual(openapi, expected);
assert.deepStrictEqual(errors, expectedErrors);
assert.deepStrictEqual(openapi, expected);
anshchaturvedi marked this conversation as resolved.
Show resolved Hide resolved
});
}

Expand Down Expand Up @@ -319,3 +319,48 @@ testCase(
},
[]
)

testCase("simple api spec with custom codec", "test/sample-types/apiSpecWithCustomCodec.ts", {
openapi: "3.0.3",
info: {
title: "simple api spec with custom codec",
version: "4.7.4",
description: "simple api spec with custom codec"
},
paths: {
"/test": {
get: {
parameters: [],
responses: {
200: {
description: "OK",
content: {
'application/json': {
schema: {
type: 'string',
description: 'Sample custom codec',
example: 'sample',
format: 'sample'
}
}
}
},
201: {
description: 'Created',
content: {
'application/json': {
schema: {
type: 'number',
description: 'Another sample codec',
}
}
}
}
}
}
}
},
components: {
schemas: {}
}
}, []);
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
import { SampleCustomCodec, AnotherSampleCodec } from '@bitgo/custom-codecs';
import * as h from '@api-ts/io-ts-http';

export const apiSpec = h.apiSpec({
'api.get.test': {
get: h.httpRoute({
path: '/test',
method: 'GET',
request: h.httpRequest({}),
response: {
200: SampleCustomCodec,
201: AnotherSampleCodec,
},
}),
},
})

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Loading