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(typings): update OpenAPI 3.0 and 3.1 typing declarations #1795

Open
wants to merge 3 commits into
base: main
Choose a base branch
from

Conversation

jeremyfiel
Copy link
Contributor

fixes #1547

What/Why/How?

The typings for OAS 3 and OAS 3_1 schemas were incorrectly defined per the OpenAPI Specification.

OAS 3.0 has a specific requirement which defines an OpenAPI Schema rather than a JSON Schema schema. There are subtleties to the OpenAPI 3.0 Schema.

OpenAPI 3.1 Schemas are equivalent to a JSON Schema 2020-12 schema.

Reference

Per the OpenAPI 3.0 Specification

4.7.24.1 JSON Schema Keywords
The following keywords are taken directly from the JSON Schema definition and follow the same specifications:

  • title
  • multipleOf
  • maximum
  • exclusiveMaximum
  • minimum
  • exclusiveMinimum
  • maxLength
  • minLength
  • pattern (This string SHOULD be a valid regular expression, according to the Ecma-262 Edition 5.1 regular expression dialect)
  • maxItems
  • minItems
  • uniqueItems
  • maxProperties
  • minProperties
  • required
  • enum

The following keywords are taken from the JSON Schema definition but their definitions were adjusted to the OpenAPI Specification.

  • type - Value MUST be a string. Multiple types via an array are not supported.
  • allOf - Inline or referenced schema MUST be of a Schema Object and not a standard JSON Schema.
  • oneOf - Inline or referenced schema MUST be of a Schema Object and not a standard JSON Schema.
  • anyOf - Inline or referenced schema MUST be of a Schema Object and not a standard JSON Schema.
  • not - Inline or referenced schema MUST be of a Schema Object and not a standard JSON Schema.
  • items - Value MUST be an object and not an array. Inline or referenced schema MUST be of a Schema Object and not a standard JSON Schema. items MUST be present if type is "array".
  • properties - Property definitions MUST be a Schema Object and not a standard JSON Schema (inline or referenced).
  • additionalProperties - Value can be boolean or object. Inline or referenced schema MUST be of a Schema Object and not a standard JSON Schema. Consistent with JSON Schema, additionalProperties defaults to true.
  • description - [CommonMark] syntax MAY be used for rich text representation.
  • format - See Data Type Formats for further details. While relying on JSON Schema’s defined formats, the OAS offers a few additional predefined formats.
  • default - The default value represents what would be assumed by the consumer of the input as the value of the schema if one is not provided. Unlike JSON Schema, the value MUST conform to the defined type for the Schema Object defined at the same level. For example, if type is "string", then default can be "foo" but cannot be 1.

Alternatively, any time a Schema Object can be used, a Reference Object can be used in its place. This allows referencing definitions instead of defining them inline.

Additional keywords defined by the JSON Schema specification that are not mentioned here are strictly unsupported.

Other than the JSON Schema subset fields, the following fields MAY be used for further schema documentation:

4.7.24.2 Fixed Fields

Field Name Type Description
nullable boolean This keyword only takes effect if type is explicitly defined within the same Schema Object. A true value indicates that both null values and values of the type specified by type are allowed. Other Schema Object constraints retain their defined behavior, and therefore may disallow the use of null as a value. A false value leaves the specified or default type unmodified. The default value is false.
discriminator Discriminator Object Adds support for polymorphism. The discriminator is used to determine which of a set of schemas a payload is expected to satisfy. See Composition and Inheritance for more details.
readOnly boolean Relevant only for Schema Object properties definitions. Declares the property as “read only”. This means that it MAY be sent as part of a response but SHOULD NOT be sent as part of the request. If the property is marked as readOnly being true and is in the required list, the required will take effect on the response only. A property MUST NOT be marked as both readOnly and writeOnly being true. Default value is false.
writeOnly boolean Relevant only for Schema Object properties definitions. Declares the property as “write only”. Therefore, it MAY be sent as part of a request but SHOULD NOT be sent as part of the response. If the property is marked as writeOnly being true and is in the required list, the required will take effect on the request only. A property MUST NOT be marked as both readOnly and writeOnly being true. Default value is false.
xml XML Object This MAY be used only on property schemas. It has no effect on root schemas. Adds additional metadata to describe the XML representation of this property.
externalDocs External Documentation Object Additional external documentation for this schema.
example Any A free-form field to include an example of an instance for this schema. To represent examples that cannot be naturally represented in JSON or YAML, a string value can be used to contain the example with escaping where necessary.
deprecated boolean Specifies that a schema is deprecated and SHOULD be transitioned out of usage. Default value is false.

