genese-complexity

0.0.6 • Public • Published

Genese Complexity

Genese Complexity is a part of the genese framework which improves you to increase your code quality by analysing the cognitive complexity and the cyclomatic complexity of your source code. This module creates an HTML report displaying an overview of the complexities index of each folder, file or method of your project. Moreover, you will find for each method the elements increasing complexity index, which will help you to refactor easier your code.

Dashboard Genese Complexity

Table of Contents

1. Why use Genese Complexity ?

Genese Complexity is an audit tool which allows you to identify quickly the bad practices concerning cognitive or cyclomatic complexity. You will be able to find quickly the methods with too high complexity index or which should be examined carefully. Genese Complexity is available for JS, TS, JSX, TSX and JAVA.

Top

2. Installation

Genese complexity is used through the genese-cli. You need to install it first:

npm i -g genese-cli

Then, you can install the complexity module:

npm i -g genese-complexity

Top

3. Usage

Genese complexity is launched with the Genese CLI module. The main command-line is

genese cpx [pathDir]

The "pathDir" param is the folder of the project to audit. It can be relative or absolute.

Example:

genese cpx ./src

By default, Genese Complexity is launched for TS files. You can modify this with the -l option

Exemple:

genese cpx -l java ./src

Available options are: ts, js, tsx, jsx and java.

This command line will generate a report in the folder genese/complexity/reports (it can be customized) named folder-report.html. You just need to open it in a browser to display the results.

Top

4. Interpretation of results

4.1 Folder reports

The dashboard's header presents the global statistics of the analyzed project. These scores will give you an overview of its global complexity.

The main part of the page consist in two pairs of charts : the left one is about cognitive complexity and the other one about cyclomatic complexity. The "doughnut chart" is an overview of the distribution of the project's methods statuses (correct, warning and error). The statistics near this chart are reminding warning and error thresholds (which can be customized), and display the number of methods by status.

The bar charts display the number of methods by complexity score.

Below these charts, the first array displays the detailed information of each subfolder. The second presents information of the files inside the current folder (but not inside its subfolders), and the third array displays the complexity indexes of each method of each file located in the current folder or its subfolders, sorted by decreasing cognitive complexity index.

4.2 File reports

As folder reports, the file reports display complexity statistics of files methods. In addition, you will find detailed information of each of these methods, with explanations of the calculation mode of their cognitive complexity.

Top

5. Configuration

Some parameters are configurable by creating a file geneseconfig.json located on the folder where you enter the command-line. This file must have this format :

{
   "complexity": {
       "option": "value"
   }
}

5.1 Thresholds

You can customize the warning and error thresholds of each kind of complexity like this :

{
   "complexity": {
       "cognitiveCpx": {
           "errorThreshold": 15,
           "warningThreshold": 10
       },
       "cyclomaticCpx": {
           "errorThreshold": 15,
           "warningThreshold": 10
       },
   }
}

The values by default are :

{
   "complexity": {
       "cognitiveCpx": {
           "errorThreshold": 10,
           "warningThreshold": 5
       },
       "cyclomaticCpx": {
           "errorThreshold": 10,
           "warningThreshold": 5
       },
   }
}

5.2 Folders to ignore

You can ignore some folders like this :

{
    "complexity": {
        "ignore": [
            ".git",
            ".idea",
            "api",
            "/node_modules",
            "./genese"
        ]
    }
}

The folders ignored by default are /node_modules and ./genese.

5.3 Path of folder to analyse

By default, the folder to analyse is the folder where you enter the command-line. You can change it like this :

{
    "complexity": {
        "pathFolderToAnalyze": "./src/"
    }
}

The last character must be a slash.

5.4 Reports path

By default, the genese complexity report will be located here : current_folder/genese/. You can change it like this:

{
    "complexity": {
        "pathReports": "./my-report-folder"
    }
}

Top

6. Documentation

6.1 Cyclomatic complexity

The cyclomatic complexity is defined as below :

Cyclomatic Complexity

The cyclomatic complexity is a quantitative measure of the number of linearly independent paths through a program’s source code

