Skip to content

Commit ecd5711

Browse files
committed
型チェッカーを強化
1 parent c5228a7 commit ecd5711

File tree

9 files changed

+589
-73
lines changed

9 files changed

+589
-73
lines changed

features/core/errors/AiTypeError.ts

+18
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,8 @@ export enum AiTypeErrorKind {
88
MissingArgumentError,
99
InvalidArgumentError,
1010
CanNotAssignToImmutableVariable,
11+
UnDeclaredVariable,
12+
CanNotReadProperty,
1113
}
1214

1315
export class AiTypeError extends Error {
@@ -64,3 +66,19 @@ export class AiCanNotAssignToImmutableVariableError extends AiTypeError {
6466
super(AiTypeErrorKind.CanNotAssignToImmutableVariable, location);
6567
}
6668
}
69+
70+
export class AiUnDeclaredVariableError extends AiTypeError {
71+
constructor(public name: string, location: SourceLocation) {
72+
super(AiTypeErrorKind.UnDeclaredVariable, location);
73+
}
74+
}
75+
76+
export class AiCanNotReadPropertyError extends AiTypeError {
77+
constructor(
78+
public targetType: string,
79+
public name: string,
80+
location: SourceLocation
81+
) {
82+
super(AiTypeErrorKind.CanNotReadProperty, location);
83+
}
84+
}

features/core/errors/index.ts

+5-1
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,8 @@ import {
1111
AiMissingArgumentError,
1212
AiInvalidArgumentError,
1313
AiCanNotAssignToImmutableVariableError,
14+
AiUnDeclaredVariableError,
15+
AiCanNotReadPropertyError,
1416
} from "./AiTypeError.js";
1517

1618
export type ParserError =
@@ -25,4 +27,6 @@ export type TypeError =
2527
| AiCanNotCallError
2628
| AiMissingArgumentError
2729
| AiInvalidArgumentError
28-
| AiCanNotAssignToImmutableVariableError;
30+
| AiCanNotAssignToImmutableVariableError
31+
| AiUnDeclaredVariableError
32+
| AiCanNotReadPropertyError;

features/core/i18n/aiscript/typeError.ts

+19-1
Original file line numberDiff line numberDiff line change
@@ -2,10 +2,12 @@ import {
22
AiAlreadyDeclaredVariableError,
33
AiCanNotAssignToImmutableVariableError,
44
AiCanNotCallError,
5+
AiCanNotReadPropertyError,
56
AiInvalidArgumentError,
67
AiMissingArgumentError,
78
AiNotAssignableTypeError,
89
AiTypeErrorKind,
10+
AiUnDeclaredVariableError,
911
} from "../../errors/AiTypeError.js";
1012
import { TypeError } from "../../index.js";
1113
import { I18nLocalizerResult, I18nMessage, I18nMessages } from "../core.js";
@@ -31,6 +33,12 @@ export const typeErrorMessage = {
3133
CanNotAssignToImmutableVariable(name: string) {
3234
return `変数\`${name}\`は不変なので、新たに代入することはできません。`;
3335
},
36+
UnDeclaredVariable(name: string) {
37+
return `変数\`${name}\`は定義されていません。`;
38+
},
39+
CanNotReadProperty(target: string, name: string) {
40+
return `型\`${target}\`から\`${name}\`プロパティを読み取れません。`;
41+
},
3442
} satisfies I18nMessage<keyof typeof AiTypeErrorKind>,
3543
} satisfies I18nMessages;
3644

@@ -52,6 +60,10 @@ export const typeErrorLocalizer = (
5260
return ["typing.MissingArgumentError", [error.pos, error.expectType]];
5361
} else if (error instanceof AiCanNotAssignToImmutableVariableError) {
5462
return ["typing.CanNotAssignToImmutableVariable", [error.name]];
63+
} else if (error instanceof AiUnDeclaredVariableError) {
64+
return ["typing.UnDeclaredVariable", [error.name]];
65+
} else if (error instanceof AiCanNotReadPropertyError) {
66+
return ["typing.CanNotReadProperty", [error.targetType, error.name]];
5567
}
5668

5769
switch (error.kind) {
@@ -71,9 +83,15 @@ export const typeErrorLocalizer = (
7183
];
7284

7385
case AiTypeErrorKind.MissingArgumentError:
74-
return ["typing.MissingArgumentError", [-1, "<:unknown:>"]];
86+
return ["typing.MissingArgumentError", [NaN, "<:unknown:>"]];
7587

7688
case AiTypeErrorKind.CanNotAssignToImmutableVariable:
7789
return ["typing.CanNotAssignToImmutableVariable", ["<:unknown:>"]];
90+
91+
case AiTypeErrorKind.UnDeclaredVariable:
92+
return ["typing.UnDeclaredVariable", ["<:unknown:>"]];
93+
94+
case AiTypeErrorKind.CanNotReadProperty:
95+
return ["typing.CanNotReadProperty", ["<:unknown:>", "<:unknown:>"]];
7896
}
7997
};

