Skip to content

Commit

Permalink
Parse Member and TypeCast expressions in super class types
Browse files Browse the repository at this point in the history
Summary: Previously, member expressions in parent classes like `class A
extends Foo.Bar`, and type casts such as `class A extends (Foo: Bar)`,
were not parseable by flow-api-translator. This change adds support.
  • Loading branch information
xleoooo committed Jun 11, 2024
1 parent 9e2bbf8 commit 398eafb
Show file tree
Hide file tree
Showing 2 changed files with 94 additions and 15 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -510,6 +510,20 @@ describe('flowToFlowDef', () => {
}`,
);
});
it('more complex', async () => {
await expectTranslateUnchanged(
`declare export class Foo<T>
extends (Bar.TClass<T>: Bar.SClass<T>)
mixins Bar.TMixin<T>
implements Bar.TInterface<T>
{
prop1: T;
method1(): T;
[T]: string;
constructor(): Foo;
}`,
);
});
});
describe('Expression', () => {
async function expectTranslateExpression(
Expand Down
95 changes: 80 additions & 15 deletions tools/hermes-parser/js/flow-api-translator/src/flowToFlowDef.js
Original file line number Diff line number Diff line change
Expand Up @@ -49,6 +49,7 @@ import type {
ObjectTypeAnnotation,
ObjectTypeProperty,
OpaqueType,
QualifiedTypeIdentifier,
Program,
RestElement,
Statement,
Expand Down Expand Up @@ -977,6 +978,35 @@ function convertClassDeclaration(
];
}

function convertExpressionToIdentifier(
node: Expression,
context: TranslationContext,
): DetachedNode<Identifier> | DetachedNode<QualifiedTypeIdentifier> {
if (node.type === 'Identifier') {
return t.Identifier({name: node.name});
} else if (
node.type === 'MemberExpression' &&
node.property.type === 'Identifier' &&
node.object.type !== 'Super'
) {
const id = t.Identifier({name: node.property.name});
return t.QualifiedTypeIdentifier({
qualification: convertExpressionToIdentifier(
// $FlowIssue[incompatible-call] conditional already checks for non-Super type
node.object,
context,
),
id,
});
}

throw translationError(
node,
`Expected ${node.type} to be an Identifier or Member with Identifier property, non-Super object.`,
context,
);
}

function convertSuperClass(
superClass: ?Expression,
superTypeParameters: ?TypeParameterInstantiation,
Expand All @@ -986,23 +1016,58 @@ function convertSuperClass(
return EMPTY_TRANSLATION_RESULT;
}

if (superClass.type !== 'Identifier') {
throw translationError(
superClass,
`SuperClass: Non identifier super type of "${superClass.type}" not supported`,
context,
);
}
const [resultTypeParams, typeParamsDeps] =
convertTypeParameterInstantiationOrNull(superTypeParameters, context);
const superDeps = analyzeTypeDependencies(superClass, context);
return [
t.InterfaceExtends({
id: asDetachedNode(superClass),
typeParameters: resultTypeParams,
}),
[...typeParamsDeps, ...superDeps],
];

switch (superClass.type) {
case 'Identifier': {
const superDeps = analyzeTypeDependencies(superClass, context);
return [
t.InterfaceExtends({
id: asDetachedNode(superClass),
typeParameters: resultTypeParams,
}),
[...typeParamsDeps, ...superDeps],
];
}
case 'MemberExpression': {
const superDeps = analyzeTypeDependencies(superClass, context);
return [
t.InterfaceExtends({
id: convertExpressionToIdentifier(superClass, context),
typeParameters: resultTypeParams,
}),
[...typeParamsDeps, ...superDeps],
];
}
case 'TypeCastExpression': {
const typeAnnotation = superClass.typeAnnotation.typeAnnotation;
const superDeps = analyzeTypeDependencies(typeAnnotation, context);

if (typeAnnotation.type === 'GenericTypeAnnotation') {
return [
t.InterfaceExtends({
id: asDetachedNode(typeAnnotation.id),
typeParameters: resultTypeParams,
}),
[...typeParamsDeps, ...superDeps],
];
}

throw translationError(
superClass,
`SuperClass: Non-identifier typecast super type of "${typeAnnotation.type}" not supported`,
context,
);
}
default: {
throw translationError(
superClass,
`SuperClass: Non identifier super type of "${superClass.type}" not supported`,
context,
);
}
}
}

function convertClassBody(
Expand Down

0 comments on commit 398eafb

Please sign in to comment.