nadesiko3 3.3.48 → 3.3.49

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (105) hide show
  1. package/core/.editorconfig +6 -0
  2. package/core/.eslintrc.cjs +33 -0
  3. package/core/.github/dependabot.yml +7 -0
  4. package/core/.github/workflows/nodejs.yml +37 -0
  5. package/core/.github/workflows/super-linter.yml +61 -0
  6. package/core/.github/workflows/textlint.yml +199 -0
  7. package/core/LICENSE +21 -0
  8. package/core/README.md +66 -0
  9. package/core/batch/build_nako_version.nako3 +42 -0
  10. package/core/command/snako.mjs +105 -0
  11. package/core/command/snako.mts +116 -0
  12. package/core/index.mjs +21 -0
  13. package/core/index.mts +21 -0
  14. package/core/package.json +47 -0
  15. package/core/sample/hello.nako3 +7 -0
  16. package/core/sample/hoge.mjs +4 -0
  17. package/core/sample/hoge.mts +6 -0
  18. package/core/src/nako3.mjs +858 -0
  19. package/core/src/nako3.mts +967 -0
  20. package/core/src/nako_colors.mjs +78 -0
  21. package/core/src/nako_colors.mts +86 -0
  22. package/core/src/nako_core_version.mjs +8 -0
  23. package/core/src/nako_core_version.mts +19 -0
  24. package/core/src/nako_csv.mjs +185 -0
  25. package/core/src/nako_csv.mts +188 -0
  26. package/core/src/nako_errors.mjs +173 -0
  27. package/core/src/nako_errors.mts +197 -0
  28. package/core/src/nako_from_dncl.mjs +255 -0
  29. package/core/src/nako_from_dncl.mts +250 -0
  30. package/core/src/nako_gen.mjs +1648 -0
  31. package/core/src/nako_gen.mts +1719 -0
  32. package/core/src/nako_gen_async.mjs +1659 -0
  33. package/core/src/nako_gen_async.mts +1732 -0
  34. package/core/src/nako_global.mjs +107 -0
  35. package/core/src/nako_global.mts +138 -0
  36. package/core/src/nako_indent.mjs +445 -0
  37. package/core/src/nako_indent.mts +492 -0
  38. package/core/src/nako_josi_list.mjs +38 -0
  39. package/core/src/nako_josi_list.mts +45 -0
  40. package/core/src/nako_lex_rules.mjs +253 -0
  41. package/core/src/nako_lex_rules.mts +260 -0
  42. package/core/src/nako_lexer.mjs +609 -0
  43. package/core/src/nako_lexer.mts +612 -0
  44. package/core/src/nako_logger.mjs +199 -0
  45. package/core/src/nako_logger.mts +232 -0
  46. package/core/src/nako_parser3.mjs +2439 -0
  47. package/core/src/nako_parser3.mts +2195 -0
  48. package/core/src/nako_parser_base.mjs +370 -0
  49. package/core/src/nako_parser_base.mts +370 -0
  50. package/core/src/nako_parser_const.mjs +37 -0
  51. package/core/src/nako_parser_const.mts +37 -0
  52. package/core/src/nako_prepare.mjs +304 -0
  53. package/core/src/nako_prepare.mts +315 -0
  54. package/core/src/nako_reserved_words.mjs +38 -0
  55. package/core/src/nako_reserved_words.mts +38 -0
  56. package/core/src/nako_source_mapping.mjs +207 -0
  57. package/core/src/nako_source_mapping.mts +262 -0
  58. package/core/src/nako_test.mjs +37 -0
  59. package/core/src/nako_types.mjs +25 -0
  60. package/core/src/nako_types.mts +151 -0
  61. package/core/src/plugin_csv.mjs +49 -0
  62. package/core/src/plugin_csv.mts +50 -0
  63. package/core/src/plugin_math.mjs +328 -0
  64. package/core/src/plugin_math.mts +326 -0
  65. package/core/src/plugin_promise.mjs +91 -0
  66. package/core/src/plugin_promise.mts +91 -0
  67. package/core/src/plugin_system.mjs +2832 -0
  68. package/core/src/plugin_system.mts +2690 -0
  69. package/core/src/plugin_test.mjs +34 -0
  70. package/core/src/plugin_test.mts +34 -0
  71. package/core/test/array_test.mjs +34 -0
  72. package/core/test/basic_test.mjs +344 -0
  73. package/core/test/calc_test.mjs +140 -0
  74. package/core/test/core_module_test.mjs +23 -0
  75. package/core/test/debug_test.mjs +16 -0
  76. package/core/test/dncl_test.mjs +94 -0
  77. package/core/test/error_message_test.mjs +210 -0
  78. package/core/test/error_test.mjs +16 -0
  79. package/core/test/flow_test.mjs +373 -0
  80. package/core/test/func_call.mjs +160 -0
  81. package/core/test/func_test.mjs +149 -0
  82. package/core/test/indent_test.mjs +364 -0
  83. package/core/test/lex_test.mjs +168 -0
  84. package/core/test/literal_test.mjs +73 -0
  85. package/core/test/nako_lexer_test.mjs +35 -0
  86. package/core/test/nako_logger_test.mjs +76 -0
  87. package/core/test/nako_logger_test.mts +78 -0
  88. package/core/test/plugin_csv_test.mjs +38 -0
  89. package/core/test/plugin_promise_test.mjs +18 -0
  90. package/core/test/plugin_system_test.mjs +630 -0
  91. package/core/test/prepare_test.mjs +96 -0
  92. package/core/test/re_test.mjs +22 -0
  93. package/core/test/side_effects_test.mjs +92 -0
  94. package/core/test/variable_scope_test.mjs +149 -0
  95. package/core/tsconfig.json +101 -0
  96. package/package.json +4 -2
  97. package/release/_hash.txt +12 -12
  98. package/release/_script-tags.txt +14 -14
  99. package/release/editor.js +1 -1
  100. package/release/stats.json +1 -1
  101. package/release/version.js +1 -1
  102. package/release/wnako3.js +1 -1
  103. package/src/nako_version.mjs +2 -2
  104. package/src/nako_version.mts +2 -2
  105. package/test/async/async_basic_test.mjs +3 -3
