aboutsummaryrefslogtreecommitdiffstats
path: root/packages/utils/src/abi_encoder/utils/signatureParser.ts
blob: 315784cead29e2fbe37c4aaa65b60a66d2b673cc (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
import { DataItem } from 'ethereum-types';
import * as _ from 'lodash';

/**
 * Returns an array of DataItem's corresponding to the input signature.
 * A signature can be in two forms: '<DataItem.type>' or '(<DataItem1.type>, <DataItem2.type>, ...)
 * An example of the first form would be 'address' or 'uint256'
 * An example of the second form would be '(address, uint256)'
 * Signatures can also include a name field, for example: 'foo address' or '(foo address, bar uint256)'
 * @param signature of input DataItems
 * @return DataItems derived from input signature
 */
export function generateDataItemsFromSignature(signature: string): DataItem[] {
    let trimmedSignature = signature;
    if (signature.startsWith('(')) {
        if (!signature.endsWith(')')) {
            throw new Error(`Failed to generate data item. Must end with ')'`);
        }
        trimmedSignature = signature.substr(1, signature.length - 2);
    }
    trimmedSignature += ',';
    let isCurrTokenArray = false;
    let currTokenArrayModifier = '';
    let isParsingArrayModifier = false;
    let currToken = '';
    let parenCount = 0;
    let currTokenName = '';
    const dataItems: DataItem[] = [];
    for (const char of trimmedSignature) {
        // Tokenize the type string while keeping track of parentheses.
        switch (char) {
            case '(':
                parenCount += 1;
                currToken += char;
                break;
            case ')':
                parenCount -= 1;
                currToken += char;
                break;
            case '[':
                if (parenCount === 0) {
                    isParsingArrayModifier = true;
                    isCurrTokenArray = true;
                    currTokenArrayModifier += '[';
                } else {
                    currToken += char;
                }
                break;
            case ']':
                if (parenCount === 0) {
                    isParsingArrayModifier = false;
                    currTokenArrayModifier += ']';
                } else {
                    currToken += char;
                }
                break;
            case ' ':
                if (parenCount === 0) {
                    currTokenName = currToken;
                    currToken = '';
                } else {
                    currToken += char;
                }
                break;
            case ',':
                if (parenCount === 0) {
                    // Generate new DataItem from token
                    const components = currToken.startsWith('(') ? generateDataItemsFromSignature(currToken) : [];
                    const isTuple = !_.isEmpty(components);
                    const dataItem: DataItem = { name: currTokenName, type: '' };
                    if (isTuple) {
                        dataItem.type = 'tuple';
                        dataItem.components = components;
                    } else {
                        dataItem.type = currToken;
                    }
                    if (isCurrTokenArray) {
                        dataItem.type += currTokenArrayModifier;
                    }
                    dataItems.push(dataItem);
                    // reset token state
                    currTokenName = '';
                    currToken = '';
                    isCurrTokenArray = false;
                    currTokenArrayModifier = '';
                    break;
                } else {
                    currToken += char;
                    break;
                }
            default:
                if (isParsingArrayModifier) {
                    currTokenArrayModifier += char;
                } else {
                    currToken += char;
                }
                break;
        }
    }
    return dataItems;
}