testcafe
Version:
Automated browser testing for the modern web development stack.
168 lines • 25.8 kB
JavaScript
"use strict";
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.getTestListFromCode = exports.getTestList = exports.EsNextTestFileParser = void 0;
const lodash_1 = require("lodash");
const core_1 = require("@babel/core");
const compiler_1 = __importDefault(require("./compiler"));
const test_file_parser_base_1 = require("../../test-file-parser-base");
const TOKEN_TYPE = {
Identifier: 'Identifier',
PropertyAccessExpression: 'MemberExpression',
CallExpression: 'CallExpression',
TaggedTemplateExpression: 'TaggedTemplateExpression',
TemplateLiteral: 'TemplateLiteral',
StringLiteral: 'StringLiteral',
ArrowFunctionExpression: 'ArrowFunctionExpression',
FunctionExpression: 'FunctionExpression',
ExpressionStatement: 'ExpressionStatement',
ReturnStatement: 'ReturnStatement',
FunctionDeclaration: 'FunctionDeclaration',
VariableStatement: 'VariableStatement',
VariableDeclaration: 'VariableDeclaration',
ObjectLiteralExpression: 'ObjectExpression',
};
class EsNextTestFileParser extends test_file_parser_base_1.TestFileParserBase {
constructor() {
super(TOKEN_TYPE);
}
static getTagStrValue(exp) {
//NOTE: we set <computed name> if template literal has at least one computed substring ${...}
return exp.expressions.length ? EsNextTestFileParser.formatComputedName(exp.loc.start.line) : exp.quasis[0].value.raw;
}
isAsyncFn(token) {
return token.async || token.generator;
}
getTokenType(token) {
return token.type;
}
getRValue(token) {
return token.declarations[0].init;
}
getStringValue(token) {
const stringTypes = [this.tokenType.StringLiteral, this.tokenType.TemplateLiteral, this.tokenType.Identifier];
if (stringTypes.indexOf(token.type) > -1)
return this.formatFnArg(token);
return null;
}
getFunctionBody(token) {
return token.body && token.body.body ? token.body.body : [];
}
getCalleeToken(token) {
return token.callee;
}
getMemberFnName(token) {
return token.callee.property.name;
}
formatFnData(name, value, token, meta = [{}]) {
return {
fnName: name,
value: value,
loc: token.loc,
start: token.start,
end: token.end,
meta: (0, lodash_1.merge)({}, ...meta),
isSkipped: test_file_parser_base_1.TestFileParserBase.isSkipped(token),
};
}
getKeyValue(prop) {
const { key, value } = prop;
return {
key: key.name || this.formatFnArg(key),
value: this.getStringValue(value),
};
}
analyzeMemberExp(token) {
let exp = token;
const tokenType = this.tokenType;
const callStack = [exp];
while (exp.type !== tokenType.Identifier) {
if (exp.type === tokenType.CallExpression)
exp = exp.callee;
else if (exp.type === tokenType.PropertyAccessExpression)
exp = exp.object;
else if (exp.type === tokenType.TaggedTemplateExpression)
exp = exp.tag;
else
return null;
if (exp.type !== tokenType.Identifier)
callStack.push(exp);
}
if (!this.isApiFn(exp.name))
return null;
const meta = this.getMetaInfo(callStack.slice());
let parentExp = callStack.pop();
if (parentExp.type === tokenType.CallExpression)
return this.formatFnData(exp.name, this.formatFnArg(parentExp.arguments[0]), token, meta);
if (parentExp.type === tokenType.TaggedTemplateExpression)
return this.formatFnData(exp.name, EsNextTestFileParser.getTagStrValue(parentExp.quasi), token, meta);
if (parentExp.type === tokenType.PropertyAccessExpression) {
while (parentExp) {
if (parentExp.type === tokenType.CallExpression && parentExp.callee) {
const calleeType = parentExp.callee.type;
const calleeMemberFn = parentExp.callee.property && parentExp.callee.property.name;
if (this.checkExpDefineTargetName(calleeType, calleeMemberFn))
return this.formatFnData(exp.name, this.formatFnArg(parentExp.arguments[0]), token, meta);
}
if (parentExp.type === tokenType.TaggedTemplateExpression && parentExp.tag) {
const tagType = parentExp.tag.type;
const tagMemberFn = parentExp.tag.property && parentExp.tag.property.name;
if (this.checkExpDefineTargetName(tagType, tagMemberFn))
return this.formatFnData(exp.name, EsNextTestFileParser.getTagStrValue(parentExp.quasi), token, meta);
}
parentExp = callStack.pop();
}
}
return null;
}
formatFnArg(arg) {
if (arg.type === this.tokenType.Identifier)
return EsNextTestFileParser.formatComputedName(arg.loc.start.line);
if (arg.type === this.tokenType.TemplateLiteral)
return EsNextTestFileParser.getTagStrValue(arg);
if (arg.type === this.tokenType.StringLiteral)
return arg.value;
return null;
}
getFnCall(token) {
if (!this.isApiFn(token.callee.name))
return null;
return this.formatFnData(token.callee.name, this.formatFnArg(token.arguments[0]), token);
}
getTaggedTemplateExp(token) {
return this.formatFnData(token.tag.name, EsNextTestFileParser.getTagStrValue(token.quasi), token);
}
analyzeFnCall(token) {
const tokenType = this.tokenType;
if (token.type === tokenType.PropertyAccessExpression)
return this.analyzeMemberExp(token);
if (token.type === tokenType.CallExpression) {
const calleeType = token.callee.type;
if (calleeType === tokenType.PropertyAccessExpression || calleeType === tokenType.CallExpression)
return this.analyzeMemberExp(token);
if (calleeType === tokenType.FunctionExpression || calleeType === tokenType.ArrowFunctionExpression)
return this.collectTestCafeCalls(token.callee.body.body);
return this.getFnCall(token);
}
if (token.type === tokenType.TaggedTemplateExpression) {
if (token.tag.type === tokenType.PropertyAccessExpression)
return this.analyzeMemberExp(token);
return this.getTaggedTemplateExp(token);
}
return null;
}
parse(code) {
const compilerOptions = compiler_1.default.getBabelOptions(null, code);
delete compilerOptions.filename;
const opts = (0, lodash_1.assign)(compilerOptions, { ast: true });
const ast = (0, core_1.transform)(code, opts).ast;
return this.analyze(ast.program.body);
}
}
exports.EsNextTestFileParser = EsNextTestFileParser;
const parser = new EsNextTestFileParser();
exports.getTestList = parser.getTestList.bind(parser);
exports.getTestListFromCode = parser.getTestListFromCode.bind(parser);
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"get-test-list.js","sourceRoot":"","sources":["../../../../../src/compiler/test-file/formats/es-next/get-test-list.js"],"names":[],"mappings":"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","sourcesContent":["import { assign, merge } from 'lodash';\nimport { transform } from '@babel/core';\nimport ESNextTestFileCompiler from './compiler';\nimport { TestFileParserBase } from '../../test-file-parser-base';\n\nconst TOKEN_TYPE = {\n    Identifier:               'Identifier',\n    PropertyAccessExpression: 'MemberExpression',\n    CallExpression:           'CallExpression',\n    TaggedTemplateExpression: 'TaggedTemplateExpression',\n    TemplateLiteral:          'TemplateLiteral',\n    StringLiteral:            'StringLiteral',\n    ArrowFunctionExpression:  'ArrowFunctionExpression',\n    FunctionExpression:       'FunctionExpression',\n    ExpressionStatement:      'ExpressionStatement',\n    ReturnStatement:          'ReturnStatement',\n    FunctionDeclaration:      'FunctionDeclaration',\n    VariableStatement:        'VariableStatement',\n    VariableDeclaration:      'VariableDeclaration',\n    ObjectLiteralExpression:  'ObjectExpression',\n};\n\nexport class EsNextTestFileParser extends TestFileParserBase {\n    constructor () {\n        super(TOKEN_TYPE);\n    }\n\n    static getTagStrValue (exp) {\n        //NOTE: we set <computed name> if template literal has at least one computed substring ${...}\n        return exp.expressions.length ? EsNextTestFileParser.formatComputedName(exp.loc.start.line) : exp.quasis[0].value.raw;\n    }\n\n    isAsyncFn (token) {\n        return token.async || token.generator;\n    }\n\n    getTokenType (token) {\n        return token.type;\n    }\n\n    getRValue (token) {\n        return token.declarations[0].init;\n    }\n\n    getStringValue (token) {\n        const stringTypes = [this.tokenType.StringLiteral, this.tokenType.TemplateLiteral, this.tokenType.Identifier];\n\n        if (stringTypes.indexOf(token.type) > -1)\n            return this.formatFnArg(token);\n\n        return null;\n    }\n\n    getFunctionBody (token) {\n        return token.body && token.body.body ? token.body.body : [];\n    }\n\n    getCalleeToken (token) {\n        return token.callee;\n    }\n\n    getMemberFnName (token) {\n        return token.callee.property.name;\n    }\n\n    formatFnData (name, value, token, meta = [{}]) {\n        return {\n            fnName:    name,\n            value:     value,\n            loc:       token.loc,\n            start:     token.start,\n            end:       token.end,\n            meta:      merge({}, ...meta),\n            isSkipped: TestFileParserBase.isSkipped(token),\n        };\n    }\n\n    getKeyValue (prop) {\n        const { key, value } = prop;\n\n        return {\n            key:   key.name || this.formatFnArg(key),\n            value: this.getStringValue(value),\n        };\n    }\n\n    analyzeMemberExp (token) {\n        let exp         = token;\n        const tokenType = this.tokenType;\n        const callStack = [exp];\n\n        while (exp.type !== tokenType.Identifier) {\n            if (exp.type === tokenType.CallExpression)\n                exp = exp.callee;\n\n            else if (exp.type === tokenType.PropertyAccessExpression)\n                exp = exp.object;\n\n            else if (exp.type === tokenType.TaggedTemplateExpression)\n                exp = exp.tag;\n\n            else\n                return null;\n\n            if (exp.type !== tokenType.Identifier)\n                callStack.push(exp);\n        }\n\n        if (!this.isApiFn(exp.name)) return null;\n\n        const meta = this.getMetaInfo(callStack.slice());\n\n        let parentExp = callStack.pop();\n\n        if (parentExp.type === tokenType.CallExpression)\n            return this.formatFnData(exp.name, this.formatFnArg(parentExp.arguments[0]), token, meta);\n\n        if (parentExp.type === tokenType.TaggedTemplateExpression)\n            return this.formatFnData(exp.name, EsNextTestFileParser.getTagStrValue(parentExp.quasi), token, meta);\n\n        if (parentExp.type === tokenType.PropertyAccessExpression) {\n            while (parentExp) {\n                if (parentExp.type === tokenType.CallExpression && parentExp.callee) {\n                    const calleeType     = parentExp.callee.type;\n                    const calleeMemberFn = parentExp.callee.property && parentExp.callee.property.name;\n\n                    if (this.checkExpDefineTargetName(calleeType, calleeMemberFn))\n                        return this.formatFnData(exp.name, this.formatFnArg(parentExp.arguments[0]), token, meta);\n                }\n\n                if (parentExp.type === tokenType.TaggedTemplateExpression && parentExp.tag) {\n                    const tagType     = parentExp.tag.type;\n                    const tagMemberFn = parentExp.tag.property && parentExp.tag.property.name;\n\n                    if (this.checkExpDefineTargetName(tagType, tagMemberFn))\n                        return this.formatFnData(exp.name, EsNextTestFileParser.getTagStrValue(parentExp.quasi), token, meta);\n                }\n\n                parentExp = callStack.pop();\n            }\n        }\n\n        return null;\n    }\n\n    formatFnArg (arg) {\n        if (arg.type === this.tokenType.Identifier)\n            return EsNextTestFileParser.formatComputedName(arg.loc.start.line);\n\n        if (arg.type === this.tokenType.TemplateLiteral)\n            return EsNextTestFileParser.getTagStrValue(arg);\n\n        if (arg.type === this.tokenType.StringLiteral)\n            return arg.value;\n\n        return null;\n    }\n\n    getFnCall (token) {\n        if (!this.isApiFn(token.callee.name)) return null;\n\n        return this.formatFnData(token.callee.name, this.formatFnArg(token.arguments[0]), token);\n    }\n\n    getTaggedTemplateExp (token) {\n        return this.formatFnData(token.tag.name, EsNextTestFileParser.getTagStrValue(token.quasi), token);\n    }\n\n    analyzeFnCall (token) {\n        const tokenType = this.tokenType;\n\n        if (token.type === tokenType.PropertyAccessExpression)\n            return this.analyzeMemberExp(token);\n\n        if (token.type === tokenType.CallExpression) {\n            const calleeType = token.callee.type;\n\n            if (calleeType === tokenType.PropertyAccessExpression || calleeType === tokenType.CallExpression)\n                return this.analyzeMemberExp(token);\n\n            if (calleeType === tokenType.FunctionExpression || calleeType === tokenType.ArrowFunctionExpression)\n                return this.collectTestCafeCalls(token.callee.body.body);\n\n            return this.getFnCall(token);\n        }\n\n        if (token.type === tokenType.TaggedTemplateExpression) {\n            if (token.tag.type === tokenType.PropertyAccessExpression)\n                return this.analyzeMemberExp(token);\n\n            return this.getTaggedTemplateExp(token);\n        }\n\n        return null;\n    }\n\n    parse (code) {\n        const compilerOptions = ESNextTestFileCompiler.getBabelOptions(null, code);\n\n        delete compilerOptions.filename;\n\n        const opts = assign(compilerOptions, { ast: true });\n        const ast  = transform(code, opts).ast;\n\n        return this.analyze(ast.program.body);\n    }\n}\n\nconst parser = new EsNextTestFileParser();\n\nexport const getTestList         = parser.getTestList.bind(parser);\nexport const getTestListFromCode = parser.getTestListFromCode.bind(parser);\n"]}