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

Add new @tagMetadata decorator to OpenAPI library #4834

Merged
merged 66 commits into from
Nov 4, 2024
Merged
Show file tree
Hide file tree
Changes from 18 commits
Commits
Show all changes
66 commits
Select commit Hold shift + click to select a range
559b63a
initial
Oct 21, 2024
30c2415
Merge branch 'microsoft:main' into tagMetadata
skywing918 Oct 22, 2024
e6c1f29
test initial
Oct 22, 2024
ccdbe98
Merge branch 'microsoft:main' into tagMetadata
skywing918 Oct 23, 2024
15a8077
Merge branch 'microsoft:main' into tagMetadata
skywing918 Oct 23, 2024
cffc5e0
update case
Oct 23, 2024
c966215
update check x-
Oct 23, 2024
65a5ba3
add change log
Oct 23, 2024
39f791a
update
Oct 23, 2024
e398261
fix build
Oct 23, 2024
81b5457
fix buiild
Oct 23, 2024
59cc624
Merge branch 'main' into tagMetadata
skywing918 Oct 24, 2024
a6c8d59
update logic
Oct 24, 2024
ba58f27
update doc
Oct 24, 2024
3e8a4a9
update
Oct 24, 2024
23c110a
update
Oct 24, 2024
e3949b6
Merge branch 'main' into tagMetadata
skywing918 Oct 25, 2024
7aad065
refactor validateAdditionalInfoModel
Oct 25, 2024
f22c587
update
Oct 25, 2024
3ce0486
add cases
Oct 25, 2024
146cdba
Merge branch 'main' into tagMetadata
skywing918 Oct 28, 2024
d6e1297
update
Oct 28, 2024
b87657c
up
Oct 28, 2024
dd95f10
up
Oct 28, 2024
2ff5d4c
up
Oct 28, 2024
a27732a
Merge branch 'main' into tagMetadata
skywing918 Oct 29, 2024
9fbdebd
update
Oct 29, 2024
8e6e1ba
update cases
Oct 29, 2024
dbc429e
Merge branch 'main' into tagMetadata
skywing918 Oct 30, 2024
f13846d
Merge branch 'main' into tagMetadata
skywing918 Oct 30, 2024
dd54601
update case name
Oct 30, 2024
f32367b
Update .chronus/changes/tagMetadata-2024-9-23-16-55-56.md
skywing918 Oct 31, 2024
7b69494
Update packages/openapi3/src/lib.ts
skywing918 Oct 31, 2024
b39c13e
Merge branch 'main' into tagMetadata
skywing918 Oct 31, 2024
c1366a4
move @tagmetadata decorator to openapi lib
Oct 31, 2024
b255e62
update
Oct 31, 2024
acb2e97
Merge branch 'main' into tagMetadata
skywing918 Oct 31, 2024
c30e0cb
Merge branch 'tagMetadata' of https://github.com/skywing918/typespec …
Oct 31, 2024
c95c6d1
up
Oct 31, 2024
71161a7
update change log
Oct 31, 2024
c52eacf
update
Oct 31, 2024
6daaf4f
validate this is the service namespace.
Oct 31, 2024
48ce431
udpate cases
Oct 31, 2024
555b6e3
Update packages/openapi/src/decorators.ts
skywing918 Oct 31, 2024
d2e672c
Update packages/openapi/src/lib.ts
skywing918 Oct 31, 2024
e6996a1
Update packages/openapi/src/lib.ts
skywing918 Oct 31, 2024
a024b64
Update packages/openapi/src/lib.ts
skywing918 Oct 31, 2024
f3dae26
Update packages/openapi/test/decorators.test.ts
skywing918 Oct 31, 2024
7dbf852
update
Oct 31, 2024
7b49b15
update
Nov 1, 2024
e881343
Merge remote-tracking branch 'origin/main' into tagMetadata
Nov 1, 2024
45736f0
Merge branch 'main' into tagMetadata
skywing918 Nov 1, 2024
4e108b6
spread
Nov 1, 2024
2d29dcd
Merge branch 'tagMetadata' of https://github.com/skywing918/typespec …
Nov 1, 2024
5e4a83f
up
Nov 1, 2024
c62f735
up
Nov 2, 2024
f74036a
update doc
Nov 2, 2024
f3cc874
Merge branch 'main' into tagMetadata
skywing918 Nov 2, 2024
741a78d
fix format
Nov 2, 2024
b768698
Merge branch 'tagMetadata' of https://github.com/skywing918/typespec …
Nov 2, 2024
bcc98bd
update
Nov 2, 2024
1536c11
up
Nov 4, 2024
d11aa87
Merge branch 'microsoft:main' into tagMetadata
skywing918 Nov 4, 2024
736f4eb
Merge branch 'tagMetadata' of https://github.com/skywing918/typespec …
Nov 4, 2024
92e830e
up
Nov 4, 2024
8307e31
Merge branch 'main' into tagMetadata
skywing918 Nov 4, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 8 additions & 0 deletions .chronus/changes/tagMetadata-2024-9-23-16-55-56.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
---
changeKind: feature
packages:
- "@typespec/openapi"
- "@typespec/openapi3"
---

