-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdfa-interpreter.ts
116 lines (100 loc) · 3.04 KB
/
dfa-interpreter.ts
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
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
export class DFANode {
/** Start State */
isStartState: boolean;
/** Accept State */
isAcceptState: boolean;
children: DFAEdge[];
label: string;
private get image(): string {
if (this.isStartState && this.isAcceptState) {
return "start_accept_state.svg";
}
if (this.isAcceptState) {
return "accept_state.svg";
}
if (this.isStartState) {
return "start_state.svg";
}
return "";
}
private get origin(): { x: number, y: number } {
return {
x: this.isStartState ? 10 : 0,
y: 0
};
}
}
export class DFAEdge {
symbol: string;
destination: DFANode;
private get label() {
if (this.symbol.length > 0) {
return this.symbol;
} else {
return "λ";
}
}
}
export class DFAGraph {
nodes: Nodes[];
}
export type Nodes = DFANode;
export type Edges = DFAEdge;
export type Graph = DFAGraph;
export class State {
constructor(public active: DFANode,
public inputLeft: string,
public message: string) {
}
}
export function start(input: DFAGraph, data: string): State | boolean {
let start: DFANode | null = null;
const accepts = new Set<DFANode>();
for (const node of input.nodes) {
if (node.isStartState) {
if (!start) {
start = node;
} else {
throw new Error("Only one start state allowed");
}
}
if (node.isAcceptState) {
accepts.add(node);
}
if (node.children) {
let transitions = new Set<string>();
for (const edge of node.children) {
if (!edge.symbol) {
throw new Error("Lambda transition from " + node.label + " to " + edge.destination.label + " is not allowed");
}
if (edge.symbol.length > 1) {
throw new Error("Edge " + edge.symbol + " must be one symbol");
}
if (transitions.has(edge.symbol)) {
throw new Error("Nondeterministic edge " + edge.symbol + (node.label ? (" from node: " + node.label) : ""));
}
transitions.add(edge.symbol);
}
}
}
if (!start) {
throw new Error("Must have one start state");
}
return new State(start, data, "starting");
}
export function step(current: State): State | boolean {
if (current.inputLeft.length === 0) {
return current.active.isAcceptState === true;
}
const destinations = current.active.children
.filter(edge => edge.symbol === current.inputLeft[0])
.map(edge => edge.destination);
if (destinations.length === 1) {
return new State(destinations[0], current.inputLeft.substr(1),
`transitioning from ${current.active.label} to ${destinations[0].label}`);
} else if (destinations.length === 0) {
return false;
} else {
throw "This is a DFA!";
}
}