Skip to content

Commit c68acd8

Browse files
authored
RFC: Lexing is Greedy (#2163)
Adds the test cases described in graphql/graphql-spec#599 Makes the lookahead restriction change necessary for the new tests to pass for numbers, all other tests are already passing.
1 parent 5c42dc6 commit c68acd8

File tree

2 files changed

+138
-0
lines changed

2 files changed

+138
-0
lines changed

src/language/__tests__/lexer-test.js

+128
Original file line numberDiff line numberDiff line change
@@ -49,6 +49,65 @@ describe('Lexer', () => {
4949
});
5050
});
5151

52+
it('tracks line breaks', () => {
53+
expect(lexOne('foo')).to.contain({
54+
kind: TokenKind.NAME,
55+
start: 0,
56+
end: 3,
57+
line: 1,
58+
column: 1,
59+
value: 'foo',
60+
});
61+
expect(lexOne('\nfoo')).to.contain({
62+
kind: TokenKind.NAME,
63+
start: 1,
64+
end: 4,
65+
line: 2,
66+
column: 1,
67+
value: 'foo',
68+
});
69+
expect(lexOne('\rfoo')).to.contain({
70+
kind: TokenKind.NAME,
71+
start: 1,
72+
end: 4,
73+
line: 2,
74+
column: 1,
75+
value: 'foo',
76+
});
77+
expect(lexOne('\r\nfoo')).to.contain({
78+
kind: TokenKind.NAME,
79+
start: 2,
80+
end: 5,
81+
line: 2,
82+
column: 1,
83+
value: 'foo',
84+
});
85+
expect(lexOne('\n\rfoo')).to.contain({
86+
kind: TokenKind.NAME,
87+
start: 2,
88+
end: 5,
89+
line: 3,
90+
column: 1,
91+
value: 'foo',
92+
});
93+
expect(lexOne('\r\r\n\nfoo')).to.contain({
94+
kind: TokenKind.NAME,
95+
start: 4,
96+
end: 7,
97+
line: 4,
98+
column: 1,
99+
value: 'foo',
100+
});
101+
expect(lexOne('\n\n\r\rfoo')).to.contain({
102+
kind: TokenKind.NAME,
103+
start: 4,
104+
end: 7,
105+
line: 5,
106+
column: 1,
107+
value: 'foo',
108+
});
109+
});
110+
52111
it('records line and column', () => {
53112
expect(lexOne('\n \r\n \r foo\n')).to.contain({
54113
kind: TokenKind.NAME,
@@ -164,6 +223,13 @@ describe('Lexer', () => {
164223
});
165224

166225
it('lexes strings', () => {
226+
expect(lexOne('""')).to.contain({
227+
kind: TokenKind.STRING,
228+
start: 0,
229+
end: 2,
230+
value: '',
231+
});
232+
167233
expect(lexOne('"simple"')).to.contain({
168234
kind: TokenKind.STRING,
169235
start: 0,
@@ -210,6 +276,10 @@ describe('Lexer', () => {
210276
it('lex reports useful string errors', () => {
211277
expectSyntaxError('"', 'Unterminated string.', { line: 1, column: 2 });
212278

279+
expectSyntaxError('"""', 'Unterminated string.', { line: 1, column: 4 });
280+
281+
expectSyntaxError('""""', 'Unterminated string.', { line: 1, column: 5 });
282+
213283
expectSyntaxError('"no end quote', 'Unterminated string.', {
214284
line: 1,
215285
column: 14,
@@ -287,6 +357,13 @@ describe('Lexer', () => {
287357
});
288358

289359
it('lexes block strings', () => {
360+
expect(lexOne('""""""')).to.contain({
361+
kind: TokenKind.BLOCK_STRING,
362+
start: 0,
363+
end: 6,
364+
value: '',
365+
});
366+
290367
expect(lexOne('"""simple"""')).to.contain({
291368
kind: TokenKind.BLOCK_STRING,
292369
start: 0,
@@ -538,6 +615,20 @@ describe('Lexer', () => {
538615
column: 2,
539616
});
540617

618+
expectSyntaxError('01', 'Invalid number, unexpected digit after 0: "1".', {
619+
line: 1,
620+
column: 2,
621+
});
622+
623+
expectSyntaxError(
624+
'01.23',
625+
'Invalid number, unexpected digit after 0: "1".',
626+
{
627+
line: 1,
628+
column: 2,
629+
},
630+
);
631+
541632
expectSyntaxError('+1', 'Cannot parse the unexpected character "+".', {
542633
line: 1,
543634
column: 1,
@@ -548,6 +639,16 @@ describe('Lexer', () => {
548639
column: 3,
549640
});
550641

642+
expectSyntaxError('1e', 'Invalid number, expected digit but got: <EOF>.', {
643+
line: 1,
644+
column: 3,
645+
});
646+
647+
expectSyntaxError('1E', 'Invalid number, expected digit but got: <EOF>.', {
648+
line: 1,
649+
column: 3,
650+
});
651+
551652
expectSyntaxError('1.e1', 'Invalid number, expected digit but got: "e".', {
552653
line: 1,
553654
column: 3,
@@ -578,6 +679,33 @@ describe('Lexer', () => {
578679
line: 1,
579680
column: 5,
580681
});
682+
683+
expectSyntaxError(
684+
'1.2e3e',
685+
'Invalid number, expected digit but got: "e".',
686+
{
687+
line: 1,
688+
column: 6,
689+
},
690+
);
691+
692+
expectSyntaxError(
693+
'1.2e3.4',
694+
'Invalid number, expected digit but got: ".".',
695+
{
696+
line: 1,
697+
column: 6,
698+
},
699+
);
700+
701+
expectSyntaxError(
702+
'1.23.4',
703+
'Invalid number, expected digit but got: ".".',
704+
{
705+
line: 1,
706+
column: 5,
707+
},
708+
);
581709
});
582710

583711
it('lexes punctuation', () => {

src/language/lexer.js

+10
Original file line numberDiff line numberDiff line change
@@ -444,6 +444,16 @@ function readNumber(source, start, firstCode, line, col, prev): Token {
444444
code = body.charCodeAt(++position);
445445
}
446446
position = readDigits(source, position, code);
447+
code = body.charCodeAt(position);
448+
}
449+
450+
// Numbers cannot be followed by . or e
451+
if (code === 46 || code === 69 || code === 101) {
452+
throw syntaxError(
453+
source,
454+
position,
455+
`Invalid number, expected digit but got: ${printCharCode(code)}.`,
456+
);
447457
}
448458
449459
return new Tok(

0 commit comments

Comments
 (0)