Per the OpenAPI 3.1 Specification

4.8.24 Schema Object
The Schema Object allows the definition of input and output data types. These types can be objects, but also primitives and arrays. This object is a superset of the JSON Schema Specification Draft 2020-12. The empty schema (which allows any instance to validate) MAY be represented by the boolean value true and a schema which allows no instance to validate MAY be represented by the boolean value false.

For more information about the keywords, see JSON Schema Core and JSON Schema Validation.

Unless stated otherwise, the keyword definitions follow those of JSON Schema and do not add any additional semantics; this includes keywords such as $schema, $id, $ref, and $dynamicRef being URIs rather than URLs. Where JSON Schema indicates that behavior is defined by the application (e.g. for annotations), OAS also defers the definition of semantics to the application consuming the OpenAPI document.

Testing

Screenshots (optional)

Check yourself

  • Code changed? - Tested with redoc/reference-docs/workflows (internal)
  • All new/updated code is covered with tests
  • New package installed? - Tested in different environments (browser/node)

Security

  • Security impact of change has been considered
  • Code follows company security practices and guidelines

@jeremyfiel jeremyfiel requested review from a team as code owners November 5, 2024 18:55
Copy link

changeset-bot bot commented Nov 5, 2024

🦋 Changeset detected

Latest commit: 5428993

The changes in this PR will be included in the next version bump.

This PR includes changesets to release 2 packages
Name Type
@redocly/openapi-core Patch
@redocly/cli Patch

Not sure what this means? Click here to learn what changesets are.

Click here if you're a maintainer who wants to add another changeset to this PR

@jeremyfiel jeremyfiel force-pushed the feat/update-openapi-typings branch 4 times, most recently from 0ba50f7 to 7463f90 Compare November 5, 2024 19:42
@jeremyfiel
Copy link
Contributor Author

ok, i think i'm done with this. missed a few keywords for Oas3_1 schemas.

ready for review. thanks

@@ -112,13 +112,10 @@ export interface Oas3Xml {
}

