Skip to content

Commit

Permalink
add handleGenericTypeAnnotation in parser-commons (#37525)
Browse files Browse the repository at this point in the history
Summary:
Part of #34872
Move the switch construct from [parsers/typescript/utils.js](https://github.com/facebook/react-native/blob/e133100721939108b0f28dfa9f60ac627c804018/packages/react-native-codegen/src/parsers/typescript/utils.js#L59-L93) and [parsers/flow/utils.js](https://github.com/facebook/react-native/blob/e133100721939108b0f28dfa9f60ac627c804018/packages/react-native-codegen/src/parsers/flow/utils.js#L56-L81) to the parsers-commons.js file, in a handleGenericTypeAnnotation function. Use that function in place of the switch.

## Changelog:

<!-- Help reviewers and the release process by writing your own changelog entry.

Pick one each for the category and type tags:

[ANDROID|GENERAL|IOS|INTERNAL] [BREAKING|ADDED|CHANGED|DEPRECATED|REMOVED|FIXED|SECURITY] - Message

For more details, see:
https://reactnative.dev/contributing/changelogs-in-pull-requests
-->

[Internal][Added]: Added handleGenericTypeAnnotation in parsers-commons

Pull Request resolved: #37525

Test Plan: `yarn test`

Reviewed By: cortinico

Differential Revision: D46264650

Pulled By: cipolleschi

fbshipit-source-id: a315ee8cad24d91c9e98e5533d4cdc8b43ebc9a0
  • Loading branch information
tarunrajput authored and facebook-github-bot committed Jun 6, 2023
1 parent 0af806e commit 8ffaede
Show file tree
Hide file tree
Showing 6 changed files with 263 additions and 65 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,8 @@ import {
getCommandOptions,
getOptions,
getCommandTypeNameAndOptionsExpression,
getTypeResolutionStatus,
handleGenericTypeAnnotation,
} from '../parsers-commons';
import type {ParserType} from '../errors';

Expand Down Expand Up @@ -1544,3 +1546,120 @@ describe('getCommandTypeNameAndOptionsExpression', () => {
});
});
});

describe('getTypeResolutionStatus', () => {
it('returns type resolution status for a type declaration', () => {
const typeAnnotation = {
id: {
name: 'TypeAnnotationName',
},
};
expect(
getTypeResolutionStatus('alias', typeAnnotation, flowParser),
).toEqual({
successful: true,
type: 'alias',
name: 'TypeAnnotationName',
});
});

it('returns type resolution status for an enum declaration', () => {
const typeAnnotation = {
id: {
name: 'TypeAnnotationName',
},
};
expect(getTypeResolutionStatus('enum', typeAnnotation, flowParser)).toEqual(
{
successful: true,
type: 'enum',
name: 'TypeAnnotationName',
},
);
});
});

describe('handleGenericTypeAnnotation', () => {
it('returns when TypeAnnotation is a type declaration', () => {
const typeAnnotation = {
id: {
name: 'TypeAnnotationName',
},
};
const resolvedTypeAnnotation = {
type: 'TypeAlias',
right: {
type: 'TypeAnnotation',
},
};
expect(
handleGenericTypeAnnotation(
typeAnnotation,
resolvedTypeAnnotation,
flowParser,
),
).toEqual({
typeAnnotation: {
type: 'TypeAnnotation',
},
typeResolutionStatus: {
successful: true,
type: 'alias',
name: 'TypeAnnotationName',
},
});
});

it('returns when TypeAnnotation is an enum declaration', () => {
const typeAnnotation = {
id: {
name: 'TypeAnnotationName',
},
};
const resolvedTypeAnnotation = {
type: 'EnumDeclaration',
body: {
type: 'TypeAnnotation',
},
};
expect(
handleGenericTypeAnnotation(
typeAnnotation,
resolvedTypeAnnotation,
flowParser,
),
).toEqual({
typeAnnotation: {
type: 'TypeAnnotation',
},
typeResolutionStatus: {
successful: true,
type: 'enum',
name: 'TypeAnnotationName',
},
});
});

it('throws when the non GenericTypeAnnotation is unsupported', () => {
const typeAnnotation = {
type: 'UnsupportedTypeAnnotation',
id: {
name: 'UnsupportedType',
},
};
const resolvedTypeAnnotation = {
type: 'UnsupportedTypeAnnotation',
};
expect(() =>
handleGenericTypeAnnotation(
typeAnnotation,
resolvedTypeAnnotation,
flowParser,
),
).toThrow(
new Error(
parser.genericTypeAnnotationErrorMessage(resolvedTypeAnnotation),
),
);
});
});
52 changes: 24 additions & 28 deletions packages/react-native-codegen/src/parsers/flow/parser.js
Original file line number Diff line number Diff line change
Expand Up @@ -55,11 +55,15 @@ const {flowTranslateTypeAnnotation} = require('./modules');
// $FlowFixMe[untyped-import] there's no flowtype flow-parser
const flowParser = require('flow-parser');

