rapydscript-ns 0.8.3 → 0.8.4

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 (116) hide show
  1. package/.agignore +1 -1
  2. package/.github/workflows/ci.yml +38 -38
  3. package/=template.pyj +5 -5
  4. package/CHANGELOG.md +8 -0
  5. package/HACKING.md +103 -103
  6. package/LICENSE +24 -24
  7. package/PYTHON_DIFFERENCES_REPORT.md +2 -2
  8. package/PYTHON_FEATURE_COVERAGE.md +13 -13
  9. package/README.md +670 -6
  10. package/TODO.md +5 -6
  11. package/add-toc-to-readme +2 -2
  12. package/bin/export +75 -75
  13. package/bin/rapydscript +70 -70
  14. package/bin/web-repl-export +102 -102
  15. package/build +2 -2
  16. package/language-service/index.js +155 -6
  17. package/package.json +1 -1
  18. package/publish.py +37 -37
  19. package/release/baselib-plain-pretty.js +2006 -229
  20. package/release/baselib-plain-ugly.js +70 -3
  21. package/release/compiler.js +11554 -3870
  22. package/release/signatures.json +31 -29
  23. package/session.vim +4 -4
  24. package/setup.cfg +2 -2
  25. package/src/ast.pyj +93 -1
  26. package/src/baselib-builtins.pyj +22 -1
  27. package/src/baselib-containers.pyj +99 -0
  28. package/src/baselib-errors.pyj +44 -0
  29. package/src/baselib-internal.pyj +94 -4
  30. package/src/baselib-itertools.pyj +97 -97
  31. package/src/baselib-str.pyj +24 -0
  32. package/src/compiler.pyj +36 -36
  33. package/src/errors.pyj +30 -30
  34. package/src/lib/aes.pyj +646 -646
  35. package/src/lib/copy.pyj +120 -0
  36. package/src/lib/elementmaker.pyj +83 -83
  37. package/src/lib/encodings.pyj +126 -126
  38. package/src/lib/gettext.pyj +569 -569
  39. package/src/lib/itertools.pyj +580 -580
  40. package/src/lib/math.pyj +193 -193
  41. package/src/lib/operator.pyj +11 -11
  42. package/src/lib/pythonize.pyj +20 -20
  43. package/src/lib/random.pyj +118 -118
  44. package/src/lib/re.pyj +470 -470
  45. package/src/lib/react.pyj +74 -0
  46. package/src/lib/traceback.pyj +63 -63
  47. package/src/lib/uuid.pyj +77 -77
  48. package/src/monaco-language-service/builtins.js +5 -0
  49. package/src/monaco-language-service/diagnostics.js +25 -3
  50. package/src/monaco-language-service/dts.js +550 -550
  51. package/src/output/classes.pyj +108 -8
  52. package/src/output/codegen.pyj +16 -2
  53. package/src/output/comments.pyj +45 -45
  54. package/src/output/exceptions.pyj +201 -105
  55. package/src/output/functions.pyj +9 -0
  56. package/src/output/jsx.pyj +164 -0
  57. package/src/output/literals.pyj +28 -2
  58. package/src/output/modules.pyj +1 -1
  59. package/src/output/operators.pyj +8 -2
  60. package/src/output/statements.pyj +2 -2
  61. package/src/output/stream.pyj +1 -0
  62. package/src/output/treeshake.pyj +182 -182
  63. package/src/output/utils.pyj +72 -72
  64. package/src/parse.pyj +417 -113
  65. package/src/string_interpolation.pyj +72 -72
  66. package/src/tokenizer.pyj +29 -0
  67. package/src/unicode_aliases.pyj +576 -576
  68. package/src/utils.pyj +192 -192
  69. package/test/_import_one.pyj +37 -37
  70. package/test/_import_two/__init__.pyj +11 -11
  71. package/test/_import_two/level2/deep.pyj +4 -4
  72. package/test/_import_two/other.pyj +6 -6
  73. package/test/_import_two/sub.pyj +13 -13
  74. package/test/aes_vectors.pyj +421 -421
  75. package/test/annotations.pyj +80 -80
  76. package/test/decorators.pyj +77 -77
  77. package/test/docstrings.pyj +39 -39
  78. package/test/elementmaker_test.pyj +45 -45
  79. package/test/functions.pyj +151 -151
  80. package/test/generators.pyj +41 -41
  81. package/test/generic.pyj +370 -370
  82. package/test/imports.pyj +72 -72
  83. package/test/internationalization.pyj +73 -73
  84. package/test/lint.pyj +164 -164
  85. package/test/loops.pyj +85 -85
  86. package/test/numpy.pyj +734 -734
  87. package/test/omit_function_metadata.pyj +20 -20
  88. package/test/python_features.pyj +19 -6
  89. package/test/regexp.pyj +55 -55
  90. package/test/repl.pyj +121 -121
  91. package/test/scoped_flags.pyj +76 -76
  92. package/test/unit/index.js +2177 -64
  93. package/test/unit/language-service-dts.js +543 -543
  94. package/test/unit/language-service-hover.js +455 -455
  95. package/test/unit/language-service.js +590 -4
  96. package/test/unit/web-repl.js +303 -0
  97. package/tools/cli.js +547 -547
  98. package/tools/compile.js +219 -219
  99. package/tools/completer.js +131 -131
  100. package/tools/embedded_compiler.js +251 -251
  101. package/tools/gettext.js +185 -185
  102. package/tools/ini.js +65 -65
  103. package/tools/msgfmt.js +187 -187
  104. package/tools/repl.js +223 -223
  105. package/tools/test.js +118 -118
  106. package/tools/utils.js +128 -128
  107. package/tools/web_repl.js +95 -95
  108. package/try +41 -41
  109. package/web-repl/env.js +196 -74
  110. package/web-repl/index.html +163 -163
  111. package/web-repl/main.js +252 -254
  112. package/web-repl/prism.css +139 -139
  113. package/web-repl/prism.js +113 -113
  114. package/web-repl/rapydscript.js +224 -102
  115. package/web-repl/sha1.js +25 -25
  116. package/hack_demo.pyj +0 -112