// common fields for OpenAPI Schema v3.x
interface Oas3XSchemaBase<T> {
interface Oas3XSchemaBase {
Copy link
Contributor

Choose a reason for hiding this comment

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

I prefer using the generic interface because it's less likely for the common properties to diverge. Not sure which approach would be cleaner though. Why did you decide to change the approach?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

the keywords which utilized the generic interface were moved into the Oas3Schema and Oas3_1Schema. So it was unused.

Copy link
Contributor

Choose a reason for hiding this comment

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

Yes, but what was the reason for moving the keywords from Oas3XSchemaBase? Was there an issue with leaving them as they were?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Yes, they have different behavior between 30 asd 31. So the extension of Oas3XSchemaBase was incorrect and the keywords should be assigned to each version

Copy link
Contributor

Choose a reason for hiding this comment

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

Ok. I'll test that once I have time. Thank you for contributing!

@jeremyfiel jeremyfiel force-pushed the feat/update-openapi-typings branch from 331b109 to 498f6b7 Compare November 6, 2024 14:38
@jeremyfiel
Copy link
Contributor Author

I added a few extra JSON Schema 2020-12 keywords that were missed

@jeremyfiel jeremyfiel requested a review from tatomyr November 6, 2024 14:55
Copy link
Contributor

@tatomyr tatomyr left a comment

Choose a reason for hiding this comment

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

Let's make it reuse the common props slightly more.
Otherwise looks good!

@@ -112,32 +112,22 @@ export interface Oas3Xml {
}

// common fields for OpenAPI Schema v3.x
interface Oas3XSchemaBase<T> {
interface Oas3XSchemaBase {
Copy link
Contributor

Choose a reason for hiding this comment

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

Suggested change
interface Oas3XSchemaBase {
interface Oas3XSchemaBase<T extends Oas3Schema | Oas3_1Schema> {

packages/core/src/typings/openapi.ts Show resolved Hide resolved
Comment on lines 148 to 150
properties?: { [name: string]: Referenced<Oas3Schema> };
additionalProperties?: boolean | Oas3Schema;
items?: Oas3Schema;
Copy link
Contributor

Choose a reason for hiding this comment

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

Suggested change
properties?: { [name: string]: Referenced<Oas3Schema> };
additionalProperties?: boolean | Oas3Schema;
items?: Oas3Schema;

Comment on lines 153 to 156
oneOf?: Oas3Schema[];
anyOf?: Oas3Schema[];
allOf?: Oas3Schema[];
not?: Oas3Schema;
Copy link
Contributor

Choose a reason for hiding this comment

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

Suggested change
oneOf?: Oas3Schema[];
anyOf?: Oas3Schema[];
allOf?: Oas3Schema[];
not?: Oas3Schema;


xml?: Oas3Xml;
'x-tags'?: string[];
}

export interface Oas3Schema extends Oas3XSchemaBase<Oas3Schema> {
export interface Oas3Schema extends Oas3XSchemaBase {
Copy link
Contributor

Choose a reason for hiding this comment

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

Suggested change
export interface Oas3Schema extends Oas3XSchemaBase {
export interface Oas3Schema extends Oas3XSchemaBase<Oas3Schema> {

}

export interface Oas3_1Schema extends Oas3XSchemaBase<Oas3_1Schema> {
export interface Oas3_1Schema extends Oas3XSchemaBase {
Copy link
Contributor

Choose a reason for hiding this comment

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

Suggested change
export interface Oas3_1Schema extends Oas3XSchemaBase {
export interface Oas3_1Schema extends Oas3XSchemaBase<Oas3_1Schema> {

Comment on lines 171 to 172
properties?: { [name: string]: Referenced<Oas3_1Schema> };
additionalProperties?: boolean | Oas3_1Schema;
Copy link
Contributor

Choose a reason for hiding this comment

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

Suggested change
properties?: { [name: string]: Referenced<Oas3_1Schema> };
additionalProperties?: boolean | Oas3_1Schema;

Comment on lines 175 to 179
items?: Oas3_1Schema;
oneOf?: Oas3_1Schema[];
anyOf?: Oas3_1Schema[];
allOf?: Oas3_1Schema[];
not?: Oas3_1Schema;
Copy link
Contributor

Choose a reason for hiding this comment

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

Suggested change
items?: Oas3_1Schema;
oneOf?: Oas3_1Schema[];
anyOf?: Oas3_1Schema[];
allOf?: Oas3_1Schema[];
not?: Oas3_1Schema;

@jeremyfiel jeremyfiel force-pushed the feat/update-openapi-typings branch from 498f6b7 to 99dbb67 Compare December 3, 2024 01:49
@jeremyfiel
Copy link
Contributor Author

jeremyfiel commented Dec 3, 2024

i'm stuck on the visitors.ts and custom rule files. Any help appreciated.

i will run prettier after the fixes

jeremyfiel and others added 2 commits December 22, 2024 04:26
Added missing keywords for OAS 3.1.x (JSON Schema 2020-12).

Co-authored-by: Andrew Tatomyr <andrew.tatomyr@gmail.com>
@jeremyfiel jeremyfiel force-pushed the feat/update-openapi-typings branch from e27090b to c03175a Compare December 22, 2024 04:26
@jeremyfiel jeremyfiel force-pushed the feat/update-openapi-typings branch from c03175a to 5428993 Compare December 22, 2024 04:40
Comment on lines +280 to 283

export interface Oas3_1Components extends Oas3ComponentsBase<Oas3Schema> {
pathItems?: { [name: string]: Referenced<Oas3PathItem<Oas3Schema | Oas3_1Schema>> };
}
Copy link
Contributor

@tatomyr tatomyr Dec 27, 2024

Choose a reason for hiding this comment

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

I'd add a separate Oas3Components to follow the same pattern as in other types, and slightly improve the Oas3_1Components type to use Oas3_1Schema only:

Suggested change
export interface Oas3_1Components extends Oas3ComponentsBase<Oas3Schema> {
pathItems?: { [name: string]: Referenced<Oas3PathItem<Oas3Schema | Oas3_1Schema>> };
}
export interface Oas3_1Components extends Oas3ComponentsBase<Oas3_1Schema> {
pathItems?: { [name: string]: Referenced<Oas3PathItem<Oas3_1Schema>> };
}
export interface Oas3Components extends Oas3ComponentsBase<Oas3Schema>

Xml?: VisitFunctionOrObject<Oas3Xml>;
SchemaProperties?: VisitFunctionOrObject<Record<string, Oas3Schema>>;
DiscriminatorMapping?: VisitFunctionOrObject<Record<string, string>>;
Discriminator?: VisitFunctionOrObject<Oas3Discriminator>;
Components?: VisitFunctionOrObject<Oas3Components>;
Components?: VisitFunctionOrObject<
Oas3ComponentsBase<Oas3Schema | Oas3_1Schema> | Oas3_1Components
Copy link
Contributor

@tatomyr tatomyr Dec 27, 2024

Choose a reason for hiding this comment

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

Related to https://github.com/Redocly/redocly-cli/pull/1795/files?diff=split&w=1#r1898410733 :

Suggested change
Oas3ComponentsBase<Oas3Schema | Oas3_1Schema> | Oas3_1Components
Oas3Components | Oas3_1Components

@@ -264,15 +262,17 @@ function getFileNamePath(componentDirPath: string, componentName: string, ext: s
}

function gatherComponentsFiles(
components: Oas3Components,
components: Oas3ComponentsBase<Oas3Schema | Oas3_1Schema> | Oas3_1Components,
Copy link
Contributor

Choose a reason for hiding this comment

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

Related to https://github.com/Redocly/redocly-cli/pull/1795/files?diff=split&w=1#r1898410733 :

Suggested change
components: Oas3ComponentsBase<Oas3Schema | Oas3_1Schema> | Oas3_1Components,
components: Oas3Components | Oas3_1Components,

{ usedIn: Location[]; componentType?: keyof Oas3Components; name: string }
{
usedIn: Location[];
componentType?: keyof (Oas3ComponentsBase<Oas3Schema | Oas3_1Schema> | Oas3_1Components);
Copy link
Contributor

Choose a reason for hiding this comment

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

Related to https://github.com/Redocly/redocly-cli/pull/1795/files?diff=split&w=1#r1898410733 :

Suggested change
componentType?: keyof (Oas3ComponentsBase<Oas3Schema | Oas3_1Schema> | Oas3_1Components);
componentType?: keyof (Oas3Components | Oas3_1Components);

>();

function registerComponent(
location: Location,
componentType: keyof Oas3Components,
componentType: keyof (Oas3ComponentsBase<Oas3Schema | Oas3_1Schema> | Oas3_1Components),
Copy link
Contributor

Choose a reason for hiding this comment

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

Suggested change
componentType: keyof (Oas3ComponentsBase<Oas3Schema | Oas3_1Schema> | Oas3_1Components),
componentType: keyof (Oas3Components | Oas3_1Components),

paths?: Oas3Paths;
components?: Oas3Components;
paths?: Oas3Paths<T>;
components?: T extends Oas3_1Schema ? Oas3_1Components : Oas3ComponentsBase<T>;
Copy link
Contributor

Choose a reason for hiding this comment

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

Related to https://github.com/Redocly/redocly-cli/pull/1795/files?diff=split&w=1#r1898410733 :

Suggested change
components?: T extends Oas3_1Schema ? Oas3_1Components : Oas3ComponentsBase<T>;
components?: T extends Oas3_1Schema ? Oas3_1Components : Oas3Components;

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Missing const typing support
2 participants