@@ -0,0 +1,370 @@
1
+ import { NakoLogger } from './nako_logger.mjs'
2
+ import { FuncList, FuncListItem, Token, Ast, SourceMap, NewEmptyToken } from './nako_types.mjs'
3
+
4
+ /**
5
+ * なでしこの構文解析のためのユーティリティクラス
6
+ */
7
+ export class NakoParserBase {
8
+ public logger: NakoLogger;
9
+ protected stackList: any[];
10
+ protected tokens: Token[];
11
+ protected stack: any[];
12
+ protected index: number;
13
+ protected y: any[];
14
+ public modName: string;
15
+ public modList: string[];
16
+ public funclist: FuncList;
17
+ protected funcLevel: number;
18
+ protected usedAsyncFn: boolean;
19
+ protected localvars: FuncList;
20
+ public genMode: string;
21
+ protected arrayIndexFrom: number;
22
+ protected flagReverseArrayIndex: boolean;
23
+ protected flagCheckArrayInit: boolean;
24
+ protected recentlyCalledFunc: FuncListItem[];
25
+
26
+ constructor (logger: NakoLogger) {
27
+ this.logger = logger
28
+ this.stackList = [] // 関数定義の際にスタックが混乱しないように整理する
29
+ this.tokens = []
30
+ /** @type {import('./nako3.mjs').Ast[]} */
31
+ this.stack = []
32
+ this.index = 0
33
+ /** トークン出現チェック(accept関数)に利用する
34
+ * @type {import('./nako3.mjs').Ast[]}
35
+ */
36
+ this.y = []
37
+ /** モジュル名 @type {string} */
38
+ this.modName = 'inline'
39
+ /**
40
+ * 利用するモジュールの名前一覧
41
+ * @type {Array<string>}
42
+ */
43
+ this.modList = []
44
+ /** グローバル変数・関数の確認用 */
45
+ this.funclist = {}
46
+ this.funcLevel = 0
47
+ this.usedAsyncFn = false // asyncFnの呼び出しがあるかどうか
48
+ /**
49
+ * ローカル変数の確認用
50
+ * @type {Object.<string,Object>}
51
+ */
52
+ this.localvars = { 'それ': { type: 'var', value: '' } }
53
+ /** コード生成器の名前 @type {string} */
54
+ this.genMode = 'sync' // #637
55
+ /** 配列のインデックスが先頭要素(#1140) @type {int} */
56
+ this.arrayIndexFrom = 0
57
+ /** 配列のインデックス順序を反対にするか(#1140) @type {boolean} */
58
+ this.flagReverseArrayIndex = false
59
+ /** 配列を自動的に初期化するか(#1140) @type {boolean} */
60
+ this.flagCheckArrayInit = false
61
+ /** 最近呼び出した関数(余剰エラーの報告に使う) */
62
+ this.recentlyCalledFunc = []
63
+
64
+ this.init()
65
+ }
66
+
67
+ init () {
68
+ this.funclist = {} // 関数の一覧
69
+ this.reset()
70
+ }
71
+
72
+ reset () {
73
+ this.tokens = [] // 字句解析済みのトークンの一覧を保存
74
+ this.index = 0 // tokens[] のどこまで読んだかを管理する
75
+ this.stack = [] // 計算用のスタック ... 直接は操作せず、pushStack() popStack() を介して使う
76
+ this.y = [] // accept()で解析済みのトークンを配列で得るときに使う
77
+ this.genMode = 'sync' // #637, #1056
78
+ }
79
+
80
+ setFuncList (funclist: FuncList) {
81
+ this.funclist = funclist
82
+ }
83
+
84
+ /**
85
+ * 特定の助詞を持つ要素をスタックから一つ下ろす、指定がなければ末尾を下ろす
86
+ * @param {string[]} josiList 下ろしたい助詞の配列
87
+ */
88
+ popStack (josiList: string[]|undefined = undefined): Ast | null {
89
+ if (!josiList) {
90
+ const t = this.stack.pop()
91
+ if (t) { return t }
92
+ return null
93
+ }
94
+
95
+ // josiList にマッチする助詞を探す
96
+ for (let i = 0; i < this.stack.length; i++) {
97
+ const t = this.stack[i]
98
+ if (josiList.length === 0 || josiList.indexOf(t.josi) >= 0) {
99
+ this.stack.splice(i, 1) // remove stack
100
+ this.logger.trace('POP :' + JSON.stringify(t))
101
+ return t
102
+ }
103
+ }
104
+ // 該当する助詞が見つからなかった場合
105
+ return null
106
+ }
107
+
108
+ /**
109
+ * saveStack と loadStack は対で使う。
110
+ * 関数定義などでスタックが混乱しないように配慮するためのもの
111
+ */
112
+ saveStack () {
113
+ this.stackList.push(this.stack)
114
+ this.stack = []
115
+ }
116
+
117
+ loadStack () {
118
+ this.stack = this.stackList.pop()
119
+ }
120
+
121
+ /** 変数名を探す
122
+ * @param {string} name
123
+ * @returns {any}変数名の情報
124
+ */
125
+ findVar (name: string): any {
126
+ // ローカル変数?
127
+ if (this.localvars[name]) {
128
+ return {
129
+ name,
130
+ scope: 'local',
131
+ info: this.localvars[name]
132
+ }
133
+ }
134
+ // モジュール名を含んでいる?
135
+ if (name.indexOf('__') >= 0) {
136
+ if (this.funclist[name]) {
137
+ return {
138
+ name,
139
+ scope: 'global',
140
+ info: this.funclist[name]
141
+ }
142
+ } else { return undefined }
143
+ }
144
+ // グローバル変数(自身)?
145
+ const gnameSelf = `${this.modName}__${name}`
146
+ if (this.funclist[gnameSelf]) {
147
+ return {
148
+ name: gnameSelf,
149
+ scope: 'global',
150
+ info: this.funclist[gnameSelf]
151
+ }
152
+ }
153
+ // グローバル変数(モジュールを検索)?
154
+ for (const mod of this.modList) {
155
+ const gname = `${mod}__${name}`
156
+ if (this.funclist[gname]) {
157
+ return {
158
+ name: gname,
159
+ scope: 'global',
160
+ info: this.funclist[gname]
161
+ }
162
+ }
163
+ }
164
+ // システム変数 (funclistを普通に検索)
165
+ if (this.funclist[name]) {
166
+ return {
167
+ name,
168
+ scope: 'system',
169
+ info: this.funclist[name]
170
+ }
171
+ }
172
+ return undefined
173
+ }
174
+
175
+ /**
176
+ * 計算用に要素をスタックに積む
177
+ */
178
+ pushStack (item: any) {
179
+ this.logger.debug('PUSH:' + JSON.stringify(item))
180
+ this.stack.push(item)
181
+ }
182
+
183
+ /**
184
+ * トークンの末尾に達したか
185
+ */
186
+ isEOF (): boolean {
187
+ return (this.index >= this.tokens.length)
188
+ }
189
+
190
+ getIndex (): number {
191
+ return this.index
192
+ }
193
+
194
+ /**
195
+ * カーソル位置にある単語の型を確かめる
196
+ */
197
+ check (ttype: string): boolean {
198
+ return (this.tokens[this.index].type === ttype)
199
+ }
200
+
201
+ /**
202
+ * カーソル位置以降にある単語の型を確かめる 2単語以上に対応
203
+ * @param a [単語1の型, 単語2の型, ... ]
204
+ */
205
+ check2 (a: any[]): boolean {
206
+ for (let i = 0; i < a.length; i++) {
207
+ const idx = i + this.index
208
+ if (this.tokens.length <= idx) { return false }
209
+ if (a[i] === '*') { continue } // ワイルドカード(どんなタイプも許容)
210
+ const t = this.tokens[idx]
211
+ if (a[i] instanceof Array) {
212
+ if (a[i].indexOf(t.type) < 0) { return false }
213
+ continue
214
+ }
215
+ if (t.type !== a[i]) { return false }
216
+ }
217
+ return true
218
+ }
219
+
220
+ /**
221
+ * カーソル位置の型を確認するが、複数の種類を確かめられる
222
+ */
223
+ checkTypes (a: string[]): boolean {
224
+ const type = this.tokens[this.index].type
225
+ return (a.indexOf(type) >= 0)
226
+ }
227
+
228
+ /**
229
+ * check2の高度なやつ、型名の他にコールバック関数を指定できる
230
+ * 型にマッチしなければ false を返し、カーソルを巻き戻す
231
+ */
232
+ accept (types: any[]): boolean {
233
+ const y = []
234
+ const tmpIndex = this.index
235
+ const rollback = () => {
236
+ this.index = tmpIndex
237
+ return false
238
+ }
239
+ for (let i = 0; i < types.length; i++) {
240
+ if (this.isEOF()) { return rollback() }
241
+ const type = types[i]
242
+ if (type == null) { return rollback() }
243
+ if (typeof type === 'string') {
244
+ const token = this.get()
245
+ if (token && token.type !== type) { return rollback() }
246
+ y[i] = token
247
+ continue
248
+ }
249
+ if (typeof type === 'function') {
250
+ const f = type.bind(this)
251
+ const r: any = f(y)
252
+ if (r === null) { return rollback() }
253
+ y[i] = r
254
+ continue
255
+ }
256
+ if (type instanceof Array) {
257
+ if (!this.checkTypes(type)) { return rollback() }
258
+ y[i] = this.get()
259
+ continue
260
+ }
261
+ throw new Error('System Error : accept broken : ' + typeof type)
262
+ }
263
+ this.y = y
264
+ return true
265
+ }
266
+
267
+ /**
268
+ * カーソル語句を取得して、カーソルを後ろに移動する
269
+ */
270
+ get (): Token | null {
271
+ if (this.isEOF()) { return null }
272
+ return this.tokens[this.index++]
273
+ }
274
+
275
+ /** カーソル語句を取得してカーソルを進める、取得できなければエラーを出す */
276
+ getCur (): Token {
277
+ if (this.isEOF()) { throw new Error('トークンが取得できません。') }
278
+ const t = this.tokens[this.index++]
279
+ if (!t) { throw new Error('トークンが取得できません。') }
280
+ return t
281
+ }
282
+
283
+ unget () {
284
+ if (this.index > 0) { this.index-- }
285
+ }
286
+
287
+ /** 解析中のトークンを返す */
288
+ peek (i = 0): Token|null {
289
+ if (this.isEOF()) { return null }
290
+ return this.tokens[this.index + i]
291
+ }
292
+
293
+ /** 解析中のトークンを返す、無理なら def を返す */
294
+ peekDef (def: Token|null = null): Token {
295
+ if (this.isEOF()) {
296
+ if (!def) { def = NewEmptyToken() }
297
+ return def
298
+ }
299
+ return this.tokens[this.index]
300
+ }
301
+
302
+ /**
303
+ * 現在のカーソル語句のソースコード上の位置を取得する。
304
+ */
305
+ peekSourceMap (): SourceMap {
306
+ const token = this.peek()
307
+ if (token === null) {
308
+ return { startOffset: undefined, endOffset: undefined, file: undefined, line: 0, column: 0 }
309
+ }
310
+ return { startOffset: token.startOffset, endOffset: token.endOffset, file: token.file, line: token.line, column: token.column }
311
+ }
312
+
313
+ /**
314
+ * depth: 表示する深さ
315
+ * typeName: 先頭のtypeの表示を上書きする場合に設定する
316
+ * @param {{ depth: number, typeName?: string }} opts
317
+ * @param {boolean} debugMode
318
+ */
319
+ nodeToStr (node: Ast|Token|null, opts: {depth: number, typeName?: string}, debugMode: boolean): string {
320
+ const depth = opts.depth - 1
321
+ const typeName = (name: string) => (opts.typeName !== undefined) ? opts.typeName : name
322
+ const debug = debugMode ? (' debug: ' + JSON.stringify(node, null, 2)) : ''
323
+ if (!node) { return '(NULL)' }
324
+ switch (node.type) {
325
+ case 'not':
326
+ if (depth >= 0) {
327
+ const subNode: Ast = node.value as Ast
328
+ return `${typeName('')}『${this.nodeToStr(subNode, { depth }, debugMode)}に演算子『not』を適用した式${debug}』`
329
+ } else {
330
+ return `${typeName('演算子')}『not』`
331
+ }
332
+ case 'op': {
333
+ const node2: Ast = node as Ast
334
+ let operator: string = node2.operator || ''
335
+ const table:{[key: string]: string} = { eq: '=', not: '!', gt: '>', lt: '<', and: 'かつ', or: 'または' }
336
+ if (operator in table) {
337
+ operator = table[operator]
338
+ }
339
+ if (depth >= 0) {
340
+ const left: string = this.nodeToStr(node2.left as Ast, { depth }, debugMode)
341
+ const right: string = this.nodeToStr(node2.right as Ast, { depth }, debugMode)
342
+ if (node2.operator === 'eq') {
343
+ return `${typeName('')}『${left}と${right}が等しいかどうかの比較${debug}』`
344
+ }
345
+ return `${typeName('')}『${left}と${right}に演算子『${operator}』を適用した式${debug}』`
346
+ } else {
347
+ return `${typeName('演算子')}『${operator}${debug}』`
348
+ }
349
+ }
350
+ case 'number':
351
+ return `${typeName('数値')}${node.value}`
352
+ case 'string':
353
+ return `${typeName('文字列')}『${node.value}${debug}』`
354
+ case 'word':
355
+ return `${typeName('単語')}『${node.value}${debug}』`
356
+ case 'func':
357
+ return `${typeName('関数')}『${node.name || node.value}${debug}』`
358
+ case 'eol':
359
+ return '行の末尾'
360
+ case 'eof':
361
+ return 'ファイルの末尾'
362
+ default: {
363
+ let name:any = node.name
364
+ if (name) { name = node.value }
365
+ if (typeof name !== 'string') { name = node.type }
366
+ return `${typeName('')}『${name}${debug}』`
367
+ }
368
+ }
369
+ }
370
+ }
@@ -0,0 +1,37 @@
1
+ /* eslint-disable quote-props */
2
+ export const opPriority = {
3
+ // and or
4
+ 'and': 1,
5
+ 'or': 1,
6
+ // compare
7
+ 'eq': 2,
8
+ 'noteq': 2,
9
+ '===': 2,
10
+ '!==': 2,
11
+ 'gt': 2,
12
+ 'gteq': 2,
13
+ 'lt': 2,
14
+ 'lteq': 2,
15
+ '&': 3,
16
+ // + - << >> >>>
17
+ '+': 4,
18
+ '-': 4,
19
+ 'shift_l': 4,
20
+ 'shift_r': 4,
21
+ 'shift_r0': 4,
22
+ // * /
23
+ '*': 5,
24
+ '/': 5,
25
+ '÷': 5,
26
+ '÷÷': 5,
27
+ '%': 5,
28
+ // ^
29
+ '^': 6
30
+ };
31
+ export const keizokuJosi = [
32
+ 'いて', 'えて', 'きて', 'けて', 'して', 'って', 'にて', 'みて', 'めて', 'ねて', 'には', 'んで'
33
+ ];
34
+ export const operatorList = [];
35
+ for (const key in opPriority) {
36
+ operatorList.push(key);
37
+ }
@@ -0,0 +1,37 @@
1
+ /* eslint-disable quote-props */
2
+ export const opPriority: {[key: string]: number} = {
3
+ // and or
4
+ 'and': 1,
5
+ 'or': 1,
6
+ // compare
7
+ 'eq': 2,
8
+ 'noteq': 2,
9
+ '===': 2,
10
+ '!==': 2,
11
+ 'gt': 2,
12
+ 'gteq': 2,
13
+ 'lt': 2,
14
+ 'lteq': 2,
15
+ '&': 3,
16
+ // + - << >> >>>
17
+ '+': 4,
18
+ '-': 4,
19
+ 'shift_l': 4,
20
+ 'shift_r': 4,
21
+ 'shift_r0': 4,
22
+ // * /
23
+ '*': 5,
24
+ '/': 5, // 一般的な割り算
25
+ '÷': 5, // 一般的な割り算
26
+ '÷÷': 5, // 整数の割り算
27
+ '%': 5,
28
+ // ^
29
+ '^': 6
30
+ }
31
+
32
+ export const keizokuJosi: string[] = [
33
+ 'いて', 'えて', 'きて', 'けて', 'して', 'って', 'にて', 'みて', 'めて', 'ねて', 'には', 'んで'
34
+ ]
35
+
36
+ export const operatorList: string[] = []
37
+ for (const key in opPriority) { operatorList.push(key) }