This notion is a good way to calculate the number of unit tests that we must do to validate the behavior of a method for every possible case. The cyclomatic complexity is highly correlated to the time required to cover all the source code by unit tests.

The force of the cyclomatic complexity is its simplicity: approximately, all breakflows (for, if, switch, ...) increase the complexity score of 1. However, this score is weakly correlated to the maintainability of the source code, which is much more important: even if your code coverage is excellent, a source code which is too complex to understand will be unmaintainable.

If you are a human, you will read your code differently than a machine would do it. So if you want to know if your code is really maintainable, you must look at its cognitive complexity.

6.2 Cognitive complexity

6.2.1 Definition

The cognitive complexity could be defined as below :

Cognitive Complexity

The Cognitive Complexity is a quantitative measure of the time required for a human to understand a program’s source code

A project is maintainable if each method of each file is easy to understand. That is the goal of the cognitive complexity, which should be seen as the most important indicator of code maintainability.

The definition above needs to be clarified: what is difficult to understand and what is not ? Is an if - else more complex than a for, a while or a switch ? How many times a recursive method is more complex than a "normal" one ? Optional chaining or nullish coalescing clearly decrease the complexity of a method, but in which proportion ? There are no indisputable responses.

That's why we need to weight each complexity factor with a value which corresponds to its relative difficulty. To be able to do that, we need at first to define the measure unit of cognitive complexity.

6.2.2 Measure unit

A measure unit must be relative to an "atomic" piece of code, something which can be defined accurately. That's why we will use the definition below :

Measure unit

The measure unit of the Cognitive Complexity is the time required for a human to understand the logic door if.

With this measure unit, we can calculate the Cognitive Complexity of any method by comparing the time needed to understand it with the time required to understand the logic door if. For example, a method with a Cognitive Complexity of 10 is a method which needs the same time to understand than a method with 10 if.

We must insist on an important point : the measure unit must be defined in relation with an "atomic" piece of code. It is necessary to be able to quantify it "in the reality", ie by measuring the average time to understand it. That's why we define the measure unit of Cognitive Complexity with the "logic door if" and not with a if "with something inside".

Let us try to be more explicit: the code below is not atomic, because there is a console.log inside it. Admittedly, the difficulty of a console.log is low, but not null. That's why we can't use this code as measure unit.

if (a) {
    console.log(a);
}

We have the same problem with the next code, which is not atomic too. The a is not "nothing" and me bust be understand too. Moreover, if(a) have different significations according to the used language. For example, in JavaScript and TypeScript, it doesn't mean "if a is true" but "if a is not a falsy value", which is slightly different.

if (a) {

}

The only real "atomic" code is the "mathematic" if, understood as a pure logic door :

if () {

}

6.2.3 Complexity Index

With the definition of the measure unit of the cognitive complexity, we can now define the Cognitive Complexity Index :

Cognitive Complexity Index

The Cognitive Complexity Index is the measure of the Cognitive Complexity of a program’s source code by considering the logic door if as measure unit.

In Genese Complexity module the expression "Complexity Index" will always imply "Cognitive Complexity Index". The following chapters provide an overview of the different ways to evaluate this Complexity Index.

6.3 Complexity Factors

The Complexity Index depends on multiple factors of different weights which can be grouped in several categories. These factors, weights and categories are for now only based on intuition and feedbacks: they SHOULD NOT be interpreted as immutable and definitive values. These elements are only the best way as things stand to measure the Complexity Index. With the help of the community and the increase of knowledge, the weights will be revalued, other factors will appear and other categories will be discovered.

Every time someone will demonstrate that a factor should be weighted differently or that another category should be taken into account, this page will be updated with a new version number.

6.3.1 Factor categories

  • Atomic

Each unbreakable piece of code have a Complexity Index which is weak, but not null. The name of a variable or a method, a keyword like this, import, class, if, ... are trivial, but they need to be red, taken in account and memorized by the human brain. A long method, even without particular problems, is more difficult to understand than a short function having the same "density of complexity".

Each of these trivial nodes have a non null Complexity Index due to their existence. Genese Complexity uses an "atomic weight" equal to 0.1.

  • Example
