firefly-compiler 0.5.35 → 0.5.37

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 (225) hide show
  1. package/.hintrc +4 -4
  2. package/.vscode/settings.json +4 -4
  3. package/bin/Release.ff +157 -154
  4. package/bin/firefly.mjs +1 -1
  5. package/compiler/Builder.ff +275 -277
  6. package/compiler/Compiler.ff +234 -233
  7. package/compiler/Dependencies.ff +186 -187
  8. package/compiler/DependencyLock.ff +17 -17
  9. package/compiler/Deriver.ff +23 -31
  10. package/compiler/Dictionaries.ff +1 -1
  11. package/compiler/Inference.ff +43 -20
  12. package/compiler/JsEmitter.ff +1437 -1282
  13. package/compiler/LspHook.ff +202 -202
  14. package/compiler/Main.ff +25 -24
  15. package/compiler/ModuleCache.ff +178 -178
  16. package/compiler/Parser.ff +36 -109
  17. package/compiler/Resolver.ff +5 -8
  18. package/compiler/Substitution.ff +1 -1
  19. package/compiler/Syntax.ff +1 -16
  20. package/compiler/Token.ff +9 -0
  21. package/compiler/Tokenizer.ff +4 -0
  22. package/compiler/Workspace.ff +88 -88
  23. package/core/.firefly/include/package.json +5 -5
  24. package/core/.firefly/package.ff +2 -2
  25. package/core/Any.ff +26 -30
  26. package/core/Array.ff +298 -265
  27. package/core/Atomic.ff +63 -64
  28. package/core/Box.ff +7 -7
  29. package/core/BrowserSystem.ff +40 -40
  30. package/core/Buffer.ff +185 -152
  31. package/core/BuildSystem.ff +156 -148
  32. package/core/Channel.ff +95 -92
  33. package/core/Char.ff +3 -2
  34. package/core/Core.ff +16 -23
  35. package/core/Crypto.ff +94 -96
  36. package/core/Equal.ff +41 -36
  37. package/core/Error.ff +15 -10
  38. package/core/FileHandle.ff +45 -37
  39. package/core/Float.ff +176 -200
  40. package/core/HttpClient.ff +142 -148
  41. package/core/Instant.ff +6 -8
  42. package/core/Int.ff +40 -24
  43. package/core/IntMap.ff +61 -39
  44. package/core/Js.ff +305 -0
  45. package/core/JsSystem.ff +135 -114
  46. package/core/JsValue.ff +303 -159
  47. package/core/Json.ff +423 -443
  48. package/core/List.ff +482 -486
  49. package/core/Lock.ff +108 -144
  50. package/core/Log.ff +25 -14
  51. package/core/NodeSystem.ff +198 -191
  52. package/core/Ordering.ff +160 -161
  53. package/core/Path.ff +377 -409
  54. package/core/Queue.ff +90 -0
  55. package/core/Random.ff +140 -134
  56. package/core/RbMap.ff +216 -216
  57. package/core/Serializable.ff +16 -13
  58. package/core/Show.ff +44 -43
  59. package/core/SourceLocation.ff +68 -68
  60. package/core/Stream.ff +1 -1
  61. package/core/String.ff +224 -202
  62. package/core/StringMap.ff +58 -36
  63. package/core/Task.ff +165 -149
  64. package/experimental/benchmarks/ListGrab.ff +23 -23
  65. package/experimental/benchmarks/ListGrab.java +55 -55
  66. package/experimental/benchmarks/Pyrotek45.ff +30 -30
  67. package/experimental/benchmarks/Pyrotek45.java +64 -64
  68. package/experimental/bidirectional/Bidi.ff +88 -88
  69. package/experimental/lines/Main.ff +40 -0
  70. package/experimental/random/Index.ff +53 -53
  71. package/experimental/random/Process.ff +120 -120
  72. package/experimental/random/RunLength.ff +65 -65
  73. package/experimental/random/Scrape.ff +51 -51
  74. package/experimental/random/Symbols.ff +73 -73
  75. package/experimental/random/Tensor.ff +52 -52
  76. package/experimental/random/Units.ff +36 -36
  77. package/experimental/s3/S3TestAuthorizationHeader.ff +39 -39
  78. package/experimental/s3/S3TestPut.ff +16 -16
  79. package/experimental/tests/TestJson.ff +26 -26
  80. package/firefly.sh +0 -0
  81. package/fireflysite/.firefly/package.ff +4 -4
  82. package/fireflysite/CommunityOverview.ff +20 -20
  83. package/fireflysite/CountingButtonDemo.ff +58 -58
  84. package/fireflysite/DocumentParser.ff +325 -331
  85. package/fireflysite/ExamplesOverview.ff +40 -40
  86. package/fireflysite/FrontPage.ff +344 -344
  87. package/fireflysite/GettingStarted.ff +45 -45
  88. package/fireflysite/Guide.ff +456 -456
  89. package/fireflysite/Main.ff +163 -152
  90. package/fireflysite/MatchingPasswordsDemo.ff +82 -82
  91. package/fireflysite/PackagesOverview.ff +49 -49
  92. package/fireflysite/PostgresqlDemo.ff +34 -34
  93. package/fireflysite/ReferenceAll.ff +18 -18
  94. package/fireflysite/ReferenceIntroduction.ff +11 -11
  95. package/fireflysite/Styles.ff +567 -567
  96. package/fireflysite/Test.ff +121 -62
  97. package/fireflysite/assets/markdown/reference/BaseTypes.md +209 -209
  98. package/fireflysite/assets/markdown/reference/EmittedJavascript.md +65 -65
  99. package/fireflysite/assets/markdown/reference/Exceptions.md +101 -101
  100. package/fireflysite/assets/markdown/reference/FunctionsAndMethods.md +364 -364
  101. package/fireflysite/assets/markdown/reference/JavascriptInterop.md +235 -172
  102. package/fireflysite/assets/markdown/reference/ModulesAndPackages.md +162 -162
  103. package/fireflysite/assets/markdown/reference/OldStructuredConcurrency.md +48 -48
  104. package/fireflysite/assets/markdown/reference/PatternMatching.md +224 -224
  105. package/fireflysite/assets/markdown/reference/StatementsAndExpressions.md +86 -86
  106. package/fireflysite/assets/markdown/reference/StructuredConcurrency.md +99 -99
  107. package/fireflysite/assets/markdown/reference/TraitsAndInstances.md +100 -100
  108. package/fireflysite/assets/markdown/reference/UserDefinedTypes.md +184 -184
  109. package/fireflysite/assets/markdown/scratch/ControlFlow.md +136 -136
  110. package/fireflysite/assets/markdown/scratch/Toc.md +40 -40
  111. package/lsp/.firefly/package.ff +1 -1
  112. package/lsp/CompletionHandler.ff +827 -827
  113. package/lsp/Handler.ff +714 -714
  114. package/lsp/HoverHandler.ff +79 -79
  115. package/lsp/LanguageServer.ff +272 -272
  116. package/lsp/SignatureHelpHandler.ff +55 -55
  117. package/lsp/SymbolHandler.ff +181 -181
  118. package/lsp/TestReferences.ff +17 -17
  119. package/lsp/TestReferencesCase.ff +7 -7
  120. package/lsp/stderr.txt +1 -1
  121. package/lsp/stdout.txt +34 -34
  122. package/lux/.firefly/package.ff +1 -1
  123. package/lux/Css.ff +648 -648
  124. package/lux/CssTest.ff +48 -48
  125. package/lux/Lux.ff +608 -617
  126. package/lux/LuxEvent.ff +79 -116
  127. package/lux/Main.ff +123 -123
  128. package/lux/Main2.ff +143 -143
  129. package/lux/TestDry.ff +28 -28
  130. package/output/js/ff/compiler/Builder.mjs +72 -71
  131. package/output/js/ff/compiler/Compiler.mjs +19 -13
  132. package/output/js/ff/compiler/Dependencies.mjs +8 -7
  133. package/output/js/ff/compiler/DependencyLock.mjs +6 -4
  134. package/output/js/ff/compiler/Deriver.mjs +26 -24
  135. package/output/js/ff/compiler/Dictionaries.mjs +14 -18
  136. package/output/js/ff/compiler/Environment.mjs +6 -4
  137. package/output/js/ff/compiler/Inference.mjs +238 -164
  138. package/output/js/ff/compiler/JsEmitter.mjs +1160 -350
  139. package/output/js/ff/compiler/JsImporter.mjs +20 -18
  140. package/output/js/ff/compiler/LspHook.mjs +12 -10
  141. package/output/js/ff/compiler/Main.mjs +61 -41
  142. package/output/js/ff/compiler/ModuleCache.mjs +10 -8
  143. package/output/js/ff/compiler/Parser.mjs +153 -669
  144. package/output/js/ff/compiler/Patterns.mjs +12 -10
  145. package/output/js/ff/compiler/Resolver.mjs +52 -78
  146. package/output/js/ff/compiler/Substitution.mjs +12 -16
  147. package/output/js/ff/compiler/Syntax.mjs +50 -341
  148. package/output/js/ff/compiler/Token.mjs +126 -4
  149. package/output/js/ff/compiler/Tokenizer.mjs +62 -52
  150. package/output/js/ff/compiler/Unification.mjs +74 -90
  151. package/output/js/ff/compiler/Wildcards.mjs +4 -2
  152. package/output/js/ff/compiler/Workspace.mjs +26 -20
  153. package/output/js/ff/core/Any.mjs +20 -20
  154. package/output/js/ff/core/Array.mjs +268 -175
  155. package/output/js/ff/core/AssetSystem.mjs +8 -6
  156. package/output/js/ff/core/Atomic.mjs +84 -52
  157. package/output/js/ff/core/Bool.mjs +6 -4
  158. package/output/js/ff/core/BrowserSystem.mjs +38 -29
  159. package/output/js/ff/core/Buffer.mjs +285 -133
  160. package/output/js/ff/core/BuildSystem.mjs +36 -56
  161. package/output/js/ff/core/Channel.mjs +250 -97
  162. package/output/js/ff/core/Char.mjs +5 -3
  163. package/output/js/ff/core/Core.mjs +28 -34
  164. package/output/js/ff/core/Crypto.mjs +30 -52
  165. package/output/js/ff/core/Duration.mjs +4 -2
  166. package/output/js/ff/core/Equal.mjs +14 -12
  167. package/output/js/ff/core/Error.mjs +17 -11
  168. package/output/js/ff/core/FileHandle.mjs +76 -38
  169. package/output/js/ff/core/Float.mjs +92 -160
  170. package/output/js/ff/core/HttpClient.mjs +208 -76
  171. package/output/js/ff/core/Instant.mjs +8 -10
  172. package/output/js/ff/core/Int.mjs +36 -26
  173. package/output/js/ff/core/IntMap.mjs +79 -33
  174. package/output/js/ff/core/Js.mjs +751 -0
  175. package/output/js/ff/core/JsSystem.mjs +54 -60
  176. package/output/js/ff/core/JsValue.mjs +294 -143
  177. package/output/js/ff/core/Json.mjs +443 -253
  178. package/output/js/ff/core/List.mjs +262 -214
  179. package/output/js/ff/core/Lock.mjs +156 -125
  180. package/output/js/ff/core/Log.mjs +20 -10
  181. package/output/js/ff/core/Map.mjs +10 -8
  182. package/output/js/ff/core/NodeSystem.mjs +189 -123
  183. package/output/js/ff/core/Nothing.mjs +4 -2
  184. package/output/js/ff/core/Option.mjs +40 -38
  185. package/output/js/ff/core/Ordering.mjs +26 -20
  186. package/output/js/ff/core/Pair.mjs +4 -2
  187. package/output/js/ff/core/Path.mjs +517 -315
  188. package/output/js/ff/core/Queue.mjs +306 -0
  189. package/output/js/ff/core/Random.mjs +141 -77
  190. package/output/js/ff/core/RbMap.mjs +36 -34
  191. package/output/js/ff/core/Serializable.mjs +44 -28
  192. package/output/js/ff/core/Set.mjs +6 -4
  193. package/output/js/ff/core/Show.mjs +8 -6
  194. package/output/js/ff/core/SourceLocation.mjs +4 -2
  195. package/output/js/ff/core/Stream.mjs +30 -50
  196. package/output/js/ff/core/String.mjs +263 -172
  197. package/output/js/ff/core/StringMap.mjs +77 -31
  198. package/output/js/ff/core/Task.mjs +91 -76
  199. package/output/js/ff/core/Try.mjs +20 -18
  200. package/output/js/ff/core/Unit.mjs +4 -2
  201. package/package.json +1 -1
  202. package/postgresql/Pg.ff +53 -59
  203. package/rpc/.firefly/package.ff +1 -1
  204. package/rpc/Rpc.ff +70 -70
  205. package/s3/.firefly/package.ff +1 -1
  206. package/s3/S3.ff +92 -94
  207. package/vscode/LICENSE.txt +21 -21
  208. package/vscode/Prepublish.ff +15 -15
  209. package/vscode/README.md +16 -16
  210. package/vscode/client/package-lock.json +544 -544
  211. package/vscode/client/package.json +22 -22
  212. package/vscode/client/src/extension.ts +104 -104
  213. package/vscode/icons/firefly-icon.svg +10 -10
  214. package/vscode/language-configuration.json +61 -61
  215. package/vscode/package-lock.json +3623 -3623
  216. package/vscode/package.json +1 -1
  217. package/vscode/snippets.json +241 -241
  218. package/vscode/syntaxes/firefly-markdown-injection.json +45 -45
  219. package/webserver/.firefly/include/package.json +5 -5
  220. package/webserver/.firefly/package.ff +2 -2
  221. package/webserver/WebServer.ff +647 -685
  222. package/websocket/.firefly/package.ff +1 -1
  223. package/websocket/WebSocket.ff +100 -131
  224. package/core/UnsafeJs.ff +0 -42
  225. package/output/js/ff/core/UnsafeJs.mjs +0 -191