const {buildSchema, buildPropSchema} = require('../parsers-commons');
const {
buildSchema,
buildPropSchema,
buildModuleSchema,
handleGenericTypeAnnotation,
} = require('../parsers-commons');
const {Visitor} = require('../parsers-primitives');
const {buildComponentSchema} = require('./components');
const {wrapComponentSchema} = require('../schema.js');
const {buildModuleSchema} = require('../parsers-commons.js');

const fs = require('fs');

Expand Down Expand Up @@ -411,36 +415,16 @@ class FlowParser implements Parser {
break;
}

const resolvedTypeAnnotation = types[node.id.name];
const typeAnnotationName = this.nameForGenericTypeAnnotation(node);
const resolvedTypeAnnotation = types[typeAnnotationName];
if (resolvedTypeAnnotation == null) {
break;
}

switch (resolvedTypeAnnotation.type) {
case parser.typeAlias: {
typeResolutionStatus = {
successful: true,
type: 'alias',
name: node.id.name,
};
node = resolvedTypeAnnotation.right;
break;
}
case parser.enumDeclaration: {
typeResolutionStatus = {
successful: true,
type: 'enum',
name: node.id.name,
};
node = resolvedTypeAnnotation.body;
break;
}
default: {
throw new TypeError(
`A non GenericTypeAnnotation must be a type declaration ('${parser.typeAlias}') or enum ('${parser.enumDeclaration}'). Instead, got the unsupported ${resolvedTypeAnnotation.type}.`,
);
}
}
const {typeAnnotation: typeAnnotationNode, typeResolutionStatus: status} =
handleGenericTypeAnnotation(node, resolvedTypeAnnotation, this);
typeResolutionStatus = status;
node = typeAnnotationNode;
}

return {
Expand Down Expand Up @@ -531,6 +515,18 @@ class FlowParser implements Parser {
);
}
}

nextNodeForTypeAlias(typeAnnotation: $FlowFixMe): $FlowFixMe {
return typeAnnotation.right;
}

nextNodeForEnum(typeAnnotation: $FlowFixMe): $FlowFixMe {
return typeAnnotation.body;
}

genericTypeAnnotationErrorMessage(typeAnnotation: $FlowFixMe): string {
return `A non GenericTypeAnnotation must be a type declaration ('${this.typeAlias}') or enum ('${this.enumDeclaration}'). Instead, got the unsupported ${typeAnnotation.type}.`;
}
}