if (a) {  // ------------------------- + 0.2 (0.1 for the "if" and 0.1 for the "a")
    console.log(a);  // -------------- + 0.3 (0.1 for the "console", 0.1 for the "log" and 0.1 for the "a")
}

=> Total of atomic complexity : 0.5

  • Structural

Some code structures present an intrinsic difficulty which implies that the human brain needs a significant time to take in account their logic implications. The cognitive complexity relative to the intrinsic difficulty of these code structures is called the structural complexity.

The structural category contains different factors: the loops (for, while, ...), the logic doors (&&, ||), the conditions (if, else, switch, ...), the recursions, the callbacks, the regular expressions, etc. All the methods (language methods like slice, map, push, ... and methods of the project and its imported libraries) are increasing structural complexity too. You will find the exhaustive list of the structural factors in the table below.

  • Nesting

Independently of their intrinsic complexity, some elements add specific difficulty due to the nesting of other elements inside them.

  • Example

    The complexity of the code below is only due to the addition of the complexities of the two if : there is no nesting complexity.

if (a) { // ------------------------------ + x
    // ----
}
if (b) { // ------------------------------ + y
    // ----
}

=> Complexity Index : x + y

If the first condition has a Complexity Index equals to x and the second equals to y, the total Complexity Index will be equal to x + y.

Now, if the same conditions are nested, an additional difficulty is due to the obligation in the second if to remember that the condition a must be true to be here. This additional complexity is called nesting complexity and increases the Complexity Index of the source code, which will be strictly higher than x + y.

  • Example
if (a) { // ---------------------------------- + x
   if (b) { // ------------------------------ + y + n (the nesting complexity due to the imbrication in the first "if")
       // ----
   }
}

=> Complexity Index : x + y + n

Genese Complexity adds nesting complexity for the loops (for, while, ...), the conditions (if, else, switch, ...), the ternaries (a = b ? 0 : 1), the arrays (a[b[c]]) and the functions (a = b.f(e => e + 1))).

  • Aggregation

Aggregation complexity is almost the same that nesting complexity, but is relative to consecutive elements and not to nested elements. The idea is simple : an array is simple to understand, but an array of arrays is clearly less trivial. This additional complexity is due to the aggregation of the different elements.

  • Example
const arr = a[b][c];  // ------------------------ + 1 aggregation cpx

We find this problematic with array of arrays, but also in other cases, like regular expressions : they have at first a structural complexity (a regex is difficult for itself), but they have too a specific difficulty in relation with their length, that is, the aggregation of their characters. increases considerably their difficulty. Of course, other factors affect the complexity of the regular expressions, but we use for now their length as a first approximation.

  • Example
const regex = /[^.[\]]+/;  // ------------------------ + 0.8 aggregation cpx (0.1 by character)

Another use case of the aggregation complexity is logic doors, which are simple to understand when they are similar and complicated when they are different and without brackets.

  • Example
if (a && b && c) { // ---------------------- Easy to understand (same logic doors)
    // ---
}
if (a && (b || c)) { // -------------------- Easy to understand (thanks to brackets)
    // ---
}
if (a && b || c) { // ----------------- Difficult to understand (due to the lack of brackets) => + 1 aggregation cpx
    // ---
}

The third example is more difficult to understand than the first and the second one because of the aggregation of different logic doors without brackets.

  • Recursion

Recursivity is easy for machines, but not for humans. A developer will always need to take care about the implications of a recursion and its side effects.

The category "recursion" includes recursive methods and callbacks.

  • Examples
function f(a) {
    return f(a + 1);
}
function f(a) {
    return a(1);
}
  • Use

The other categories were relative to the complexity relative with the structure of the code (loops, conditions, recursivity, etc.). There is another kind of cognitive complexity which is induced by the difficulty to understand the role and the use of a Node. It's typically important when the Node is a method : we need to understand what is its role and how to use it correctly.

  • Examples
function f(a: string) {
    return a.toString();        // The method 'toString()' is easy to understand, no 'use complexity' to add
}
function f(a) {
    return a.map(...).reduce(...);      // The method 'reduce()' is difficult to understand and to use : we must add some 'use complexity'
}

6.3.2 Table of weights (v1.0.0)