a decorator for specify OpenAPI tag properties
skywing918 marked this conversation as resolved.
Show resolved Hide resolved
35 changes: 35 additions & 0 deletions docs/emitters/openapi3/reference/data-types.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
---
title: "Data types"
---

## TypeSpec.OpenAPI

### `ExternalDocs` {#TypeSpec.OpenAPI.ExternalDocs}

External Docs information.

```typespec
model TypeSpec.OpenAPI.ExternalDocs
```

#### Properties

| Name | Type | Description |
| ------------ | -------- | -------------------- |
| url | `string` | Documentation url |
| description? | `string` | Optional description |

### `TagMetadata` {#TypeSpec.OpenAPI.TagMetadata}

Additional information for the OpenAPI document.
skywing918 marked this conversation as resolved.
Show resolved Hide resolved

```typespec
model TypeSpec.OpenAPI.TagMetadata
```

#### Properties

| Name | Type | Description |
| ------------- | --------------------------------------------------------------- | --------------------------------------- |
| description? | `string` | A description of the API. |
| externalDocs? | [`ExternalDocs`](./data-types.md#TypeSpec.OpenAPI.ExternalDocs) | a external Docs information of the API. |
skywing918 marked this conversation as resolved.
Show resolved Hide resolved
19 changes: 19 additions & 0 deletions docs/emitters/openapi3/reference/decorators.md
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,25 @@ Specify that `oneOf` should be used instead of `anyOf` for that union.

None

### `@tagMetadata` {#@TypeSpec.OpenAPI.tagMetadata}

Specify OpenAPI additional information.

```typespec
@TypeSpec.OpenAPI.tagMetadata(name: valueof string, tagMetadata?: TypeSpec.OpenAPI.TagMetadata)
```

#### Target

`Namespace`

#### Parameters

| Name | Type | Description |
| ----------- | ------------------------------------------------------------- | ----------- |
| name | `valueof string` | tag name |
| tagMetadata | [`TagMetadata`](./data-types.md#TypeSpec.OpenAPI.TagMetadata) | |

### `@useRef` {#@TypeSpec.OpenAPI.useRef}

Specify an external reference that should be used inside of emitting this type.
Expand Down
6 changes: 6 additions & 0 deletions docs/emitters/openapi3/reference/index.mdx
Original file line number Diff line number Diff line change
Expand Up @@ -38,4 +38,10 @@ npm install --save-peer @typespec/openapi3
### Decorators

- [`@oneOf`](./decorators.md#@TypeSpec.OpenAPI.oneOf)
- [`@tagMetadata`](./decorators.md#@TypeSpec.OpenAPI.tagMetadata)
- [`@useRef`](./decorators.md#@TypeSpec.OpenAPI.useRef)

### Models

- [`ExternalDocs`](./data-types.md#TypeSpec.OpenAPI.ExternalDocs)
- [`TagMetadata`](./data-types.md#TypeSpec.OpenAPI.TagMetadata)
83 changes: 16 additions & 67 deletions packages/openapi/src/decorators.ts
Original file line number Diff line number Diff line change
@@ -1,10 +1,7 @@
import {
compilerAssert,
DecoratorContext,
Diagnostic,
DiagnosticTarget,
getDoc,
getProperty,
getService,
getSummary,
Model,
Expand All @@ -23,7 +20,8 @@ import {
InfoDecorator,
OperationIdDecorator,
} from "../generated-defs/TypeSpec.OpenAPI.js";
import { createDiagnostic, createStateSymbol, reportDiagnostic } from "./lib.js";
import { checkNoAdditionalProperties, isOpenAPIExtensionKey, validateIsUri } from "./helpers.js";
import { createStateSymbol, reportDiagnostic } from "./lib.js";
import { AdditionalInfo, ExtensionKey, ExternalDocs } from "./types.js";

const operationIdsKey = createStateSymbol("operationIds");
Expand Down Expand Up @@ -114,10 +112,6 @@ export function getExtensions(program: Program, entity: Type): ReadonlyMap<Exten
return program.stateMap(openApiExtensionKey).get(entity) ?? new Map<ExtensionKey, any>();
}

function isOpenAPIExtensionKey(key: string): key is ExtensionKey {
return key.startsWith("x-");
}

/**
* The @defaultResponse decorator can be applied to a model. When that model is used
* as the return type of an operation, this return type will be the default response.
Expand Down Expand Up @@ -189,12 +183,7 @@ export const $info: InfoDecorator = (
if (data === undefined) {
return;
}
validateAdditionalInfoModel(context, model);
if (data.termsOfService) {
if (!validateIsUri(context, data.termsOfService, "TermsOfService")) {
return;
}
}
validateAdditionalInfoModel(context, model, data);
skywing918 marked this conversation as resolved.
Show resolved Hide resolved
setInfo(context.program, entity, data);
skywing918 marked this conversation as resolved.
Show resolved Hide resolved
};

Expand Down Expand Up @@ -225,64 +214,24 @@ function omitUndefined<T extends Record<string, unknown>>(data: T): T {
return Object.fromEntries(Object.entries(data).filter(([k, v]) => v !== undefined)) as any;
}

function validateIsUri(context: DecoratorContext, url: string, propertyName: string) {
try {
new URL(url);
return true;
} catch {
reportDiagnostic(context.program, {
code: "not-url",
target: context.getArgumentTarget(0)!,
format: { property: propertyName, value: url },
});
return false;
}
}

function validateAdditionalInfoModel(context: DecoratorContext, typespecType: TypeSpecValue) {
function validateAdditionalInfoModel(
context: DecoratorContext,
typespecType: TypeSpecValue,
data: AdditionalInfo & Record<`x-${string}`, unknown>,
) {
const propertyModel = context.program.resolveTypeReference(
"TypeSpec.OpenAPI.AdditionalInfo",
)[0]! as Model;

const diagnostics: Diagnostic[] = [];
if (typeof typespecType === "object" && propertyModel) {
const diagnostics = checkNoAdditionalProperties(
typespecType,
context.getArgumentTarget(0)!,
propertyModel,
diagnostics.push(
skywing918 marked this conversation as resolved.
Show resolved Hide resolved
...checkNoAdditionalProperties(typespecType, context.getArgumentTarget(0)!, propertyModel),
);
context.program.reportDiagnostics(diagnostics);
}
}

function checkNoAdditionalProperties(
typespecType: Type,
target: DiagnosticTarget,
source: Model,
): Diagnostic[] {
const diagnostics: Diagnostic[] = [];
compilerAssert(typespecType.kind === "Model", "Expected type to be a Model.");

for (const [name, type] of typespecType.properties.entries()) {
const sourceProperty = getProperty(source, name);
if (sourceProperty) {
if (sourceProperty.type.kind === "Model") {
const nestedDiagnostics = checkNoAdditionalProperties(
type.type,
target,
sourceProperty.type,
);
diagnostics.push(...nestedDiagnostics);
}
} else if (!isOpenAPIExtensionKey(name)) {
diagnostics.push(
createDiagnostic({
code: "invalid-extension-key",
format: { value: name },
target,
}),
);
}
if (data.termsOfService) {
diagnostics.push(
...validateIsUri(context.getArgumentTarget(0)!, data.termsOfService, "TermsOfService"),
);
}

return diagnostics;
context.program.reportDiagnostics(diagnostics);
}
79 changes: 78 additions & 1 deletion packages/openapi/src/helpers.ts
Original file line number Diff line number Diff line change
@@ -1,18 +1,24 @@
import {
compilerAssert,
Diagnostic,
DiagnosticTarget,
getFriendlyName,
getProperty,
getTypeName,
getVisibility,
isGlobalNamespace,
isService,
isTemplateInstance,
Model,
ModelProperty,
Operation,
Program,
Type,
TypeNameOptions,
} from "@typespec/compiler";
import { getOperationId } from "./decorators.js";
import { reportDiagnostic } from "./lib.js";
import { createDiagnostic, reportDiagnostic } from "./lib.js";
import { ExtensionKey } from "./types.js";

/**
* Determines whether a type will be inlined in OpenAPI rather than defined
Expand Down Expand Up @@ -164,3 +170,74 @@ export function isReadonlyProperty(program: Program, property: ModelProperty) {
// readonly: true, but using separate schemas.
return visibility?.length === 1 && visibility[0] === "read";
}

/**
* Determines if a OpenAPIExtensionKey is start with `x-`.
*/
export function isOpenAPIExtensionKey(key: string): key is ExtensionKey {
return key.startsWith("x-");
}

/**
* Check Additional Properties
*/
export function checkNoAdditionalProperties(
typespecType: Type,
target: DiagnosticTarget,
source: Model,
): Diagnostic[] {
const diagnostics: Diagnostic[] = [];
compilerAssert(typespecType.kind === "Model", "Expected type to be a Model.");
skywing918 marked this conversation as resolved.
Show resolved Hide resolved

for (const [name, type] of typespecType.properties.entries()) {
const sourceProperty = getProperty(source, name);
if (sourceProperty) {
if (sourceProperty.type.kind === "Model") {
const nestedDiagnostics = checkNoAdditionalProperties(
type.type,
target,
sourceProperty.type,
);
diagnostics.push(...nestedDiagnostics);
}
} else if (!isOpenAPIExtensionKey(name)) {
diagnostics.push(
createDiagnostic({
code: "invalid-extension-key",
format: { value: name },
target,
}),
);
}
}

return diagnostics;
}

/**
* Validate a string as a URI.
* @param target The target of the diagnostic
* @param url The URL to validate
* @param propertyName The name of the property being validated
*/
export function validateIsUri(
target: DiagnosticTarget,
url: string,
propertyName: string,
): Diagnostic[] {
const diagnostics: Diagnostic[] = [];
try {
// Attempt to construct a new URL
new URL(url);
} catch {
// If the construction fails, create a diagnostic
diagnostics.push(
createDiagnostic({
code: "not-url",
format: { property: propertyName, value: url },
target,
}),
);
}
return diagnostics;
}
2 changes: 2 additions & 0 deletions packages/openapi/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -21,11 +21,13 @@ export {
} from "./decorators.js";
export {
checkDuplicateTypeName,
checkNoAdditionalProperties,
getOpenAPITypeName,
getParameterKey,
isReadonlyProperty,
resolveOperationId,
shouldInline,
validateIsUri,
} from "./helpers.js";
export { AdditionalInfo, Contact, ExtensionKey, ExternalDocs, License } from "./types.js";

Expand Down
20 changes: 20 additions & 0 deletions packages/openapi3/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -110,6 +110,7 @@ Default: `int64`
### TypeSpec.OpenAPI

- [`@oneOf`](#@oneof)
- [`@tagMetadata`](#@tagmetadata)
- [`@useRef`](#@useref)

#### `@oneOf`
Expand All @@ -128,6 +129,25 @@ Specify that `oneOf` should be used instead of `anyOf` for that union.

None

#### `@tagMetadata`

Specify OpenAPI additional information.

```typespec
@TypeSpec.OpenAPI.tagMetadata(name: valueof string, tagMetadata?: TypeSpec.OpenAPI.TagMetadata)
```

##### Target

`Namespace`

##### Parameters

| Name | Type | Description |
| ----------- | ----------------------------- | ----------- |
| name | `valueof string` | tag name |
| tagMetadata | [`TagMetadata`](#tagmetadata) | |

#### `@useRef`

Specify an external reference that should be used inside of emitting this type.
Expand Down
23 changes: 22 additions & 1 deletion packages/openapi3/generated-defs/TypeSpec.OpenAPI.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,11 @@
import type { DecoratorContext, Model, ModelProperty, Union } from "@typespec/compiler";
import type {
DecoratorContext,
Model,
ModelProperty,
Namespace,
Type,
Union,
} from "@typespec/compiler";

/**
* Specify that `oneOf` should be used instead of `anyOf` for that union.
Expand All @@ -16,7 +23,21 @@ export type UseRefDecorator = (
ref: string,
) => void;

/**
* Specify OpenAPI additional information.
*
* @param name tag name
* @param additionalTag Additional information
*/
export type TagMetadataDecorator = (
context: DecoratorContext,
skywing918 marked this conversation as resolved.
Show resolved Hide resolved
target: Namespace,
name: string,
tagMetadata?: Type,
) => void;

export type TypeSpecOpenAPIDecorators = {
oneOf: OneOfDecorator;
useRef: UseRefDecorator;
tagMetadata: TagMetadataDecorator;
};
Loading
Loading