@@ -1,148 +1,156 @@
1
- capability BuildSystem {}
2
- capability BrowserCode(packageGroup: String, packageName: String, mainFile: Path, assetSystem: AssetSystem)
3
- capability BrowserBundle(assetSystem: AssetSystem)
4
-
5
- extend self: BuildSystem {
6
-
7
- compileForBrowser(mainFile: String): BrowserCode {
8
- // TODO: Check that the mainFile is in the current package directory
9
- internalCompile(self, internalPath(self, mainFile), "browser")
10
- let streams = internalListDirectory(internalPath(self, ".firefly/output/browser"))
11
- let mainPackagePair = internalMainPackagePair(self)
12
- BrowserCode(
13
- packageGroup = mainPackagePair.first
14
- packageName = mainPackagePair.second
15
- mainFile = internalPath(self, mainFile)
16
- assetSystem = AssetSystem(streams.toMap())
17
- )
18
- }
19
-
20
- buildMode(): Bool
21
- target node async "return !!self_.buildMode_"
22
-
23
- setAssets(assetSystem: AssetSystem): Unit
24
- target node async "self_.assets_ = assetSystem_"
25
-
26
- packageAssets(): AssetSystem {
27
- let streams = internalListDirectory(internalPath(self, "."))
28
- AssetSystem(streams.toMap())
29
- }
30
-
31
- dependencyAssets(user: String, package: String): AssetSystem {
32
- panic("dependencyAssets not yet implemented")
33
- }
34
-
35
- arguments(): List[String]
36
- target node async "return self_.array_"
37
-
38
- mainTask(): Task
39
- target js async "return self_.task_"
40
-
41
- crypto(): Crypto
42
- target js async "return (typeof globalThis !== 'undefined' ? globalThis : window).crypto"
43
-
44
- }
45
-
46
- extend self: BrowserCode {
47
-
48
- assets(): AssetSystem {
49
- self.assetSystem
50
- }
51
-
52
- bundle(minify: Bool = True, sourceMap: Bool = False): BrowserBundle {
53
- let prefix = ".firefly/output/browser"
54
- let mainJsBaseFile = self.mainFile.base().removeLast(".ff").grab() + ".mjs"
55
- let mainJsFile = prefix + "/" + self.packageGroup + "/" + self.packageName + "/" + mainJsBaseFile
56
- let mainDirectory = self.mainFile.parent().grab()
57
- let file = prefix + "/Main.bundle.js"
58
- internalCallEsBuild(self, mainJsFile = mainJsFile, outputPath = file, minify = minify, sourceMap = sourceMap)
59
- let assets = AssetSystem([
60
- Pair(file.dropFirst(prefix.size()), {mainDirectory.path(file).readStream()})
61
- ...if(sourceMap) {[
62
- Pair(file.dropFirst(prefix.size()) + ".map", {mainDirectory.path(file + ".map").readStream()})
63
- ]} else {[]}
64
- ].toMap())
65
- BrowserBundle(assets)
66
- }
67
-
68
- }
69
-
70
- extend self: BrowserBundle {
71
-
72
- assets(): AssetSystem {
73
- self.assetSystem
74
- }
75
-
76
- }
77
-
78
-
79
- internalCallEsBuild(
80
- self: BrowserCode
81
- mainJsFile: String
82
- outputPath: String
83
- minify: Bool
84
- sourceMap: Bool
85
- ): Unit
86
- target node async """
87
- import * as esbuild from 'esbuild'
88
- return await esbuild.build({
89
- stdin: {contents: `import {$run$} from './${mainJsFile_}';`, resolveDir: '.'},
90
- bundle: true,
91
- minify: minify_,
92
- sourcemap: sourceMap_,
93
- platform: 'browser',
94
- target: 'es2017',
95
- external: ['../../../node_modules/*'], // TODO
96
- outfile: outputPath_
97
- })
98
- """
99
-
100
- internalNodeCallEsBuild(
101
- self: NodeSystem
102
- mainJsFile: String
103
- outputPath: String
104
- minify: Bool
105
- ): Unit
106
- target node async """
107
- import * as esbuild from 'esbuild'
108
- return await esbuild.build({
109
- entryPoints: [mainJsFile_],
110
- bundle: true,
111
- minify: minify_,
112
- sourcemap: true,
113
- platform: 'node',
114
- target: 'es6',
115
- external: ['../../../node_modules/*'], // TODO
116
- outfile: outputPath_
117
- })
118
- """
119
-
120
- internalListDirectory(path: Path): List[Pair[String, () => Stream[Buffer]]] {
121
- function go(currentPath: Path): Stream[Path] {
122
- currentPath.entries().flatMap {file =>
123
- if(file.isDirectory()) {
124
- go(file.path())
125
- } else {
126
- [file.path()].toStream()
127
- }
128
- }
129
- }
130
- go(path).map {file =>
131
- Pair("/" + file.relativeTo(path).replace("\\", "/"), {file.readStream()})
132
- }.toList()
133
- }
134
-
135
- internalPath(buildSystem: BuildSystem, absoluteOrRelative: String): Path
136
- target node async """
137
- return absoluteOrRelative_
138
- """
139
-
140
- internalCompile(buildSystem: BuildSystem, mainFile: Path, target: String): Unit
141
- target node async """
142
- return await $firefly_compiler.buildViaBuildSystem_$(buildSystem_, buildSystem_.fireflyPath_, mainFile_, target_, $task)
143
- """
144
-
145
- internalMainPackagePair(buildSystem: BuildSystem): Pair[String, String]
146
- target node async """
147
- return {first_: buildSystem_.mainPackagePair_.group_, second_: buildSystem_.mainPackagePair_.name_}
148
- """
1
+ capability BuildSystem {}
2
+ capability BrowserCode(packageGroup: String, packageName: String, mainFile: Path, assetSystem: AssetSystem)
3
+ capability BrowserBundle(assetSystem: AssetSystem)
4
+
5
+ extend self: BuildSystem {
6
+
7
+ compileForBrowser(mainFile: String): BrowserCode {
8
+ // TODO: Check that the mainFile is in the current package directory
9
+ internalCompile(self, internalPath(self, mainFile), "browser")
10
+ let streams = internalListDirectory(internalPath(self, ".firefly/output/browser"))
11
+ let mainPackagePair = internalMainPackagePair(self)
12
+ BrowserCode(
13
+ packageGroup = mainPackagePair.first
14
+ packageName = mainPackagePair.second
15
+ mainFile = internalPath(self, mainFile)
16
+ assetSystem = AssetSystem(streams.toMap())
17
+ )
18
+ }
19
+
20
+ buildMode(): Bool {
21
+ !!self!->"buildMode_"?
22
+ }
23
+
24
+ setAssets(assetSystem: AssetSystem): Unit {
25
+ self!->"assets_" = assetSystem!
26
+ }
27
+
28
+ packageAssets(): AssetSystem {
29
+ let streams = internalListDirectory(internalPath(self, "."))
30
+ AssetSystem(streams.toMap())
31
+ }
32
+
33
+ dependencyAssets(user: String, package: String): AssetSystem {
34
+ panic("dependencyAssets not yet implemented")
35
+ }
36
+
37
+ arguments(): List[String] {
38
+ self!->"array_"?
39
+ }
40
+
41
+ mainTask(): Task {
42
+ self!->"task_"?
43
+ }
44
+
45
+ crypto(): Crypto {
46
+ Js->crypto?
47
+ }
48
+
49
+ }
50
+
51
+ extend self: BrowserCode {
52
+
53
+ assets(): AssetSystem {
54
+ self.assetSystem
55
+ }
56
+
57
+ bundle(minify: Bool = True, sourceMap: Bool = False): BrowserBundle {
58
+ let prefix = ".firefly/output/browser"
59
+ let mainJsBaseFile = self.mainFile.base().removeLast(".ff").grab() + ".mjs"
60
+ let mainJsFile = prefix + "/" + self.packageGroup + "/" + self.packageName + "/" + mainJsBaseFile
61
+ let mainDirectory = self.mainFile.parent().grab()
62
+ let file = prefix + "/Main.bundle.js"
63
+ internalCallEsBuild(self, mainJsFile = mainJsFile, outputPath = file, minify = minify, sourceMap = sourceMap)
64
+ let assets = AssetSystem([
65
+ Pair(file.dropFirst(prefix.size()), {mainDirectory.path(file).readStream()})
66
+ ...if(sourceMap) {[
67
+ Pair(file.dropFirst(prefix.size()) + ".map", {mainDirectory.path(file + ".map").readStream()})
68
+ ]} else {[]}
69
+ ].toMap())
70
+ BrowserBundle(assets)
71
+ }
72
+
73
+ }
74
+
75
+ extend self: BrowserBundle {
76
+
77
+ assets(): AssetSystem {
78
+ self.assetSystem
79
+ }
80
+
81
+ }
82
+
83
+
84
+ internalCallEsBuild(
85
+ self: BrowserCode
86
+ mainJsFile: String
87
+ outputPath: String
88
+ minify: Bool
89
+ sourceMap: Bool
90
+ ): Unit {
91
+ let esbuild = Js.import("esbuild")
92
+ esbuild->build(Js->(
93
+ stdin = Js->(contents = "import {$run$} from './" + mainJsFile + "';", resolveDir = ".")
94
+ bundle = True
95
+ minify = minify
96
+ sourcemap = sourceMap
97
+ platform = "browser"
98
+ target = "es2017"
99
+ external = ["../../../node_modules/*"] // TODO
100
+ outfile = outputPath
101
+ ))
102
+ }
103
+
104
+ internalNodeCallEsBuild(
105
+ self: NodeSystem
106
+ mainJsFile: String
107
+ outputPath: String
108
+ minify: Bool
109
+ ): Unit {
110
+ let esbuild = Js.import("esbuild")
111
+ esbuild->build(Js->(
112
+ entryPoints = [mainJsFile]
113
+ bundle = True
114
+ minify = minify
115
+ sourcemap = True
116
+ platform = "node"
117
+ target = "es2017"
118
+ external = ["../../../node_modules/*"] // TODO
119
+ outfile = outputPath
120
+ ))
121
+ }
122
+
123
+ internalListDirectory(path: Path): List[Pair[String, () => Stream[Buffer]]] {
124
+ function go(currentPath: Path): Stream[Path] {
125
+ currentPath.entries().flatMap {file =>
126
+ if(file.isDirectory()) {
127
+ go(file.path())
128
+ } else {
129
+ [file.path()].toStream()
130
+ }
131
+ }
132
+ }
133
+ go(path).map {file =>
134
+ Pair("/" + file.relativeTo(path).replace("\\", "/"), {file.readStream()})
135
+ }.toList()
136
+ }
137
+
138
+ internalPath(buildSystem: BuildSystem, absoluteOrRelative: String): Path {
139
+ let nodePath = Js.import("path")
140
+ Path(nodePath->resolve(absoluteOrRelative)?)
141
+ }
142
+
143
+ internalCompile(buildSystem: BuildSystem, mainFile: Path, target: String): Unit {
144
+ // Ported from the old FFI. It's quite fragile w.r.t. changes in code generation.
145
+ Js.await(Js.rawIdentifier("$firefly_compiler")->"buildViaBuildSystem_$"(
146
+ buildSystem!
147
+ internalPath(buildSystem, buildSystem!->"fireflyPath_"?)!
148
+ mainFile.base()
149
+ target
150
+ Js.currentTask()!
151
+ ))
152
+ }
153
+
154
+ internalMainPackagePair(buildSystem: BuildSystem): Pair[String, String] {
155
+ Pair(buildSystem!->"mainPackagePair_"->"group_"?, buildSystem!->"mainPackagePair_"->"name_"?)
156
+ }
package/core/Channel.ff CHANGED
@@ -1,4 +1,9 @@
1
- capability Channel[T] {}
1
+ capability Channel[T](
2
+ capacity: Int
3
+ buffer: Array[T]
4
+ readers: JsValue
5
+ writers: JsValue
6
+ )
2
7
 