module.exports = {
Expand Down
15 changes: 15 additions & 0 deletions packages/react-native-codegen/src/parsers/parser.js
Original file line number Diff line number Diff line change
Expand Up @@ -378,4 +378,19 @@ export interface Parser {
};

getProperties(typeName: string, types: TypeDeclarationMap): $FlowFixMe;

/**
* Given a typeAlias, it returns the next node.
*/
nextNodeForTypeAlias(typeAnnotation: $FlowFixMe): $FlowFixMe;

/**
* Given an enum Declaration, it returns the next node.
*/
nextNodeForEnum(typeAnnotation: $FlowFixMe): $FlowFixMe;

/**
* Given a unsupported typeAnnotation, returns an error message.
*/
genericTypeAnnotationErrorMessage(typeAnnotation: $FlowFixMe): string;
}
12 changes: 12 additions & 0 deletions packages/react-native-codegen/src/parsers/parserMock.js
Original file line number Diff line number Diff line change
Expand Up @@ -456,4 +456,16 @@ export class MockedParser implements Parser {
);
}
}

nextNodeForTypeAlias(typeAnnotation: $FlowFixMe): $FlowFixMe {
return typeAnnotation.right;
}

nextNodeForEnum(typeAnnotation: $FlowFixMe): $FlowFixMe {
return typeAnnotation.body;
}

genericTypeAnnotationErrorMessage(typeAnnotation: $FlowFixMe): string {
return `A non GenericTypeAnnotation must be a type declaration ('${this.typeAlias}') or enum ('${this.enumDeclaration}'). Instead, got the unsupported ${typeAnnotation.type}.`;
}
}
74 changes: 73 additions & 1 deletion packages/react-native-codegen/src/parsers/parsers-commons.js
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,12 @@ import type {

import type {Parser} from './parser';
import type {ParserType} from './errors';
import type {ParserErrorCapturer, TypeDeclarationMap, PropAST} from './utils';
import type {
ParserErrorCapturer,
TypeDeclarationMap,
PropAST,
TypeResolutionStatus,
} from './utils';
import type {ComponentSchemaBuilderConfig} from './schema.js';

const {
Expand Down Expand Up @@ -983,6 +988,71 @@ function getCommandProperties(ast: $FlowFixMe, parser: Parser) {
return properties;
}

function getTypeResolutionStatus(
type: 'alias' | 'enum',
typeAnnotation: $FlowFixMe,
parser: Parser,
): TypeResolutionStatus {
return {
successful: true,
type,
name: parser.nameForGenericTypeAnnotation(typeAnnotation),
};
}

function handleGenericTypeAnnotation(
typeAnnotation: $FlowFixMe,
resolvedTypeAnnotation: TypeDeclarationMap,
parser: Parser,
): {
typeAnnotation: $FlowFixMe,
typeResolutionStatus: TypeResolutionStatus,
} {
let typeResolutionStatus;
let node;

switch (resolvedTypeAnnotation.type) {
case parser.typeAlias: {
typeResolutionStatus = getTypeResolutionStatus(
'alias',
typeAnnotation,
parser,
);
node = parser.nextNodeForTypeAlias(resolvedTypeAnnotation);
break;
}
case parser.enumDeclaration: {
typeResolutionStatus = getTypeResolutionStatus(
'enum',
typeAnnotation,
parser,
);
node = parser.nextNodeForEnum(resolvedTypeAnnotation);
break;
}
// parser.interfaceDeclaration is not used here because for flow it should fall through to default case and throw an error
case 'TSInterfaceDeclaration': {
typeResolutionStatus = getTypeResolutionStatus(
'alias',
typeAnnotation,
parser,
);
node = resolvedTypeAnnotation;
break;
}
default: {
throw new TypeError(
parser.genericTypeAnnotationErrorMessage(resolvedTypeAnnotation),
);
}
}

return {
typeAnnotation: node,
typeResolutionStatus,
};
}

module.exports = {
wrapModuleSchema,
unwrapNullable,
Expand All @@ -1007,4 +1077,6 @@ module.exports = {
getEventArgument,
findComponentConfig,
getCommandProperties,
handleGenericTypeAnnotation,
getTypeResolutionStatus,
};
Loading

0 comments on commit 8ffaede

Please sign in to comment.