features/core/server/LanguageServer.ts

+15-8
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,8 @@ import { parserErrorLocalizer } from "../i18n/aiscript/syntaxError.js";
1616
import { aiLocation2LspPosition } from "./location.js";
1717
import { TypeChecker } from "../typing/TypeChecker.js";
1818
import { typeErrorLocalizer } from "../i18n/aiscript/typeError.js";
19+
import { AiScriptError } from "@syuilo/aiscript/error.js";
20+
import { SourceLocation } from "../parser/SourceRange.js";
1921

2022
export * as Lsp from "vscode-languageserver";
2123
export * as TextDocument from "vscode-languageserver-textdocument";
@@ -61,7 +63,9 @@ export class LanguageServer {
6163
this.conn.console.log("did change content - " + change.document.uri);
6264

6365
try {
66+
this.conn.console.log("start parse");
6467
const ast = this.parser.parseFromString(text, parserErrors);
68+
this.conn.console.log("end parse");
6569
for (const error of parserErrors) {
6670
const message = this.i18n.localize(parserErrorLocalizer, error);
6771

@@ -77,8 +81,10 @@ export class LanguageServer {
7781
}
7882

7983
const scope = this.typeChecker.globalScope.copy();
84+
this.conn.console.log("start type check");
8085
this.typeChecker.preRunBlock(ast, scope, typeErrors);
8186
this.typeChecker.runBlock(ast, scope, typeErrors);
87+
this.conn.console.log("end type check");
8288

8389
for (const error of typeErrors) {
8490
const message = this.i18n.localize(typeErrorLocalizer, error);
@@ -99,20 +105,21 @@ export class LanguageServer {
99105
diagnostics: diagnostics,
100106
});
101107
} catch (err) {
108+
const loc: SourceLocation = (err instanceof AiScriptError
109+
? err.loc
110+
: null) ?? {
111+
column: 0,
112+
line: 0,
113+
};
114+
102115
this.conn.sendDiagnostics({
103116
uri: change.document.uri,
104117
diagnostics: [
105118
{
106119
message: "" + err,
107120
range: {
108-
start: {
109-
character: 0,
110-
line: 0,
111-
},
112-
end: {
113-
character: 0,
114-
line: 9,
115-
},
121+
start: aiLocation2LspPosition(loc),
122+
end: aiLocation2LspPosition(loc),
116123
},
117124
},
118125
],

features/core/typing/Scope.ts

+12-4
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,14 @@
11
import { Ast } from "@syuilo/aiscript/index.js";
22
import { AiAlreadyDeclaredVariableError } from "../errors/AiTypeError.js";
3-
import { TypeValue } from "./TypeValue.js";
3+
import { TypeValue, ident, primitiveType } from "./TypeValue.js";
44
import { Variable } from "./Variable.js";
55

66
export class Scope {
77
constructor(
88
private parent?: Scope,
99
private types = new Map<string, TypeValue>(),
10-
private variables = new Map<string, Variable>()
10+
private variables = new Map<string, Variable>(),
11+
private overridedVariables = new Map<string, Variable>()
1112
) {}
1213

1314
copy() {
@@ -35,16 +36,23 @@ export class Scope {
3536
}
3637

3738
defineVariable(ident: Ast.Identifier, variable: Variable) {
38-
if (this.variables.has(ident.name) == null) {
39+
if (this.variables.has(ident.name)) {
3940
return new AiAlreadyDeclaredVariableError(ident.name, ident.loc);
4041
}
4142

4243
this.variables.set(ident.name, variable);
4344
}
4445

46+
overrideVariable(ident: Ast.Identifier, variable: Variable) {
47+
this.overridedVariables.set(ident.name, variable);
48+
}
49+
4550
getVariable(ident: Ast.Identifier): Variable | null {
4651
return (
47-
this.parent?.getVariable(ident) ?? this.variables.get(ident.name) ?? null
52+
this.overridedVariables.get(ident.name) ??
53+
this.parent?.getVariable(ident) ??
54+
this.variables.get(ident.name) ??
55+
null
4856
);
4957
}
5058

0 commit comments

Comments
 (0)