This table of weights should never be seen as the exact way to calculate the Complexity Index. It's only the best approximation on the basis of the current knowledge.

Category Factor Weight Example Comments
Aggregation Arrays 1 a[b][c] // ---- Aggregation cpx = 1
Aggregation Regex 0.1 by char /[^.[\]]+/ // ---- Aggregation cpx = 0.8
Aggregation Different logic doors 1 if (a && b || c) // - Aggregation cpx = 1 The brackets cancel the aggregation complexity
Atomic Atomic 0.1 console.log(3) // ---- Atomic cpx = 0.3 (3 atoms) Applies to each identifier, parameter, keyword, literal, etc.
Nesting Arrays 1.5 a[b[c]]
Nesting Conditions 0.5 if (a) {
if (b) { // ---- Nesting cpx = 0.5
if (c) { // ---- Nesting cpx = 1
}
}
}
Applies to if, else, else if, switch
Nesting Loops 0.5 for (const a of arr) {
for (const b of otherArr) { // ---- Nesting cpx = 0.5
}
}
Applies to for, forEach, do ... while
Nesting Ternaries 1 a = b ? c ? : 0 : 1;
Recursion Recursive methods 3 f(a) {
return f(a + 1);
}
Recursion Callbacks 2 f(a) {
return a(2);
}
Structural Conditions 1 if (a) { ... } Applies to if, else, else if, switch
Structural Functions 1 a.filter(elt => { ... })
Structural Jumps 1 for (const a of arr) {
if (b) {
continue;
}
}
Applies to elements breaking loops
Structural Logic door 1 && or ||
Structural Bit door 1 << or >> or >>> or & or | or ^
Structural Loops 1 for (const a of arr) { ... } Applies to for, forEach, do ... while
Structural Methods 1 a.slice(1) Applies to each method, defined in the current language or in the project itself
Structural Regex 1 /[^.[\]]+/ // ---- Structural cpx = 1
Structural Ternary 1 const a = b ? 0 : 1;

Top

7. How to contribute ?

7.1 Confirm, refute, specify

The estimation of the cognitive complexity will always be a simple approximation. The time required for a human to understand a source code depends of thousands of factors which must be studied severely. Our goal is only to give the better approximation of the measure of the Cognitive Complexity, that is, the Complexity Index.

You can help in many ways by confirming, refuting or specifying the actual mode of computation of the Cognitive Complexity. This page is the receptacle of the different propositions coming from the community. Of course, we accept results coming from research labs or statistic studies, but we accept too the simple feedbacks coming from developers themselves. Each element or idea which is able to improve our algorithm is welcome. If you think that something is wrong in our approach, your opinion is welcome. If you think about a new complexity factor or a new way to estimate some kind of complexity, your idea is welcome too !

Each time the algorithm will be updated, the version indicated at the top of the Table of Weights will be updated too.

7.2 Add new languages

Genese Complexity is available for JS, TS, JSX, TSX and JAVA, but you can "plug" any language into this module. What does it mean ? To be simple, Genese Complexity parses a Json file with a specific format : JsonAst. This format corresponds to a simplified AST (Abstract Syntax Tree) of the source code. So if you want to be able to "plug" your language into Genese Complexity, you "just" need to convert the specific AST structure of your language into JsonAst format. In other words, your AST nodes must "match" with the nodes of the JsonAst format. If your plugin is correct, we will add it to Genese Complexity module.

There are hundreds kinds of TypeScript AST nodes, so it can be fastidious to "bind" all of them to the AST nodes of your language. Fortunately, JsonAst only needs few kinds of nodes; you will find them below.

7.3 Add 'use complexity' values for other languages or libraries

In genese-complexity, the use complexity is defined in Json files defining the use complexity values for methods which can increase significantly the difficulty to understand them. This is an 'in progress' feature, which is only able for now to add use complexity for TypeScript methods. We need at first to complete the list of TypeScript methods which have a significant use complexity. In a second time, we need to do the same for other languages : any help will be appreciated. And last but not least, we need to add use complexity Json files for libraries which are widely used.

Top

8. Specifications

8.1 Kinds of nodes

