rapydscript-ns 0.8.2 → 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 (141) 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 +39 -0
  5. package/HACKING.md +103 -103
  6. package/LICENSE +24 -24
  7. package/PYTHON_DIFFERENCES_REPORT.md +291 -0
  8. package/PYTHON_FEATURE_COVERAGE.md +106 -15
  9. package/README.md +831 -52
  10. package/TODO.md +4 -286
  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 +4623 -0
  17. package/language-service/language-service.d.ts +40 -0
  18. package/package.json +9 -7
  19. package/publish.py +37 -37
  20. package/release/baselib-plain-pretty.js +2006 -229
  21. package/release/baselib-plain-ugly.js +70 -3
  22. package/release/compiler.js +11554 -3870
  23. package/release/signatures.json +31 -29
  24. package/session.vim +4 -4
  25. package/setup.cfg +2 -2
  26. package/src/ast.pyj +93 -1
  27. package/src/baselib-builtins.pyj +99 -2
  28. package/src/baselib-containers.pyj +107 -4
  29. package/src/baselib-errors.pyj +44 -0
  30. package/src/baselib-internal.pyj +124 -5
  31. package/src/baselib-itertools.pyj +97 -97
  32. package/src/baselib-str.pyj +32 -1
  33. package/src/compiler.pyj +36 -36
  34. package/src/errors.pyj +30 -30
  35. package/src/lib/aes.pyj +646 -646
  36. package/src/lib/collections.pyj +1 -1
  37. package/src/lib/copy.pyj +120 -0
  38. package/src/lib/elementmaker.pyj +83 -83
  39. package/src/lib/encodings.pyj +126 -126
  40. package/src/lib/gettext.pyj +569 -569
  41. package/src/lib/itertools.pyj +580 -580
  42. package/src/lib/math.pyj +193 -193
  43. package/src/lib/numpy.pyj +10 -10
  44. package/src/lib/operator.pyj +11 -11
  45. package/src/lib/pythonize.pyj +20 -20
  46. package/src/lib/random.pyj +118 -118
  47. package/src/lib/re.pyj +470 -470
  48. package/src/lib/react.pyj +74 -0
  49. package/src/lib/traceback.pyj +63 -63
  50. package/src/lib/uuid.pyj +77 -77
  51. package/src/monaco-language-service/analyzer.js +131 -9
  52. package/src/monaco-language-service/builtins.js +17 -2
  53. package/src/monaco-language-service/completions.js +170 -1
  54. package/src/monaco-language-service/diagnostics.js +25 -3
  55. package/src/monaco-language-service/dts.js +550 -550
  56. package/src/monaco-language-service/index.js +17 -0
  57. package/src/monaco-language-service/scope.js +3 -0
  58. package/src/output/classes.pyj +128 -11
  59. package/src/output/codegen.pyj +17 -3
  60. package/src/output/comments.pyj +45 -45
  61. package/src/output/exceptions.pyj +201 -105
  62. package/src/output/functions.pyj +13 -16
  63. package/src/output/jsx.pyj +164 -0
  64. package/src/output/literals.pyj +28 -2
  65. package/src/output/loops.pyj +0 -9
  66. package/src/output/modules.pyj +2 -5
  67. package/src/output/operators.pyj +22 -2
  68. package/src/output/statements.pyj +2 -2
  69. package/src/output/stream.pyj +1 -13
  70. package/src/output/treeshake.pyj +182 -182
  71. package/src/output/utils.pyj +72 -72
  72. package/src/parse.pyj +434 -114
  73. package/src/string_interpolation.pyj +72 -72
  74. package/src/tokenizer.pyj +29 -0
  75. package/src/unicode_aliases.pyj +576 -576
  76. package/src/utils.pyj +192 -192
  77. package/test/_import_one.pyj +37 -37
  78. package/test/_import_two/__init__.pyj +11 -11
  79. package/test/_import_two/level2/deep.pyj +4 -4
  80. package/test/_import_two/other.pyj +6 -6
  81. package/test/_import_two/sub.pyj +13 -13
  82. package/test/aes_vectors.pyj +421 -421
  83. package/test/annotations.pyj +80 -80
  84. package/test/baselib.pyj +4 -4
  85. package/test/classes.pyj +56 -17
  86. package/test/collections.pyj +5 -5
  87. package/test/decorators.pyj +77 -77
  88. package/test/docstrings.pyj +39 -39
  89. package/test/elementmaker_test.pyj +45 -45
  90. package/test/functions.pyj +151 -151
  91. package/test/generators.pyj +41 -41
  92. package/test/generic.pyj +370 -370
  93. package/test/imports.pyj +72 -72
  94. package/test/internationalization.pyj +73 -73
  95. package/test/lint.pyj +164 -164
  96. package/test/loops.pyj +85 -85
  97. package/test/numpy.pyj +734 -734
  98. package/test/omit_function_metadata.pyj +20 -20
  99. package/test/python_compat.pyj +326 -0
  100. package/test/python_features.pyj +129 -29
  101. package/test/regexp.pyj +55 -55
  102. package/test/repl.pyj +121 -121
  103. package/test/scoped_flags.pyj +76 -76
  104. package/test/slice.pyj +105 -0
  105. package/test/str.pyj +25 -0
  106. package/test/unit/fixtures/fibonacci_expected.js +1 -1
  107. package/test/unit/index.js +2296 -71
  108. package/test/unit/language-service-builtins.js +70 -0
  109. package/test/unit/language-service-bundle.js +5 -5
  110. package/test/unit/language-service-completions.js +180 -0
  111. package/test/unit/language-service-dts.js +543 -543
  112. package/test/unit/language-service-hover.js +455 -455
  113. package/test/unit/language-service-index.js +350 -0
  114. package/test/unit/language-service-scope.js +255 -0
  115. package/test/unit/language-service.js +625 -4
  116. package/test/unit/run-language-service.js +1 -0
  117. package/test/unit/web-repl.js +437 -0
  118. package/tools/build-language-service.js +2 -2
  119. package/tools/cli.js +547 -547
  120. package/tools/compile.js +219 -219
  121. package/tools/compiler.js +0 -24
  122. package/tools/completer.js +131 -131
  123. package/tools/embedded_compiler.js +251 -251
  124. package/tools/export.js +3 -37
  125. package/tools/gettext.js +185 -185
  126. package/tools/ini.js +65 -65
  127. package/tools/msgfmt.js +187 -187
  128. package/tools/repl.js +223 -223
  129. package/tools/test.js +118 -118
  130. package/tools/utils.js +128 -128
  131. package/tools/web_repl.js +95 -95
  132. package/try +41 -41
  133. package/web-repl/env.js +196 -74
  134. package/web-repl/index.html +163 -163
  135. package/web-repl/main.js +252 -254
  136. package/web-repl/prism.css +139 -139
  137. package/web-repl/prism.js +113 -113
  138. package/web-repl/rapydscript.js +227 -139
  139. package/web-repl/sha1.js +25 -25
  140. package/hack_demo.pyj +0 -112
  141. package/web-repl/language-service.js +0 -4187
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;