3
8
  extend self[T]: Channel[T] {
4
9
 
@@ -13,15 +18,22 @@ extend self[T]: Channel[T] {
13
18
  }
14
19
 
15
20
 
16
- capability ChannelAction[T] {}
21
+ capability ChannelAction[T](
22
+ channel: Channel[JsValue]
23
+ body: JsValue
24
+ message: Option[JsValue]
25
+ previous: Option[ChannelAction[T]]
26
+ )
17
27
 
18
28
  extend self[T]: ChannelAction[T] {
29
+
30
+ readOr[M](channel: Channel[M], body: M => T): ChannelAction[T] {
31
+ ChannelAction(channel!?, body!, None, Some(self))
32
+ }
19
33
 
20
- readOr[M](channel: Channel[M], body: M => T): ChannelAction[T]
21
- target js async "return {channel: channel_, body: body_, previous: self_}"
22
-
23
- writeOr[M](channel: Channel[M], message: M, body: () => T): ChannelAction[T]
24
- target js async "return {channel: channel_, body: body_, message: message_, previous: self_}"
34
+ writeOr[M](channel: Channel[M], message: M, body: () => T): ChannelAction[T] {
35
+ ChannelAction(channel!?, body!, Some(message!), Some(self))
36
+ }
25
37
 
26
38
  wait(): T {
27
39
  internalRunChannelAction(self, None)
@@ -37,95 +49,86 @@ extend self[T]: ChannelAction[T] {
37
49
 
38
50
  }
39
51
 
40
- readOr[M, T](channel: Channel[M], body: M => T): ChannelAction[T]
41
- target js async "return {channel: channel_, body: body_, previous: null}"
42
-
43
- writeOr[M, T](channel: Channel[M], message: M, body: () => T): ChannelAction[T]
44
- target js async "return {channel: channel_, body: body_, message: message_, previous: null}"
45
-
46
-
47
- internalRunChannelAction[T](action: ChannelAction[T], mode: Option[Pair[() => T, Option[Duration]]]): T
48
- target js async """
49
- ff_core_Task.Task_throwIfAborted($task)
52
+ readOr[M, T](channel: Channel[M], body: M => T): ChannelAction[T] {
53
+ ChannelAction(channel!?, body!, None, None)
54
+ }
50
55
 
51
- // Convert the linked actions into an array.
52
- let actions = []
53
- while(action_ != null) {
54
- actions.push(action_)
55
- action_ = action_.previous
56
- }
57
- actions.reverse()
56
+ writeOr[M, T](channel: Channel[M], message: M, body: () => T): ChannelAction[T] {
57
+ ChannelAction(channel!?, body!, Some(message!), None)
58
+ }
58
59
 
59
- // If any reads or writes can be done immediately, do the first one and return.
60
- for(let action of actions) {
61
- if(action.hasOwnProperty("message")) {
62
- if(action.channel.readers.size != 0) {
63
- let reader = action.channel.readers.values().next().value
64
- action.channel.readers.delete(reader)
65
- reader.resolve(action.message)
66
- return await action.body($task)
67
- } else if(action.channel.buffer.length < action.channel.capacity) {
68
- action.channel.buffer.push(action.message)
69
- return await action.body($task)
70
- }
71
- } else {
72
- if(action.channel.buffer.length != 0) {
73
- return await action.body(action.channel.buffer.shift(), $task)
74
- } else if(action.channel.writers.size != 0) {
75
- let writer = action.channel.writers.values().next().value
76
- action.channel.writers.delete(writer)
77
- writer.resolve()
78
- return await action.body(writer.message, $task)
79
- }
80
- }
60
+ internalRunChannelAction[T](action: ChannelAction[T], mode: Option[Pair[() => T, Option[Duration]]]): T {
61
+ Js.throwIfCancelled()
62
+
63
+ // Convert the linked actions into an array.
64
+ let actions = [].toArray()
65
+ function findActions(action: ChannelAction[T]) {
66
+ actions.push(action)
67
+ action.previous.each {findActions(_)}
68
+ }
69
+ findActions(action)
70
+
71
+ // If any reads or writes can be done immediately, do the first one and return.
72
+ mutable foundPromise = None
73
+ actions.eachWhile {action =>
74
+ action.message.{
75
+ | Some(message) {action.channel.readers->size !== 0} =>
76
+ let reader = action.channel.readers->values()->next()->value
77
+ action.channel.readers->delete(reader)
78
+ reader->resolve(message)
79
+ foundPromise = Some(action.body.callValue1(Js.currentTask()!))
80
+ | Some(message) {action.channel.buffer.size() < action.channel.capacity} =>
81
+ action.channel.buffer.push(message)
82
+ foundPromise = Some(action.body.callValue1(Js.currentTask()!))
83
+ | Some(_) =>
84
+ | None {action.channel.buffer.size() != 0} =>
85
+ action.channel.buffer.reverse()
86
+ let message = action.channel.buffer.pop()
87
+ action.channel.buffer.reverse()
88
+ foundPromise = Some(action.body.callValue2(message!, Js.currentTask()!))
89
+ | None {action.channel.writers->size !== 0} =>
90
+ let writer = action.channel.writers->values()->next()->value
91
+ action.channel.writers->delete(writer)
92
+ writer->resolve()
93
+ foundPromise = Some(action.body.callValue2(writer->message!, Js.currentTask()!))
94
+ | None =>
81
95
  }
82
-
83
- // If there's an "immediately(body)" action, do that now.
84
- if(mode_.value_ && mode_.value_.second_.value_ == null) return await mode_.value_.first_($task)
85
-
86
- // Otherwise, start waiting for one of the readers or writers (or timeout(body), or cancellation) to happen.
87
- let abort = null
88
- let finish = null
89
- let cleanups = []
90
- function doCleanup() {
91
- for(let cleanup of cleanups) cleanup()
96
+ foundPromise.isEmpty()
97
+ }
98
+ foundPromise.map {Js.await(_)?}.else:
99
+
100
+ // If there's an "immediately(body)" action, do that now.
101
+ if(mode.any {_.second.isEmpty()}) {
102
+ let makePromise = mode.grab().first
103
+ Js.await(makePromise()!)?
104
+ } else:
105
+
106
+ // Otherwise, start waiting for one of the readers or writers (or timeout(body), or cancellation) to happen.
107
+ let body = Js.awaitCancellablePromise {resolve, reject, onSettled =>
108
+ mode.each {m =>
109
+ let timeout = Js->setTimeout(Js->{resolve(m.first)}, m.second.grab().seconds * 1000.0)
110
+ onSettled {_ => Js->clearTimeout(timeout)}
92
111
  }
93
- let promise = new Promise((resolve, reject) => {
94
- if(mode_.value_) finish = () => {doCleanup(); resolve(() => mode_.value_.first_($task))}
95
- abort = () => {doCleanup(); reject($task.controller.signal.reason)}
96
- for(let action of actions) {
97
- if(action.hasOwnProperty("message")) {
98
- let writer = {
99
- resolve: () => {
100
- doCleanup()
101
- resolve(() => action.body($task))
102
- },
103
- message: action.message
104
- }
105
- cleanups.push(() => action.channel.writers.delete(writer))
106
- action.channel.writers.add(writer)
107
- } else {
108
- let reader = {
109
- resolve: m => {
110
- doCleanup()
111
- resolve(() => action.body(m, $task))
112
+ actions.each {action =>
113
+ action.message.{
114
+ | Some(message) =>
115
+ let writer = Js->(
116
+ resolve = Js->{resolve({action.body.callValue1(Js.currentTask()!)?})}
117
+ message = message
118
+ )
119
+ action.channel.writers->add(writer)
120
+ onSettled {_ => action.channel.writers->delete(writer)}
121
+ | None =>
122
+ let reader = Js->(
123
+ resolve = Js->{m =>
124
+ resolve({action.body.callValue2(m, Js.currentTask()!)?})
112
125
  }
113
- }
114
- cleanups.push(() => action.channel.readers.delete(reader))
115
- action.channel.readers.add(reader)
116
- }
126
+ )
127
+ action.channel.readers->add(reader)
128
+ onSettled {_ => action.channel.readers->delete(reader)}
117
129
  }
118
- })
119
- let timeout = null
120
- try {
121
- $task.controller.signal.addEventListener('abort', abort)
122
- if(finish != null) timeout = setTimeout(finish, mode_.value_.second_.value_ * 1000)
123
- let body = await promise
124
- if(timeout != null) { clearTimeout(timeout); timeout = null }
125
- return await body()
126
- } finally {
127
- if(timeout != null) clearTimeout(timeout)
128
- $task.controller.signal.removeEventListener('abort', abort)
129
- if($task.controller.signal.aborted) $task.controller = new AbortController()
130
130
  }
131
- """
131
+ }
132
+ body()
133
+
134
+ }
package/core/Char.ff CHANGED
@@ -12,7 +12,8 @@ extend self: Char {
12
12
 
13
13
  isAsciiDigit(): Bool {self >= '0' && self <= '9'}
14
14
 
15
- toString(): String
16
- target js sync "return String.fromCharCode(self_)"
15
+ toString(): String {
16
+ Js->String->fromCharCode(self)?
17
+ }
17
18
 
18
19
  }
package/core/Core.ff CHANGED
@@ -29,34 +29,27 @@ do[T](body: () => T): T {
29
29
  body()
30
30
  }
31
31
 
32
- try[T](body: () => T): Try[T]
33
- target js sync """
34
- try {
35
- return {Success: true, value_: body_()}
36
- } catch(e) {
37
- return {Failure: true, error_: e}
38
- }
39
- """
40
- target js async """
41
- try {
42
- return {Success: true, value_: await body_($task)}
43
- } catch(e) {
44
- return {Failure: true, error_: e}
45
- }
46
- """
32
+ try[T](body: () => T): Try[T] {
33
+ // Special cased by the compiler
34
+ try {
35
+ Success(body())
36
+ } catchAny {e =>
37
+ Failure(e)
38
+ }
39
+ }
47
40
 
48
41
  throw[E: HasAnyTag, T](exception: E): T {
49
42
  throwAny(Any.toAny(exception))
50
43
  }
51
44
 
52
- throwAny[T](exception: Any): T
53
- target js sync "throw Object.assign(new Error(), {ffException: exception_})"
54
-
55
- panic[T](message: String): T
56
- target js sync "throw new Error(message_)"
45
+ throwAny[T](exception: Any): T {
46
+ let e = Js->Error->()
47
+ e->ffException = exception!
48
+ Js.throw(e)
49
+ }
57
50
 
51
+ panic[T](message: String): T {
52
+ Js.throw(Js->Error->(message))
53
+ }
58
54
 
59
- same[T](x: T, y: T): Bool
60
- target js sync """return x_ === y_"""
61
-
62
55
  data GrabException()