Skip to content

Commit 00df71d

Browse files
committed
First version of object access dot operator
1 parent 93c776d commit 00df71d

File tree

8 files changed

+83
-21
lines changed

8 files changed

+83
-21
lines changed

src/Syntax.js

+1
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ const Syntax = {
1313
Punctuator: "Punctuator",
1414
Identifier: "Identifier",
1515
ArraySubscript: "ArraySubscript",
16+
AccessIdentifier: "AccessIdentifier",
1617
Constant: "Constant",
1718
Type: "Type",
1819
Declaration: "Declaration",

src/__tests__/object-literal-spec.js

+18
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,24 @@ const compileAndRun = (src, imports) =>
66
const outputIs = (t, value) => result =>
77
t.is(result.instance.exports.test(), value);
88

9+
test("dot operator", t => {
10+
return compileAndRun(
11+
`
12+
const memory: Memory = { 'initial': 1 };
13+
14+
type TestType = { 'foo': i32, 'bar': i32 };
15+
16+
export function test(): i32 {
17+
let obj: TestType = 0;
18+
19+
obj["foo"] = 42;
20+
obj.bar = 20;
21+
22+
return obj.foo + obj.bar;
23+
}`
24+
).then(outputIs(t, 62));
25+
});
26+
927
test("types and assignment", t => {
1028
return compileAndRun(
1129
`

src/parser/access-identifier.js

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
//@flow
2+
import Syntax from "../Syntax";
3+
import type Context from "./context";
4+
import type { Node } from "../flow/types";
5+
6+
export function accessIdentifier(ctx: Context): Node {
7+
const node = ctx.startNode();
8+
// We're removing first character which is access dot operator
9+
node.value = ctx.token.value.substring(1, ctx.token.value.length);
10+
return ctx.endNode(node, Syntax.StringLiteral);
11+
}
12+
13+
export default accessIdentifier;

src/parser/expression.js

+14
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@ import Syntax from "../Syntax";
33
import operator from "./operator";
44
import constant from "./constant";
55
import stringLiteral from "./string-literal";
6+
import accessIdentifier from "./access-identifier";
67
import builtInType from "./builtin-type";
78
import { getAssociativty, getPrecedence } from "./introspection";
89
import maybeIdentifier from "./maybe-identifier";
@@ -174,6 +175,19 @@ break;
174175
eatFunctionCall = true;
175176
operands.push(maybeIdentifier(ctx));
176177
break;
178+
case Syntax.AccessIdentifier:
179+
// We're manually creating StringLiteral from AccessIdentifier
180+
operators.push({
181+
type: "Punctuator",
182+
value: "[",
183+
start: {},
184+
end: {}
185+
});
186+
operands.push(accessIdentifier(ctx));
187+
eatUntil(isLSqrBracket);
188+
consume();
189+
eatFunctionCall = false;
190+
break;
177191
case Syntax.StringLiteral:
178192
eatFunctionCall = false;
179193
operands.push(stringLiteral(ctx));

src/parser/maybe-assignment.js

+10-10
Original file line numberDiff line numberDiff line change
@@ -1,18 +1,18 @@
11
//@flow
2-
import Syntax from "../Syntax";
3-
import maybeIdentifier from "./maybe-identifier";
4-
import memoryStore from "./memory-store";
5-
import expression from "./expression";
6-
import type Context from "./context";
7-
import type { NodeType } from "../flow/types";
2+
import Syntax from '../Syntax';
3+
import maybeIdentifier from './maybe-identifier';
4+
import memoryStore from './memory-store';
5+
import expression from './expression';
6+
import type Context from './context';
7+
import type { NodeType } from '../flow/types';
88

99
// It is easier to parse assignment this way as we need to maintain a valid type
1010
// through out the right-hand side of the expression
1111
function maybeAssignment(ctx: Context): NodeType {
12-
const nextValue = ctx.stream.peek().value;
13-
if (nextValue === "[") {
14-
return memoryStore(ctx);
15-
}
12+
const { type, value } = ctx.stream.peek();
13+
if (value === '[' || type === Syntax.AccessIdentifier) {
14+
return memoryStore(ctx);
15+
}
1616

1717
const target = maybeIdentifier(ctx);
1818
if (target.Type === Syntax.FunctionCall) {
+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
// @flow
2+
3+
import token from "../token";
4+
import Syntax from "../../Syntax";
5+
import { maybeIdentifier } from "../identifier";
6+
7+
const maybeAccessIdentifier = char => {
8+
if (char === ".") {
9+
return maybeIdentifier;
10+
}
11+
return null;
12+
};
13+
14+
export default token(maybeAccessIdentifier, Syntax.AccessIdentifier);

src/tokenizer/identifier/index.js

+11-11
Original file line numberDiff line numberDiff line change
@@ -1,15 +1,15 @@
1-
import token from "../token";
2-
import punctuator from "../punctuator";
3-
import constant from "../constant";
4-
import string from "../string";
5-
import Syntax from "../../Syntax";
1+
import token from '../token';
2+
import punctuator from '../punctuator';
3+
import constant from '../constant';
4+
import string from '../string';
5+
import Syntax from '../../Syntax';
66

7-
const parse = char => {
7+
export const maybeIdentifier = char => {
88
// Don't allow these
9-
if (!string(char) && !punctuator(char) && !constant(char) && char !== " ") {
10-
return parse;
11-
}
9+
if (!string(char) && !punctuator(char) && !constant(char) && char !== ' ') {
10+
return maybeIdentifier;
11+
}
1212
return null;
1313
};
14-
const tokenParser = token(parse, Syntax.Identifier);
15-
export default tokenParser;
14+
15+
export default token(maybeIdentifier, Syntax.Identifier);

src/tokenizer/index.js

+2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
import Stream from "../utils/stream";
22
import punctuator from "./punctuator";
33
import constant from "./constant";
4+
import accessIdentifier from "./access-identifier";
45
import identifier from "./identifier";
56
import keyword from "./keyword";
67
import string from "./string";
@@ -11,6 +12,7 @@ class Tokenizer {
1112
constructor(
1213
stream,
1314
parsers = [
15+
accessIdentifier,
1416
punctuator,
1517
constant,
1618
identifier,

0 commit comments

Comments
 (0)