package/tools/gettext.js CHANGED
@@ -1,185 +1,185 @@
1
- /* vim:fileencoding=utf-8
2
- *
3
- * Copyright (C) 2015 Kovid Goyal <kovid at kovidgoyal.net>
4
- *
5
- * Distributed under terms of the BSD license
6
- */
7
- "use strict"; /*jshint node:true */
8
-
9
- var fs = require('fs');
10
- var RapydScript = (typeof create_rapydscript_compiler === 'function') ? create_rapydscript_compiler() : require('./compiler').create_compiler();
11
- var path = require('path');
12
-
13
- function parse_file(code, filename) {
14
- return RapydScript.parse(code, {
15
- filename: filename,
16
- basedir: path.dirname(filename),
17
- libdir: path.dirname(filename),
18
- for_linting: true,
19
- });
20
- }
21
-
22
- function detect_format(msgid) {
23
- var q = msgid.replace('{{', '');
24
- if (/\{[0-9a-zA-Z_}]+/.test(q)) return 'python-brace-format';
25
- return null;
26
- }
27
-
28
- function Gettext(catalog, filename) {
29
- this._visit = function (node, cont) {
30
- if (node instanceof RapydScript.AST_Call && node.args && node.args.length && node.expression instanceof RapydScript.AST_Symbol) {
31
- var name = node.expression.name;
32
- if (name === '_' || name === 'gettext' || name === 'ngettext') {
33
- var nargs = (name === 'ngettext') ? 2 : 1;
34
- var line = node.start.line;
35
- for (var i = 0; i < nargs; i++) {
36
- if (!(node.args[i] instanceof RapydScript.AST_String)) {
37
- console.error('Translation function: ' + name + ' does not have a string literal argument at line: ' + line + ' of ' + filename);
38
- process.exit(1);
39
- }
40
- }
41
- var msgid = node.args[0].value;
42
- if (!Object.prototype.hasOwnProperty.call(catalog, msgid)) {
43
- catalog[msgid] = {
44
- 'locations': [],
45
- 'plural': null,
46
- 'format': detect_format(msgid),
47
- };
48
- }
49
- if (name === 'ngettext') catalog[msgid].plural = node.args[1].value;
50
- if (filename) catalog[msgid].locations.push(filename + ':' + line);
51
- }
52
-
53
- }
54
- if (cont !== undefined) cont();
55
- };
56
- }
57
-
58
- function gettext(catalog, code, filename) {
59
- var toplevel;
60
-
61
- try {
62
- toplevel = parse_file(code, filename);
63
- } catch(e) {
64
- if (e instanceof RapydScript.SyntaxError) {
65
- console.error('Failed to parse: ' + filename + ' with error: ' + e.line + ':' + e.col + ':' + e.message);
66
- process.exit(1);
67
- } else throw e;
68
- }
69
-
70
- if (toplevel) {
71
- var gt = new Gettext(catalog, filename);
72
- toplevel.walk(gt);
73
- }
74
- }
75
-
76
- function esc(string) {
77
- return (string || '').replace(/\\/g, '\\\\').replace(/"/g, '\\"').replace(/\n/g, '\\n').replace(/\t/g, '\\t').replace(/\r/g, '');
78
- }
79
-
80
- function entry_to_string(msgid, data) {
81
- var ans = [];
82
- data.locations.forEach(function (loc) { ans.push('#: ' + loc); });
83
- if (data.format) ans.push('#, ' + data.format);
84
- ans.push('msgid "' + esc(msgid) + '"');
85
- if (data.plural) {
86
- ans.push('msgid_plural "' + esc(data.plural) + '"');
87
- ans.push('msgstr[0] ""');
88
- ans.push('msgstr[1] ""');
89
- } else ans.push('msgstr ""');
90
- return ans.join('\n');
91
- }
92
-
93
- function write_output(catalog, options, write) {
94
- write = write || (function(x) { process.stdout.write(new Buffer(x, 'utf-8')); });
95
- function print() {
96
- var val = Array.prototype.slice.call(arguments).join(' ') + '\n';
97
- write(val);
98
- }
99
- function header_line() {
100
- var val = '"' + Array.prototype.slice.call(arguments).join(' ') + '\\n"\n';
101
- write(val);
102
- }
103
- if (!options.omit_header) {
104
- var now = (new Date()).toISOString();
105
- print('msgid', '""');
106
- print('msgstr', '""');
107
- header_line('Project-Id-Version:', esc(options.package_name), esc(options.package_version));
108
- header_line('POT-Creation-Date:', now);
109
- header_line("PO-Revision-Date:", now);
110
- header_line("Report-Msgid-Bugs-To:", esc(options.bugs_address));
111
- header_line("Last-Translator: Automatically generated");
112
- header_line("Language-Team: LANGUAGE");
113
- header_line("MIME-Version: 1.0");
114
- header_line("Plural-Forms: nplurals=INTEGER; plural=EXPRESSION;");
115
- header_line("Content-Type: text/plain; charset=UTF-8");
116
- header_line("Content-Transfer-Encoding: 8bit");
117
- print();
118
- }
119
- Object.keys(catalog).forEach(function(msgid) {
120
- var data = catalog[msgid];
121
- print(entry_to_string(msgid, data));
122
- print();
123
- });
124
- }
125
-
126
- // CLI {{{
127
-
128
- function read_whole_file(filename, cb) {
129
- if (!filename) {
130
- var chunks = [];
131
- process.stdin.setEncoding('utf-8');
132
- process.stdin.on('data', function (chunk) {
133
- chunks.push(chunk);
134
- }).on('end', function () {
135
- cb(null, chunks.join(""));
136
- });
137
- process.openStdin();
138
- } else {
139
- fs.readFile(filename, "utf-8", cb);
140
- }
141
- }
142
-
143
- module.exports.cli = function(argv, base_path, src_path, lib_path) {
144
- var files = [];
145
- var num_of_files = files.length || 1;
146
- var catalog = {};
147
-
148
- function read_files(src) {
149
- src.forEach(function(f) {
150
- if (fs.lstatSync(f).isDirectory()) {
151
- var children = [];
152
- fs.readdirSync(f).forEach(function(x) { children.push(path.join(f, x)); });
153
- read_files(children);
154
- }
155
- else files.push(f);
156
- });
157
- }
158
- read_files(argv.files);
159
-
160
- function process_single_file(err, code) {
161
- if (err) {
162
- console.error("ERROR: can't read file: " + files[0]);
163
- process.exit(1);
164
- }
165
-
166
- gettext(catalog, code, files[0]);
167
-
168
- files = files.slice(1);
169
- if (files.length) {
170
- setImmediate(read_whole_file, files[0], process_single_file);
171
- return;
172
- } else {
173
- write_output(catalog, argv);
174
- process.exit(0);
175
- }
176
- }
177
-
178
- setImmediate(read_whole_file, files[0], process_single_file);
179
-
180
- };
181
-
182
- module.exports.gettext = gettext;
183
- module.exports.entry_to_string = entry_to_string;
184
- module.exports.write_output = write_output;
185
- // }}}
1
+ /* vim:fileencoding=utf-8
2
+ *
3
+ * Copyright (C) 2015 Kovid Goyal <kovid at kovidgoyal.net>
4
+ *
5
+ * Distributed under terms of the BSD license
6
+ */
7
+ "use strict"; /*jshint node:true */
8
+
9
+ var fs = require('fs');
10
+ var RapydScript = (typeof create_rapydscript_compiler === 'function') ? create_rapydscript_compiler() : require('./compiler').create_compiler();
11
+ var path = require('path');
12
+
13
+ function parse_file(code, filename) {
14
+ return RapydScript.parse(code, {
15
+ filename: filename,
16
+ basedir: path.dirname(filename),
17
+ libdir: path.dirname(filename),
18
+ for_linting: true,
19
+ });
20
+ }
21
+
22
+ function detect_format(msgid) {
23
+ var q = msgid.replace('{{', '');
24
+ if (/\{[0-9a-zA-Z_}]+/.test(q)) return 'python-brace-format';
25
+ return null;
26
+ }
27
+
28
+ function Gettext(catalog, filename) {
29
+ this._visit = function (node, cont) {
30
+ if (node instanceof RapydScript.AST_Call && node.args && node.args.length && node.expression instanceof RapydScript.AST_Symbol) {
31
+ var name = node.expression.name;
32
+ if (name === '_' || name === 'gettext' || name === 'ngettext') {
33
+ var nargs = (name === 'ngettext') ? 2 : 1;
34
+ var line = node.start.line;
35
+ for (var i = 0; i < nargs; i++) {
36
+ if (!(node.args[i] instanceof RapydScript.AST_String)) {
37
+ console.error('Translation function: ' + name + ' does not have a string literal argument at line: ' + line + ' of ' + filename);
38
+ process.exit(1);
39
+ }
40
+ }
41
+ var msgid = node.args[0].value;
42
+ if (!Object.prototype.hasOwnProperty.call(catalog, msgid)) {
43
+ catalog[msgid] = {
44
+ 'locations': [],
45
+ 'plural': null,
46
+ 'format': detect_format(msgid),
47
+ };
48
+ }
49
+ if (name === 'ngettext') catalog[msgid].plural = node.args[1].value;
50
+ if (filename) catalog[msgid].locations.push(filename + ':' + line);
51
+ }
52
+
53
+ }
54
+ if (cont !== undefined) cont();
55
+ };
56
+ }
57
+
58
+ function gettext(catalog, code, filename) {
59
+ var toplevel;
60
+
61
+ try {
62
+ toplevel = parse_file(code, filename);
63
+ } catch(e) {
64
+ if (e instanceof RapydScript.SyntaxError) {
65
+ console.error('Failed to parse: ' + filename + ' with error: ' + e.line + ':' + e.col + ':' + e.message);
66
+ process.exit(1);
67
+ } else throw e;
68
+ }
69
+
70
+ if (toplevel) {
71
+ var gt = new Gettext(catalog, filename);
72
+ toplevel.walk(gt);
73
+ }
74
+ }
75
+
76
+ function esc(string) {
77
+ return (string || '').replace(/\\/g, '\\\\').replace(/"/g, '\\"').replace(/\n/g, '\\n').replace(/\t/g, '\\t').replace(/\r/g, '');
78
+ }
79
+
80
+ function entry_to_string(msgid, data) {
81
+ var ans = [];
82
+ data.locations.forEach(function (loc) { ans.push('#: ' + loc); });
83
+ if (data.format) ans.push('#, ' + data.format);
84
+ ans.push('msgid "' + esc(msgid) + '"');
85
+ if (data.plural) {
86
+ ans.push('msgid_plural "' + esc(data.plural) + '"');
87
+ ans.push('msgstr[0] ""');
88
+ ans.push('msgstr[1] ""');
89
+ } else ans.push('msgstr ""');
90
+ return ans.join('\n');
91
+ }
92
+
93
+ function write_output(catalog, options, write) {
94
+ write = write || (function(x) { process.stdout.write(new Buffer(x, 'utf-8')); });
95
+ function print() {
96
+ var val = Array.prototype.slice.call(arguments).join(' ') + '\n';
97
+ write(val);
98
+ }
99
+ function header_line() {
100
+ var val = '"' + Array.prototype.slice.call(arguments).join(' ') + '\\n"\n';
101
+ write(val);
102
+ }
103
+ if (!options.omit_header) {
104
+ var now = (new Date()).toISOString();
105
+ print('msgid', '""');
106
+ print('msgstr', '""');
107
+ header_line('Project-Id-Version:', esc(options.package_name), esc(options.package_version));
108
+ header_line('POT-Creation-Date:', now);
109
+ header_line("PO-Revision-Date:", now);
110
+ header_line("Report-Msgid-Bugs-To:", esc(options.bugs_address));
111
+ header_line("Last-Translator: Automatically generated");
112
+ header_line("Language-Team: LANGUAGE");
113
+ header_line("MIME-Version: 1.0");
114
+ header_line("Plural-Forms: nplurals=INTEGER; plural=EXPRESSION;");
115
+ header_line("Content-Type: text/plain; charset=UTF-8");
116
+ header_line("Content-Transfer-Encoding: 8bit");
117
+ print();
118
+ }
119
+ Object.keys(catalog).forEach(function(msgid) {
120
+ var data = catalog[msgid];
121
+ print(entry_to_string(msgid, data));
122
+ print();
123
+ });
124
+ }
125
+
126
+ // CLI {{{
127
+
128
+ function read_whole_file(filename, cb) {
129
+ if (!filename) {
130
+ var chunks = [];
131
+ process.stdin.setEncoding('utf-8');
132
+ process.stdin.on('data', function (chunk) {
133
+ chunks.push(chunk);
134
+ }).on('end', function () {
135
+ cb(null, chunks.join(""));
136
+ });
137
+ process.openStdin();
138
+ } else {
139
+ fs.readFile(filename, "utf-8", cb);
140
+ }
141
+ }
142
+
143
+ module.exports.cli = function(argv, base_path, src_path, lib_path) {
144
+ var files = [];
145
+ var num_of_files = files.length || 1;
146
+ var catalog = {};
147
+
148
+ function read_files(src) {
149
+ src.forEach(function(f) {
150
+ if (fs.lstatSync(f).isDirectory()) {
151
+ var children = [];
152
+ fs.readdirSync(f).forEach(function(x) { children.push(path.join(f, x)); });
153
+ read_files(children);
154
+ }
155
+ else files.push(f);
156
+ });
157
+ }
158
+ read_files(argv.files);
159
+
160
+ function process_single_file(err, code) {
161
+ if (err) {
162
+ console.error("ERROR: can't read file: " + files[0]);
163
+ process.exit(1);
164
+ }
165
+
166
+ gettext(catalog, code, files[0]);
167
+
168
+ files = files.slice(1);
169
+ if (files.length) {
170
+ setImmediate(read_whole_file, files[0], process_single_file);
171
+ return;
172
+ } else {
173
+ write_output(catalog, argv);
174
+ process.exit(0);
175
+ }
176
+ }
177
+
178
+ setImmediate(read_whole_file, files[0], process_single_file);
179
+
180
+ };
181
+
182
+ module.exports.gettext = gettext;
183
+ module.exports.entry_to_string = entry_to_string;
184
+ module.exports.write_output = write_output;
185
+ // }}}
package/tools/ini.js CHANGED
@@ -1,65 +1,65 @@
1
- /* vim:fileencoding=utf-8
2
- *
3
- * Copyright (C) 2015 Kovid Goyal <kovid at kovidgoyal.net>
4
- *
5
- * Distributed under terms of the BSD license
6
- */
7
- "use strict"; /*jshint node:true */
8
-
9
- var fs = require('fs');
10
- var path = require('path');
11
-
12
- function parse_ini_data(data) {
13
- // Based on MIT licensed code from:
14
- // https://github.com/shockie/node-iniparser/blob/master/lib/node-iniparser.js
15
- var ans = {}, match;
16
- var lines = data.split(/\r\n|\r|\n/);
17
- var section = null;
18
- var section_pat = /^\s*\[\s*([^\]]*)\s*\]\s*$/;
19
- var param_pat = /^\s*([\w\.\-\_]+)\s*=\s*(.*?)\s*$/;
20
- var comment_pat = /^\s*;.*$/;
21
-
22
- lines.forEach(function(line) {
23
- if(comment_pat.test(line)) {
24
- return;
25
- } else if(param_pat.test(line)) {
26
- match = line.match(param_pat);
27
- if(section) {
28
- ans[section][match[1]] = match[2];
29
- } else {
30
- ans[match[1]] = match[2];
31
- }
32
- } else if(section_pat.test(line)) {
33
- match = line.match(section_pat);
34
- ans[match[1]] = {};
35
- section = match[1];
36
- } else if(line.length === 0 && section) {
37
- section = null;
38
- }
39
- });
40
- return ans;
41
- }
42
-
43
- function find_cfg_file(toplevel_dir) {
44
- var current_dir = toplevel_dir, previous_dir = toplevel_dir;
45
- do {
46
- try {
47
- return fs.readFileSync(path.join(current_dir, 'setup.cfg'), 'utf-8');
48
- } catch (e) {
49
- if (e.code !== 'ENOENT') throw e;
50
- }
51
- previous_dir = current_dir;
52
- current_dir = path.dirname(current_dir);
53
- } while(current_dir != previous_dir && current_dir);
54
-
55
- return null;
56
- }
57
-
58
- function read_config(toplevel_dir) {
59
- var data = find_cfg_file(toplevel_dir);
60
- if (!data) return {};
61
- return parse_ini_data(data);
62
- }
63
-
64
-
65
- exports.read_config = read_config;
1
+ /* vim:fileencoding=utf-8
2
+ *
3
+ * Copyright (C) 2015 Kovid Goyal <kovid at kovidgoyal.net>
4
+ *
5
+ * Distributed under terms of the BSD license
6
+ */
7
+ "use strict"; /*jshint node:true */
8
+
9
+ var fs = require('fs');
10
+ var path = require('path');
11
+
12
+ function parse_ini_data(data) {
13
+ // Based on MIT licensed code from:
14
+ // https://github.com/shockie/node-iniparser/blob/master/lib/node-iniparser.js
15
+ var ans = {}, match;
16
+ var lines = data.split(/\r\n|\r|\n/);
17
+ var section = null;
18
+ var section_pat = /^\s*\[\s*([^\]]*)\s*\]\s*$/;
19
+ var param_pat = /^\s*([\w\.\-\_]+)\s*=\s*(.*?)\s*$/;
20
+ var comment_pat = /^\s*;.*$/;
21
+
22
+ lines.forEach(function(line) {
23
+ if(comment_pat.test(line)) {
24
+ return;
25
+ } else if(param_pat.test(line)) {
26
+ match = line.match(param_pat);
27
+ if(section) {
28
+ ans[section][match[1]] = match[2];
29
+ } else {
30
+ ans[match[1]] = match[2];
31
+ }
32
+ } else if(section_pat.test(line)) {
33
+ match = line.match(section_pat);
34
+ ans[match[1]] = {};
35
+ section = match[1];
36
+ } else if(line.length === 0 && section) {
37
+ section = null;
38
+ }
39
+ });
40
+ return ans;
41
+ }
42
+
43
+ function find_cfg_file(toplevel_dir) {
44
+ var current_dir = toplevel_dir, previous_dir = toplevel_dir;
45
+ do {
46
+ try {
47
+ return fs.readFileSync(path.join(current_dir, 'setup.cfg'), 'utf-8');
48
+ } catch (e) {
49
+ if (e.code !== 'ENOENT') throw e;
50
+ }
51
+ previous_dir = current_dir;
52
+ current_dir = path.dirname(current_dir);
53
+ } while(current_dir != previous_dir && current_dir);
54
+
55
+ return null;
56
+ }
57
+
58
+ function read_config(toplevel_dir) {
59
+ var data = find_cfg_file(toplevel_dir);
60
+ if (!data) return {};
61
+ return parse_ini_data(data);
62
+ }
63
+
64
+
65
+ exports.read_config = read_config;