You will find below the list of all the different kinds of AST nodes. If you want to understand exactly what they mean, you may refer to the TypeScript documentation : for example, the node's kind IfStatement refers to the TypeScript AST node ts.SyntaxKind.IfStatement. The exhaustive list of TypeScript SyntaxKinds are accessible here (from line 77 to 447).

Node Kind Example Comments
AmpersandAmpersandToken && The AND logic door.
ArrayType a: string[]; The declaration of a type which is an array type
ArrowFunction () => { ... } An arrowed function or method
BarBarToken || The OR logic door.
BinaryExpression a > 0
a === b
Comparison between two elements.
Block { .... } Abstract node containing some children nodes, like IfStatement. This node doesn't increase complexity (empty category).
CallExpression a.filter(e => e + 1) Abstract node containing a call to a function. In this example, the CallExpression contains a first child which is a PropertyExpression (a.filter) and a second one which is an ArrowFunction (e => e + 1).
CatchClause try { ... }
catch(error) { ... }
This node is considered as a conditional node and increases the nesting complexity in the same way.
ClassDeclaration class MyClass { ... } Abstract node designating a declaration of a class. This node is the root node of a class. It doesn't increase complexity (empty category).
ConditionalExpression a = b ? 0 : 1; This node is a conditional node and increases the nesting complexity. In this example, the ConditionalExpression node have 5 children : Identifier b, QuestionToken ?, NumericLiteral 0, ColonToken : and NumericLiteral 1.
DoStatement do { ... } Do instruction. Increases the nesting complexity.
ElementAccessExpression a[b] Considered as an array by Genese Complexity. In this example, the ElementAccessExpression is a node with two children : an Identifier a and another Identifier b.
EndOfFileToken ... } The last element of the source code.
ExpressionStatement a = b ? 0 : 1;
a.filter(e => e + 1)
Abstract node containing an expression, like a BinaryExpression or a CallExpression. This node doesn't increase complexity (empty category).
FirstAssignment let a = b === 1 Abstract node corresponding to the EqualsToken in a binary expression. It doesn't increase complexity (empty category).
FirstStatement let a = 3; Abstract node declaring a variable (alias of VariableStatement). This node doesn't increase complexity (empty category).
ForStatement for (let i = 0; i < 2; i++) { ... } For loop. Increases the nesting complexity.
Caution : a.forEach(...) is considered by TypeScript as a PropertyAccessExpression and not as a ForStatement, but Genese Complexity consider it as a classic for loop.
ForInStatement for (let a of arr) { ... } For loop with in statement. Increases the nesting complexity.
ForOfStatement for (let a of arr) { ... } For loop with of statement. Increases the nesting complexity.
FunctionDeclaration function f() { ... } Abstract node designating a declaration of a function. This node doesn't increase complexity (empty category).
FunctionExpression f(function(b) { ... } Abstract node designating a function expression. Increases the nesting complexity.
Identifier f(a) { ... } The node corresponding to the identifier of a variable, a function, etc. In this example, there are two identifiers : f and a. An identifier is considered by Genese Complexity as an atomic node which increases the atomic complexity.
IfStatement if(a) { ... } The IF condition. Increases the nesting complexity.
Keyword return Alias for any language keyword (let, const, var, export, return, ...). Increases atomic complexity.
Literal 2
'a'
Alias for any kind of literal assignment (StringLiteral, NumericLiteral, ...). Increases atomic complexity.
MethodDeclaration myMethod() { ... } Abstract node designating a declaration of a method. This node is the root node of the method. It doesn't increase complexity (empty category).
Parameter myMethod(a) { ... } Abstract node designating a parameter.
Caution : the Parameter a is different than the Identifier a, which is a child of the AST node "Parameter". This node doesn't increase complexity (empty category).
PropertyAccessExpression a.b = 3; Abstract node designating the access to a given property. The first child (a) is the expression and the second (b) is the property. This node doesn't increase complexity (empty category).
RegularExpressionLiteral /a-z/g Regular expression.
SwitchStatement switch(a) { ... } Switch statement. Increases the nesting complexity.
ThisKeyword this.a = 3 This keyword (returning context). Increases the atomic complexity
Union Type a: string | number Union of two types. Increases the atomic complexity
VariableDeclarationList for (const elt of arr) { ... } Abstract node declaring a list of variables. In this example, the VariableDeclarationList is const elt. This node doesn't increase complexity (empty category).
WhileStatement while (a < 10) { ... } While loop. Increases the nesting complexity.

Genese Complexity will consider all the other kinds of nodes as atomic nodes. This means that every node in the JsonAst which is not in the previous list file will add a cognitive complexity corresponding to atomic nodes, as StringLiteral, TrueKeyword, etc. If you don't want to increase complexity for a given kind of node, you will be able to set a property "empty" to true in the corresponding field of the JsonAst file.

8.2 JsonAst specifications

JsonAst

This is the root document (the .json file itself)

  • Fixed fields
Field name Type Required Description
astFolder AstFolder yes The object containing all the information about the folder to analyze

astFolder

Corresponds to a folder to analyze.

  • Fixed fields
Field name Type Required Description
astFiles AstFile[] no The array of AstFile corresponding to the files inside the folder (but not inside its subfolders)
children AstFolder[] no The array of AstFolder corresponding to the subfolders of this AstFolder
path String yes The absolute path of the folder

astFile

Corresponds to a file to analyze.

  • Fixed fields
Field name Type Required Description
astNode AstNode yes The AstNode corresponding to the sourceFile itself (in Typescript, it is ts.SourceFile)
name String yes The name of the file
text String yes The source code of the file, including break lines

astNode

Corresponds to an AST node of the source code of a file.

  • Fixed fields
Field name Type Required Description
children AstNode[] no The array of AstNode corresponding to the children of the AST node
empty Boolean no If true, the corresponding AstNode will not add any complexity
end Integer yes The position of the last character of the AST node in the source code of the file
kind SyntaxKind yes The kind of the AST node
name String yes/no The name of the AST node. This field MUST be present in the following cases: ClassDeclaration, MethodDeclaration, FunctionDeclaration, Parameter, Identifier
pos Integer yes The position of the first character of the AST node in the source code of the file

8.3 Structure of the AST nodes

You must respect some conventions to be able to create JsonAst files correctly interpreted by Genese Complexity.

  • IfStatement

    Supposing to be in this case :

if (a) {
    // ---
} else if (b) {
    // ---
} else {
    // ---
}

Your JsonAst MUST be structured like this :

{
    "kind": "IfStatement",
    "children": [
    	{
    		"kind": "Identifier",
    		"name": "a"
    	},
    	{
    		"kind": "Block"
    	},
    	{
    		"kind": "IfStatement",
    		"children": [
    			{
    				"kind": "Identifier",
    				"name": "b"
    			},
    			{
    				"kind": "Block"
    			},
    			{
    				"kind": "Block"
    			}
    		]
    	}
    ]
}

The AstNode "IfStatement" always have a first son which is what is inside the if brackets and a second son which is what is inside the if condition (the curly brackets). This AstNode MAY has a third son which is the AstNode corresponding to the ElseStatement, like in the example above.

8.4 Exhaustive list of the kinds of node

This list corresponds to the ts.SyntaxKind enum (from line 77 to 447)

    AbstractKeyword
    AmpersandAmpersandToken
    AmpersandEqualsToken
    AmpersandToken
    AnyKeyword
    ArrayBindingPattern
    ArrayLiteralExpression
    ArrayType
    ArrowFunction
    AsExpression
    AsKeyword
    AssertsKeyword
    AsteriskAsteriskEqualsToken
    AsteriskAsteriskToken
    AsteriskEqualsToken
    AsteriskToken
    AsyncKeyword
    AtToken
    AwaitExpression
    AwaitKeyword
    BacktickToken
    BarBarToken
    BarEqualsToken
    BarToken
    BigIntKeyword
    BigIntLiteral
    BinaryExpression
    BindingElement
    Block
    BooleanKeyword
    BreakKeyword
    BreakStatement
    Bundle
    CallExpression
    CallSignature
    CaretEqualsToken
    CaretToken
    CaseBlock
    CaseClause
    CaseKeyword
    CatchClause
    CatchKeyword
    ClassDeclaration
    ClassExpression
    ClassKeyword
    CloseBraceToken
    CloseBracketToken
    CloseParenToken
    ColonToken
    CommaListExpression
    CommaToken
    ComputedPropertyName
    ConditionalExpression
    ConditionalType
    ConflictMarkerTrivia
    ConstKeyword
    Constructor
    ConstructorKeyword
    ConstructorType
    ConstructSignature
    ContinueKeyword
    ContinueStatement
    Count
    DebuggerKeyword
    DebuggerStatement
    DeclareKeyword
    Decorator
    DefaultClause
    DefaultKeyword
    DeleteExpression
    DeleteKeyword
    DoKeyword
    DoStatement
    DotDotDotToken
    DotToken
    ElementAccessExpression
    ElseKeyword
    EmptyStatement
    EndOfDeclarationMarker
    EndOfFileToken
    EnumDeclaration
    EnumKeyword
    EnumMember
    EqualsEqualsEqualsToken
    EqualsEqualsToken
    EqualsGreaterThanToken
    EqualsToken
    ExclamationEqualsEqualsToken
    ExclamationEqualsToken
    ExclamationToken
    ExportAssignment
    ExportDeclaration
    ExportKeyword
    ExportSpecifier
    ExpressionStatement
    ExpressionWithTypeArguments
    ExtendsKeyword
    ExternalModuleReference
    FalseKeyword
    FinallyKeyword
    FirstAssignment
    FirstBinaryOperator
    FirstCompoundAssignment
    FirstFutureReservedWord
    FirstJSDocNode
    FirstJSDocTagNode
    FirstKeyword
    FirstLiteralToken
    FirstNode
    FirstPunctuation
    FirstReservedWord
    FirstStatement
    FirstTemplateToken
    FirstToken
    FirstTriviaToken
    FirstTypeNode
    ForInStatement
    ForKeyword
    ForOfStatement
    ForStatement
    FromKeyword
    FunctionDeclaration
    FunctionExpression
    FunctionKeyword
    FunctionType
    GetAccessor
    GetKeyword
    GlobalKeyword
    GreaterThanEqualsToken
    GreaterThanGreaterThanEqualsToken
    GreaterThanGreaterThanGreaterThanEqualsToken
    GreaterThanGreaterThanGreaterThanToken
    GreaterThanGreaterThanToken
    GreaterThanToken
    HeritageClause
    Identifier
    IfKeyword
    IfStatement
    ImplementsKeyword
    ImportClause
    ImportDeclaration
    ImportEqualsDeclaration
    ImportKeyword
    ImportSpecifier
    ImportType
    IndexedAccessType
    IndexSignature
    InferKeyword
    InferType
    InKeyword
    InputFiles
    InstanceOfKeyword
    InterfaceDeclaration
    InterfaceKeyword
    IntersectionType
    IsKeyword
    JSDocAllType
    JSDocAugmentsTag
    JSDocAuthorTag
    JSDocCallbackTag
    JSDocClassTag
    JSDocComment
    JSDocEnumTag
    JSDocFunctionType
    JSDocImplementsTag
    JSDocNamepathType
    JSDocNonNullableType
    JSDocNullableType
    JSDocOptionalType
    JSDocParameterTag
    JSDocPrivateTag
    JSDocPropertyTag
    JSDocProtectedTag
    JSDocPublicTag
    JSDocReadonlyTag
    JSDocReturnTag
    JSDocSignature
    JSDocTag
    JSDocTemplateTag
    JSDocThisTag
    JSDocTypedefTag
    JSDocTypeExpression
    JSDocTypeLiteral
    JSDocTypeTag
    JSDocUnknownType
    JSDocVariadicType
    JsxAttribute
    JsxAttributes
    JsxClosingElement
    JsxClosingFragment
    JsxElement
    JsxExpression
    JsxFragment
    JsxOpeningElement
    JsxOpeningFragment
    JsxSelfClosingElement
    JsxSpreadAttribute
    JsxText
    JsxTextAllWhiteSpaces
    KeyOfKeyword
    LabeledStatement
    LastAssignment
    LastBinaryOperator
    LastCompoundAssignment
    LastFutureReservedWord
    LastJSDocNode
    LastJSDocTagNode
    LastKeyword
    LastLiteralToken
    LastPunctuation
    LastReservedWord
    LastStatement
    LastTemplateToken
    LastToken
    LastTriviaToken
    LastTypeNode
    LessThanEqualsToken
    LessThanLessThanEqualsToken
    LessThanLessThanToken
    LessThanSlashToken
    LessThanToken
    LetKeyword
    LiteralType
    MappedType
    MergeDeclarationMarker
    MetaProperty
    MethodDeclaration
    MethodSignature
    MinusEqualsToken
    MinusMinusToken
    MinusToken
    MissingDeclaration
    ModuleBlock
    ModuleDeclaration
    ModuleKeyword
    MultiLineCommentTrivia
    NamedExports
    NamedImports
    NamespaceExport
    NamespaceExportDeclaration
    NamespaceImport
    NamespaceKeyword
    NeverKeyword
    NewExpression
    NewKeyword
    NewLineTrivia
    NonNullExpression
    NoSubstitutionTemplateLiteral
    NotEmittedStatement
    NullKeyword
    NumberKeyword
    NumericLiteral
    ObjectBindingPattern
    ObjectKeyword
    ObjectLiteralExpression
    OfKeyword
    OmittedExpression
    OpenBraceToken
    OpenBracketToken
    OpenParenToken
    OptionalType
    PackageKeyword
    Parameter
    ParenthesizedExpression
    ParenthesizedType
    PartiallyEmittedExpression
    PercentEqualsToken
    PercentToken
    PlusEqualsToken
    PlusPlusToken
    PlusToken
    PostfixUnaryExpression
    PrefixUnaryExpression
    PrivateIdentifier
    PrivateKeyword
    PropertyAccessExpression
    PropertyAssignment
    PropertyDeclaration
    PropertySignature
    ProtectedKeyword
    PublicKeyword
    QualifiedName
    QuestionDotToken
    QuestionQuestionToken
    QuestionToken
    ReadonlyKeyword
    RegularExpressionLiteral
    RequireKeyword
    RestType
    ReturnKeyword
    ReturnStatement
    SemicolonClassElement
    SemicolonToken
    SetAccessor
    SetKeyword
    ShebangTrivia
    ShorthandPropertyAssignment
    SingleLineCommentTrivia
    SlashEqualsToken
    SlashToken
    SourceFile
    SpreadAssignment
    SpreadElement
    StaticKeyword
    StringKeyword
    StringLiteral
    SuperKeyword
    SwitchKeyword
    SwitchStatement
    SymbolKeyword
    SyntaxList
    SyntheticExpression
    SyntheticReferenceExpression
    TaggedTemplateExpression
    TemplateExpression
    TemplateHead
    TemplateMiddle
    TemplateSpan
    TemplateTail
    ThisKeyword
    ThisType
    ThrowKeyword
    ThrowStatement
    TildeToken
    TrueKeyword
    TryKeyword
    TryStatement
    TupleType
    TypeAliasDeclaration
    TypeAssertionExpression
    TypeKeyword
    TypeLiteral
    TypeOfExpression
    TypeOfKeyword
    TypeOperator
    TypeParameter
    TypePredicate
    TypeQuery
    TypeReference
    UndefinedKeyword
    UnionType
    UniqueKeyword
    Unknown
    UnknownKeyword
    UnparsedInternalText
    UnparsedPrepend
    UnparsedPrologue
    UnparsedSource
    UnparsedSyntheticReference
    UnparsedText
    VariableDeclaration
    VariableDeclarationList
    VariableStatement
    VarKeyword
    VoidExpression
    VoidKeyword
    WhileKeyword
    WhileStatement
    WhitespaceTrivia
    WithKeyword
    WithStatement
    YieldExpression
    YieldKeyword

Readme

Keywords

none

Package Sidebar

Install

npm i genese-complexity

Weekly Downloads

11

Version

0.0.6

License

MIT

Unpacked Size

3.17 MB

Total Files

1541

Last publish

Collaborators

  • gillesfabre