Skip to content

Commit 3f5c846

Browse files
authored
Improve declaration of untagged unions to allow for generator optimizations (elastic#2548)
1 parent 42aaa69 commit 3f5c846

File tree

16 files changed

+839
-625
lines changed

16 files changed

+839
-625
lines changed

compiler-rs/clients_schema/src/lib.rs

+9
Original file line numberDiff line numberDiff line change
@@ -380,6 +380,7 @@ pub enum ServerDefault {
380380
pub enum Variants {
381381
ExternalTag(ExternalTag),
382382
InternalTag(InternalTag),
383+
Untagged(Untagged),
383384
Container(Container),
384385
}
385386

@@ -404,6 +405,13 @@ pub struct InternalTag {
404405
pub default_tag: Option<String>,
405406
}
406407

408+
#[derive(Debug, Clone, Serialize, Deserialize)]
409+
#[serde(rename_all = "camelCase")]
410+
pub struct Untagged {
411+
#[serde(default)]
412+
pub non_exhaustive: bool,
413+
}
414+
407415
#[derive(Debug, Clone, Serialize, Deserialize)]
408416
#[serde(rename_all = "camelCase")]
409417
pub struct Container {
@@ -799,6 +807,7 @@ impl TypeAlias {
799807
pub enum TypeAliasVariants {
800808
ExternalTag(ExternalTag),
801809
InternalTag(InternalTag),
810+
Untagged(Untagged),
802811
}
803812

804813
//------------------------------------------------------------------------------------------------------------

compiler-rs/clients_schema_to_openapi/src/main.rs

+5-1
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@
1616
// under the License.
1717

1818
use std::path::{Path, PathBuf};
19+
use anyhow::bail;
1920

2021
use clap::{Parser, ValueEnum};
2122
use clients_schema::{Availabilities, Visibility};
@@ -71,7 +72,10 @@ impl Cli {
7172
std::fs::read_to_string(self.schema)?
7273
};
7374

74-
let mut model: clients_schema::IndexedModel = serde_json::from_str(&json)?;
75+
let mut model: clients_schema::IndexedModel = match serde_json::from_str(&json) {
76+
Ok(indexed_model) => indexed_model,
77+
Err(e) => bail!("cannot parse schema json: {}", e)
78+
};
7579

7680
if let Some(flavor) = self.flavor {
7781
if flavor != SchemaFlavor::All {

compiler-rs/clients_schema_to_openapi/src/schemas.rs

+2
Original file line numberDiff line numberDiff line change
@@ -395,6 +395,8 @@ impl<'a> TypesAndComponents<'a> {
395395
extensions: Default::default(),
396396
});
397397
}
398+
Some(TypeAliasVariants::Untagged(_tag)) => {
399+
}
398400
};
399401

400402
Ok(schema)

compiler-rs/openapi_to_clients_schema/src/types.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -323,7 +323,7 @@ fn generate_schema_kind_one_of(
323323
let mut variants: Option<TypeAliasVariants> = None;
324324

325325
// TODO: do we want to allow untagged unions (those that are disambiguated by inspecting property names)?
326-
326+
327327
if let Some(discriminator) = discriminator {
328328
variants = Some(TypeAliasVariants::InternalTag(InternalTag {
329329
default_tag: None,

compiler/src/model/metamodel.ts

+11-3
Original file line numberDiff line numberDiff line change
@@ -180,7 +180,7 @@ export abstract class BaseType {
180180
specLocation: string
181181
}
182182

183-
export type Variants = ExternalTag | InternalTag | Container
183+
export type Variants = ExternalTag | InternalTag | Container | Untagged
184184

185185
export class VariantBase {
186186
/**
@@ -207,6 +207,11 @@ export class Container extends VariantBase {
207207
kind: 'container'
208208
}
209209

210+
export class Untagged extends VariantBase {
211+
kind: 'untagged'
212+
untypedVariant: Inherits
213+
}
214+
210215
/**
211216
* Inherits clause (aka extends or implements) for an interface or request
212217
*/
@@ -364,8 +369,11 @@ export class TypeAlias extends BaseType {
364369
type: ValueOf
365370
/** generic parameters: either concrete types or open parameters from the enclosing type */
366371
generics?: TypeName[]
367-
/** Only applicable to `union_of` aliases: identify typed_key unions (external) and variant inventories (internal) */
368-
variants?: InternalTag | ExternalTag
372+
/**
373+
* Only applicable to `union_of` aliases: identify typed_key unions (external), variant inventories (internal)
374+
* and untagged variants
375+
*/
376+
variants?: InternalTag | ExternalTag | Untagged
369377
}
370378

371379
// ------------------------------------------------------------------------------------------------

compiler/src/model/utils.ts

+28-11
Original file line numberDiff line numberDiff line change
@@ -536,8 +536,8 @@ export function modelTypeAlias (declaration: TypeAliasDeclaration): model.TypeAl
536536
if (variants != null) {
537537
assert(
538538
declaration.getJsDocs(),
539-
variants.kind === 'internal_tag' || variants.kind === 'external_tag',
540-
'Type Aliases can only have internal or external variants'
539+
variants.kind === 'internal_tag' || variants.kind === 'external_tag' || variants.kind === 'untagged',
540+
'Type Aliases can only have internal, external or untagged variants'
541541
)
542542
typeAlias.variants = variants
543543
}
@@ -1110,17 +1110,34 @@ export function parseVariantsTag (jsDoc: JSDoc[]): model.Variants | undefined {
11101110
}
11111111
}
11121112

1113-
assert(jsDoc, type === 'internal', `Bad variant type: ${type}`)
1114-
1115-
const pairs = parseKeyValues(jsDoc, values, 'tag', 'default')
1116-
assert(jsDoc, typeof pairs.tag === 'string', 'Internal variant requires a tag definition')
1113+
if (type === 'internal') {
1114+
const pairs = parseKeyValues(jsDoc, values, 'tag', 'default')
1115+
assert(jsDoc, typeof pairs.tag === 'string', 'Internal variant requires a tag definition')
1116+
return {
1117+
kind: 'internal_tag',
1118+
nonExhaustive: nonExhaustive,
1119+
tag: pairs.tag,
1120+
defaultTag: pairs.default
1121+
}
1122+
}
11171123

1118-
return {
1119-
kind: 'internal_tag',
1120-
nonExhaustive: nonExhaustive,
1121-
tag: pairs.tag,
1122-
defaultTag: pairs.default
1124+
if (type === 'untagged') {
1125+
const pairs = parseKeyValues(jsDoc, values, 'untyped')
1126+
assert(jsDoc, typeof pairs.untyped === 'string', 'Untagged variant requires an untyped definition')
1127+
const fqn = pairs.untyped.split('.')
1128+
return {
1129+
kind: 'untagged',
1130+
nonExhaustive: nonExhaustive,
1131+
untypedVariant: {
1132+
type: {
1133+
namespace: fqn.slice(0, fqn.length - 1).join('.'),
1134+
name: fqn[fqn.length - 1]
1135+
}
1136+
}
1137+
}
11231138
}
1139+
1140+
assert(jsDoc, false, `Bad variant type: ${type}`)
11241141
}
11251142

11261143
/**

compiler/src/steps/validate-model.ts

+68-2
Original file line numberDiff line numberDiff line change
@@ -559,14 +559,14 @@ export default async function validateModel (apiModel: model.Model, restSpec: Ma
559559
if (typeDef.type.kind !== 'union_of') {
560560
modelError('The "variants" tag only applies to unions')
561561
} else {
562-
validateTaggedUnion(typeDef.type, typeDef.variants)
562+
validateTaggedUnion(typeDef.name, typeDef.type, typeDef.variants)
563563
}
564564
} else {
565565
validateValueOf(typeDef.type, openGenerics)
566566
}
567567
}
568568

569-
function validateTaggedUnion (valueOf: model.UnionOf, variants: model.InternalTag | model.ExternalTag): void {
569+
function validateTaggedUnion (parentName: TypeName, valueOf: model.UnionOf, variants: model.InternalTag | model.ExternalTag | model.Untagged): void {
570570
if (variants.kind === 'external_tag') {
571571
// All items must have a 'variant' attribute
572572
const items = flattenUnionMembers(valueOf)
@@ -610,6 +610,72 @@ export default async function validateModel (apiModel: model.Model, restSpec: Ma
610610
}
611611

612612
validateValueOf(valueOf, new Set())
613+
} else if (variants.kind === 'untagged') {
614+
if (fqn(parentName) !== '_types.query_dsl:DecayFunction' &&
615+
fqn(parentName) !== '_types.query_dsl:DistanceFeatureQuery' &&
616+
fqn(parentName) !== '_types.query_dsl:RangeQuery') {
617+
throw new Error(`Please contact the devtools team before adding new untagged variant ${fqn(parentName)}`)
618+
}
619+
620+
const untypedVariant = getTypeDef(variants.untypedVariant.type)
621+
if (untypedVariant == null) {
622+
modelError(`Type ${fqn(variants.untypedVariant.type)} not found`)
623+
}
624+
625+
const items = flattenUnionMembers(valueOf)
626+
const baseTypes = new Set<string>()
627+
let foundUntyped = false
628+
629+
for (const item of items) {
630+
if (item.kind !== 'instance_of') {
631+
modelError('Items of type untagged unions must be type references')
632+
} else {
633+
validateTypeRef(item.type, undefined, new Set<string>())
634+
const type = getTypeDef(item.type)
635+
if (type == null) {
636+
modelError(`Type ${fqn(item.type)} not found`)
637+
} else {
638+
if (type.kind !== 'interface') {
639+
modelError(`Type ${fqn(item.type)} must be an interface to be used in an untagged union`)
640+
continue
641+
}
642+
643+
if (untypedVariant != null && fqn(item.type) === fqn(untypedVariant.name)) {
644+
foundUntyped = true
645+
}
646+
647+
if (type.inherits == null) {
648+
modelError(`Type ${fqn(item.type)} must derive from a base type to be used in an untagged union`)
649+
continue
650+
}
651+
652+
baseTypes.add(fqn(type.inherits.type))
653+
654+
const baseType = getTypeDef(type.inherits.type)
655+
if (baseType == null) {
656+
modelError(`Type ${fqn(type.inherits.type)} not found`)
657+
continue
658+
}
659+
660+
if (baseType.kind !== 'interface') {
661+
modelError(`Type ${fqn(type.inherits.type)} must be an interface to be used as the base class of another interface`)
662+
continue
663+
}
664+
665+
if (baseType.generics == null || baseType.generics.length === 0) {
666+
modelError('The common base type of an untagged union must accept at least one generic type argument')
667+
}
668+
}
669+
}
670+
}
671+
672+
if (baseTypes.size !== 1) {
673+
modelError('All items of an untagged union must derive from the same common base type')
674+
}
675+
676+
if (!foundUntyped) {
677+
modelError('The untyped variant of an untagged variant must be contained in the union items')
678+
}
613679
}
614680
}
615681

docs/modeling-guide.md

+34-1
Original file line numberDiff line numberDiff line change
@@ -418,7 +418,7 @@ An annotation allows distinguishing these properties from container variants:
418418

419419
For example:
420420

421-
```
421+
```ts
422422
/**
423423
* @variants container
424424
*/
@@ -435,6 +435,39 @@ class AggregationContainer {
435435
...
436436
```
437437
438+
#### Untagged
439+
440+
The untagged variant is used for unions that can only be distinguished by the type of one or more fields.
441+
442+
> [!WARNING]
443+
> This variant should only be used for legacy types and should otherwise be avoided as far as possible, as it leads to less optimal code generation in the client libraries.
444+
445+
The syntax is:
446+
447+
```ts
448+
/** @variants untagged */
449+
```
450+
451+
Untagged variants must exactly follow a defined pattern.
452+
453+
For example:
454+
455+
```ts
456+
export class MyTypeBase<T1, T2, ...> { ... }
457+
458+
export class MyTypeUntyped extends MyTypeBase<UserDefinedValue> {}
459+
export class MyTypeSpecialized1 extends MyTypeBase<int> {}
460+
export class MyTypeSpecialized2 extends MyTypeBase<string> {}
461+
export class MyTypeSpecialized3 extends MyTypeBase<bool> {}
462+
463+
/**
464+
* @codegen_names untyped, mytype1, mytypet2, mytype3
465+
* @variant untagged untyped=_types.MyTypeUntyped
466+
*/
467+
// Note: deserialization depends on value types
468+
export type MyType = MyTypeUntyped | MyTypeSpecialized1 | MyTypeSpecialized2 | MyTypeSpecialized3
469+
```
470+
438471
### Shortcut properties
439472
440473
In many places Elasticsearch accepts a property value to be either a complete data structure or a single value, that value being a shortcut for a property in the data structure.

0 commit comments

Comments
 (0)