firefly-compiler 0.4.40 → 0.4.48

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 (109) hide show
  1. package/.hintrc +4 -4
  2. package/.vscode/settings.json +4 -4
  3. package/bin/Release.ff +136 -71
  4. package/bin/firefly.mjs +1 -1
  5. package/compiler/Builder.ff +257 -257
  6. package/compiler/Compiler.ff +227 -227
  7. package/compiler/Dependencies.ff +186 -186
  8. package/compiler/DependencyLock.ff +17 -17
  9. package/compiler/JsEmitter.ff +946 -946
  10. package/compiler/LspHook.ff +202 -202
  11. package/compiler/ModuleCache.ff +178 -178
  12. package/compiler/Workspace.ff +88 -88
  13. package/core/.firefly/include/package-lock.json +394 -394
  14. package/core/.firefly/include/package.json +5 -5
  15. package/core/.firefly/include/prepare.sh +1 -1
  16. package/core/.firefly/package.ff +2 -2
  17. package/core/Array.ff +265 -265
  18. package/core/Atomic.ff +64 -64
  19. package/core/Box.ff +7 -7
  20. package/core/BrowserSystem.ff +40 -40
  21. package/core/BuildSystem.ff +148 -148
  22. package/core/Crypto.ff +96 -102
  23. package/core/Equal.ff +36 -36
  24. package/core/HttpClient.ff +87 -87
  25. package/core/JsSystem.ff +69 -69
  26. package/core/Json.ff +434 -434
  27. package/core/List.ff +486 -415
  28. package/core/Lock.ff +144 -144
  29. package/core/NodeSystem.ff +189 -189
  30. package/core/Ordering.ff +161 -161
  31. package/core/Path.ff +401 -401
  32. package/core/Random.ff +134 -134
  33. package/core/RbMap.ff +216 -216
  34. package/core/Show.ff +43 -43
  35. package/core/SourceLocation.ff +68 -68
  36. package/core/Task.ff +141 -141
  37. package/experimental/benchmarks/ListGrab.ff +23 -23
  38. package/experimental/benchmarks/ListGrab.java +55 -55
  39. package/experimental/benchmarks/Pyrotek45.ff +30 -30
  40. package/experimental/benchmarks/Pyrotek45.java +64 -64
  41. package/experimental/bidirectional/Bidi.ff +88 -88
  42. package/experimental/random/Index.ff +53 -53
  43. package/experimental/random/Process.ff +120 -120
  44. package/experimental/random/Scrape.ff +51 -51
  45. package/experimental/random/Symbols.ff +73 -73
  46. package/experimental/random/Tensor.ff +52 -52
  47. package/experimental/random/Units.ff +36 -36
  48. package/experimental/s3/S3TestAuthorizationHeader.ff +38 -38
  49. package/experimental/s3/S3TestPut.ff +15 -15
  50. package/experimental/tests/TestJson.ff +26 -26
  51. package/firefly.sh +0 -0
  52. package/fireflysite/Main.ff +13 -13
  53. package/lsp/.firefly/package.ff +1 -1
  54. package/lsp/CompletionHandler.ff +808 -811
  55. package/lsp/Handler.ff +714 -714
  56. package/lsp/HoverHandler.ff +79 -79
  57. package/lsp/LanguageServer.ff +272 -272
  58. package/lsp/SignatureHelpHandler.ff +55 -55
  59. package/lsp/SymbolHandler.ff +181 -181
  60. package/lsp/TestReferences.ff +16 -16
  61. package/lsp/TestReferencesCase.ff +7 -7
  62. package/lsp/stderr.txt +1 -1
  63. package/lsp/stdin.txt +10 -10
  64. package/lsp/stdout.txt +40 -40
  65. package/lux/.firefly/package.ff +1 -1
  66. package/lux/Css.ff +648 -648
  67. package/lux/CssTest.ff +48 -48
  68. package/lux/Lux.ff +487 -487
  69. package/lux/LuxEvent.ff +116 -116
  70. package/lux/Main.ff +128 -128
  71. package/lux/Main2.ff +144 -144
  72. package/output/js/ff/compiler/Builder.mjs +43 -43
  73. package/output/js/ff/compiler/Dependencies.mjs +3 -3
  74. package/output/js/ff/core/Array.mjs +59 -59
  75. package/output/js/ff/core/Atomic.mjs +36 -36
  76. package/output/js/ff/core/BrowserSystem.mjs +11 -11
  77. package/output/js/ff/core/BuildSystem.mjs +30 -30
  78. package/output/js/ff/core/Crypto.mjs +58 -62
  79. package/output/js/ff/core/HttpClient.mjs +24 -24
  80. package/output/js/ff/core/Json.mjs +147 -147
  81. package/output/js/ff/core/List.mjs +196 -50
  82. package/output/js/ff/core/Lock.mjs +97 -97
  83. package/output/js/ff/core/NodeSystem.mjs +77 -77
  84. package/output/js/ff/core/Ordering.mjs +8 -8
  85. package/output/js/ff/core/Path.mjs +231 -231
  86. package/output/js/ff/core/Random.mjs +56 -56
  87. package/output/js/ff/core/Task.mjs +31 -31
  88. package/package.json +29 -29
  89. package/rpc/.firefly/package.ff +1 -1
  90. package/rpc/Rpc.ff +69 -69
  91. package/s3/.firefly/package.ff +1 -1
  92. package/s3/S3.ff +93 -93
  93. package/unsafejs/UnsafeJs.ff +19 -19
  94. package/vscode/LICENSE.txt +21 -21
  95. package/vscode/Prepublish.ff +15 -15
  96. package/vscode/README.md +16 -16
  97. package/vscode/client/package.json +22 -22
  98. package/vscode/client/src/extension.ts +104 -104
  99. package/vscode/icons/firefly-icon.svg +10 -10
  100. package/vscode/language-configuration.json +61 -61
  101. package/vscode/package-lock.json +3623 -3623
  102. package/vscode/package.json +160 -160
  103. package/vscode/snippets.json +241 -241
  104. package/webserver/.firefly/include/package-lock.json +16 -16
  105. package/webserver/.firefly/include/package.json +5 -5
  106. package/webserver/.firefly/package.ff +2 -2
  107. package/webserver/WebServer.ff +685 -685
  108. package/websocket/.firefly/package.ff +1 -1
  109. package/websocket/WebSocket.ff +131 -131
@@ -110,36 +110,36 @@ return ff_core_Random.seedFloat_(seed_)
110
110
  }
111
111
 
112
112
  export function seedBuffer_(buffer_) {
113
-
114
- var n = 0xefc8249d;
115
- function mash(data) {
116
- for(var i = 0; i < data.byteLength; i++) {
117
- n += data.getUint8(i);
118
- var h = 0.02519603282416938 * n;
119
- n = h >>> 0;
120
- h -= n;
121
- h *= n;
122
- n = h >>> 0;
123
- h -= n;
124
- n += h * 0x100000000; // 2^32
125
- }
126
- return (n >>> 0) * 2.3283064365386963e-10; // 2^-32
127
- }
128
- var space = new DataView(new Uint8Array([32]).buffer);
129
- var r = {
130
- s0: mash(space),
131
- s1: mash(space),
132
- s2: mash(space),
133
- c: 1,
134
- spareGauss: NaN
135
- };
136
- r.s0 -= mash(buffer_);
137
- if(r.s0 < 0) r.s0 += 1;
138
- r.s1 -= mash(buffer_);
139
- if(r.s1 < 0) r.s1 += 1;
140
- r.s2 -= mash(buffer_);
141
- if(r.s2 < 0) r.s2 += 1;
142
- return r;
113
+
114
+ var n = 0xefc8249d;
115
+ function mash(data) {
116
+ for(var i = 0; i < data.byteLength; i++) {
117
+ n += data.getUint8(i);
118
+ var h = 0.02519603282416938 * n;
119
+ n = h >>> 0;
120
+ h -= n;
121
+ h *= n;
122
+ n = h >>> 0;
123
+ h -= n;
124
+ n += h * 0x100000000; // 2^32
125
+ }
126
+ return (n >>> 0) * 2.3283064365386963e-10; // 2^-32
127
+ }
128
+ var space = new DataView(new Uint8Array([32]).buffer);
129
+ var r = {
130
+ s0: mash(space),
131
+ s1: mash(space),
132
+ s2: mash(space),
133
+ c: 1,
134
+ spareGauss: NaN
135
+ };
136
+ r.s0 -= mash(buffer_);
137
+ if(r.s0 < 0) r.s0 += 1;
138
+ r.s1 -= mash(buffer_);
139
+ if(r.s1 < 0) r.s1 += 1;
140
+ r.s2 -= mash(buffer_);
141
+ if(r.s2 < 0) r.s2 += 1;
142
+ return r;
143
143
 
144
144
  }
145
145
 
@@ -162,24 +162,24 @@ throw new Error('Function seedBuffer is missing on this target in async context.
162
162
  }
163
163
 
164
164
  export function Random_copy(self_) {
165
-
166
- return {...self_};
165
+
166
+ return {...self_};
167
167
 
168
168
  }
169
169
 
170
170
  export function Random_nextInt(self_, from_, until_) {
171
-
172
- return Random_nextFloat(self_, from_, until_) | 0;
171
+
172
+ return Random_nextFloat(self_, from_, until_) | 0;
173
173
 
174
174
  }
175
175
 
176
176
  export function Random_nextFloat(self_, from_, until_) {
177
-
178
- var t = 2091639 * self_.s0 + self_.c * 2.3283064365386963e-10; // 2^-32
179
- self_.s0 = self_.s1;
180
- self_.s1 = self_.s2;
181
- var uniform = self_.s2 = t - (self_.c = t | 0);
182
- return from_ + uniform * (until_ - from_);
177
+
178
+ var t = 2091639 * self_.s0 + self_.c * 2.3283064365386963e-10; // 2^-32
179
+ self_.s0 = self_.s1;
180
+ self_.s1 = self_.s2;
181
+ var uniform = self_.s2 = t - (self_.c = t | 0);
182
+ return from_ + uniform * (until_ - from_);
183
183
 
184
184
  }
185
185
 
@@ -194,22 +194,22 @@ ff_core_Buffer.Buffer_setUint8(buffer_, i_, ff_core_Random.Random_nextInt(self_,
194
194
  }
195
195
 
196
196
  export function Random_nextGauss(self_, mean_, standardDeviation_) {
197
-
198
- if(!isNaN(self_.spareGauss)) {
199
- const result = self_.spareGauss * standardDeviation_ + mean_;
200
- self_.spareGauss = NaN;
201
- return result;
202
- } else {
203
- let u = 0.5, v = 0.5, s = 0.5;
204
- do {
205
- u = Random_nextFloat(self_, 0.0, 1.0) * 2 - 1;
206
- v = Random_nextFloat(self_, 0.0, 1.0) * 2 - 1;
207
- s = u * u + v * v;
208
- } while(s >= 1 || s == 0);
209
- s = Math.sqrt(-2.0 * Math.log(s) / s);
210
- self_.spareGauss = v * s;
211
- return mean_ + standardDeviation_ * u * s;
212
- }
197
+
198
+ if(!isNaN(self_.spareGauss)) {
199
+ const result = self_.spareGauss * standardDeviation_ + mean_;
200
+ self_.spareGauss = NaN;
201
+ return result;
202
+ } else {
203
+ let u = 0.5, v = 0.5, s = 0.5;
204
+ do {
205
+ u = Random_nextFloat(self_, 0.0, 1.0) * 2 - 1;
206
+ v = Random_nextFloat(self_, 0.0, 1.0) * 2 - 1;
207
+ s = u * u + v * v;
208
+ } while(s >= 1 || s == 0);
209
+ s = Math.sqrt(-2.0 * Math.log(s) / s);
210
+ self_.spareGauss = v * s;
211
+ return mean_ + standardDeviation_ * u * s;
212
+ }
213
213
 
214
214
  }
215
215
 
@@ -100,39 +100,39 @@ import * as ff_core_Unit from "../../ff/core/Unit.mjs"
100
100
 
101
101
 
102
102
  export function Task_spawn(self_, body_) {
103
-
104
- ff_core_Task.Task_throwIfAborted(self_)
105
- const task = {controller: new AbortController(), subtasks: new Set(), started: performance.now() * 0.001}
106
- self_.subtasks.add(task)
107
- task.promise = Promise.resolve(task).then(async () => {
108
- try {
109
- await body_(task, task)
110
- } catch(e) {
111
- await ff_core_Task.Task_abort$(self_)
112
- throw e
113
- } finally {
114
- for(const subtask of task.subtasks) subtask.controller.abort()
115
- await Promise.allSettled([...task.subtasks].map(subtask => subtask.promise))
116
- self_.subtasks.delete(task)
117
- }
118
- })
119
- return task
103
+
104
+ ff_core_Task.Task_throwIfAborted(self_)
105
+ const task = {controller: new AbortController(), subtasks: new Set(), started: performance.now() * 0.001}
106
+ self_.subtasks.add(task)
107
+ task.promise = Promise.resolve(task).then(async () => {
108
+ try {
109
+ await body_(task, task)
110
+ } catch(e) {
111
+ await ff_core_Task.Task_abort$(self_)
112
+ throw e
113
+ } finally {
114
+ for(const subtask of task.subtasks) subtask.controller.abort()
115
+ await Promise.allSettled([...task.subtasks].map(subtask => subtask.promise))
116
+ self_.subtasks.delete(task)
117
+ }
118
+ })
119
+ return task
120
120
 
121
121
  }
122
122
 
123
123
  export function Task_throwIfAborted(self_) {
124
-
125
- if(self_.controller.signal.aborted) {
126
- const signal = self_.controller.signal
127
- self_.controller = new AbortController()
128
- signal.throwIfAborted()
129
- }
124
+
125
+ if(self_.controller.signal.aborted) {
126
+ const signal = self_.controller.signal
127
+ self_.controller = new AbortController()
128
+ signal.throwIfAborted()
129
+ }
130
130
 
131
131
  }
132
132
 
133
133
  export function Task_abort(self_) {
134
-
135
- self_.controller.abort()
134
+
135
+ self_.controller.abort()
136
136
 
137
137
  }
138
138
 
@@ -161,20 +161,20 @@ return ff_core_Pair.Pair(result_, duration_)
161
161
  }
162
162
 
163
163
  export async function Task_spawn$(self_, body_, $task) {
164
-
165
- return ff_core_Task.Task_spawn(self_, body_)
164
+
165
+ return ff_core_Task.Task_spawn(self_, body_)
166
166
 
167
167
  }
168
168
 
169
169
  export async function Task_throwIfAborted$(self_, $task) {
170
-
171
- ff_core_Task.Task_throwIfAborted(self_)
170
+
171
+ ff_core_Task.Task_throwIfAborted(self_)
172
172
 
173
173
  }
174
174
 
175
175
  export async function Task_abort$(self_, $task) {
176
-
177
- self_.controller.abort()
176
+
177
+ self_.controller.abort()
178
178
 
179
179
  }
180
180
 
package/package.json CHANGED
@@ -1,29 +1,29 @@
1
- {
2
- "name": "firefly-compiler",
3
- "displayName": "Firefly Compiler",
4
- "description": "Firefly compiler",
5
- "author": "Firefly team",
6
- "license": "MIT",
7
- "version": "0.4.40",
8
- "repository": {
9
- "type": "git",
10
- "url": "https://github.com/Ahnfelt/firefly-boot"
11
- },
12
- "publisher": "firefly-team",
13
- "categories": [
14
- "Programming Languages"
15
- ],
16
- "engines": {
17
- "node": ">=18.0.0"
18
- },
19
- "main": "./vscode/client/out/extension",
20
- "icon": "./vscode/icons/firefly-logo.png",
21
- "dependencies": {
22
- "pkg": "^5.8.0",
23
- "tar": "^6.1.11",
24
- "esbuild": "^0.21.3"
25
- },
26
- "bin": {
27
- "firefly": "./bin/firefly.mjs"
28
- }
29
- }
1
+ {
2
+ "name": "firefly-compiler",
3
+ "displayName": "Firefly Compiler",
4
+ "description": "Firefly compiler",
5
+ "author": "Firefly team",
6
+ "license": "MIT",
7
+ "version": "0.4.48",
8
+ "repository": {
9
+ "type": "git",
10
+ "url": "https://github.com/Ahnfelt/firefly-boot"
11
+ },
12
+ "publisher": "firefly-team",
13
+ "categories": [
14
+ "Programming Languages"
15
+ ],
16
+ "engines": {
17
+ "node": ">=18.0.0"
18
+ },
19
+ "main": "./vscode/client/out/extension",
20
+ "icon": "./vscode/icons/firefly-logo.png",
21
+ "dependencies": {
22
+ "pkg": "^5.8.0",
23
+ "tar": "^6.1.11",
24
+ "esbuild": "^0.21.3"
25
+ },
26
+ "bin": {
27
+ "firefly": "./bin/firefly.mjs"
28
+ }
29
+ }
@@ -1 +1 @@
1
- package ff:rpc:0.0.0
1
+ package ff:rpc:0.0.0
package/rpc/Rpc.ff CHANGED
@@ -1,69 +1,69 @@
1
- trait I: Rpc[O] {}
2
-
3
- capability RpcServer[I, O, C](
4
- handlers: StringMap[(C, I) => O]
5
- )
6
-
7
- newServer[I, O, C](): RpcServer[I, O, C] {
8
- RpcServer(StringMap.new())
9
- }
10
-
11
- extend self[I, O, C]: RpcServer[I, O, C] {
12
- handle(name: String, context: C, input: I): Option[O] {
13
- self.handlers.get(name).map {handler =>
14
- handler(context, input)
15
- }
16
- }
17
-
18
- addCustom(
19
- handler: (C, I) => O
20
- ): Unit {
21
- let name = "TODO"
22
- self.handlers.set(name) {context, input =>
23
- handler(context, input)
24
- }
25
- }
26
- }
27
-
28
- extend self[C]: RpcServer[Buffer, Buffer, C] {
29
- add[I: Rpc[O]: HasAnyTag: Serializable, O: Serializable](
30
- handler: (C, I) => O
31
- ): Unit {
32
- let name = Any.anyTag[I]().show()
33
- self.handlers.set(name) {context, message =>
34
- let input = Serializable.deserialize(message)
35
- let output = handler(context, input)
36
- Serializable.serialize(output)
37
- }
38
- }
39
- }
40
-
41
- capability RpcClient(
42
- binaryCall: (String, Buffer) => Buffer
43
- )
44
-
45
- extend self: RpcClient {
46
- call[I: Rpc[O]: HasAnyTag: Serializable, O: Serializable](
47
- message: I
48
- ): O {
49
- let name = Any.anyTag[I]().show()
50
- let input = Serializable.serialize(message)
51
- let response = self.binaryCall(name, input)
52
- Serializable.deserialize(response)
53
- }
54
- }
55
-
56
- newClient(
57
- httpClient: HttpClient
58
- prefix: String
59
- method: String = "POST"
60
- ): RpcClient {
61
- RpcClient {name, buffer =>
62
- let response = httpClient.fetch(
63
- url = prefix + name,
64
- method = method,
65
- body = Some(HttpClient.bodyBuffer(buffer))
66
- )
67
- response.readBuffer()
68
- }
69
- }
1
+ trait I: Rpc[O] {}
2
+
3
+ capability RpcServer[I, O, C](
4
+ handlers: StringMap[(C, I) => O]
5
+ )
6
+
7
+ newServer[I, O, C](): RpcServer[I, O, C] {
8
+ RpcServer(StringMap.new())
9
+ }
10
+
11
+ extend self[I, O, C]: RpcServer[I, O, C] {
12
+ handle(name: String, context: C, input: I): Option[O] {
13
+ self.handlers.get(name).map {handler =>
14
+ handler(context, input)
15
+ }
16
+ }
17
+
18
+ addCustom(
19
+ handler: (C, I) => O
20
+ ): Unit {
21
+ let name = "TODO"
22
+ self.handlers.set(name) {context, input =>
23
+ handler(context, input)
24
+ }
25
+ }
26
+ }
27
+
28
+ extend self[C]: RpcServer[Buffer, Buffer, C] {
29
+ add[I: Rpc[O]: HasAnyTag: Serializable, O: Serializable](
30
+ handler: (C, I) => O
31
+ ): Unit {
32
+ let name = Any.anyTag[I]().show()
33
+ self.handlers.set(name) {context, message =>
34
+ let input = Serializable.deserialize(message)
35
+ let output = handler(context, input)
36
+ Serializable.serialize(output)
37
+ }
38
+ }
39
+ }
40
+
41
+ capability RpcClient(
42
+ binaryCall: (String, Buffer) => Buffer
43
+ )
44
+
45
+ extend self: RpcClient {
46
+ call[I: Rpc[O]: HasAnyTag: Serializable, O: Serializable](
47
+ message: I
48
+ ): O {
49
+ let name = Any.anyTag[I]().show()
50
+ let input = Serializable.serialize(message)
51
+ let response = self.binaryCall(name, input)
52
+ Serializable.deserialize(response)
53
+ }
54
+ }
55
+
56
+ newClient(
57
+ httpClient: HttpClient
58
+ prefix: String
59
+ method: String = "POST"
60
+ ): RpcClient {
61
+ RpcClient {name, buffer =>
62
+ let response = httpClient.fetch(
63
+ url = prefix + name,
64
+ method = method,
65
+ body = Some(HttpClient.bodyBuffer(buffer))
66
+ )
67
+ response.readBuffer()
68
+ }
69
+ }
@@ -1 +1 @@
1
- package ff:s3:0.0.0
1
+ package ff:s3:0.0.0
package/s3/S3.ff CHANGED
@@ -1,93 +1,93 @@
1
- put(
2
- system: NodeSystem
3
- accessKeyId: String
4
- secretAccessKey: String
5
- region: String
6
- bucket: String
7
- objectKey: String
8
- body: Buffer
9
- headers: List[Pair[String, String]]
10
- ): Unit {
11
- let now = system.mainTask().now()
12
- let amzDate = toIsoString(now)
13
- let contentHash = system.crypto().sha256(body).toHex()
14
- let host = bucket + ".s3.amazonaws.com"
15
- let encodedKey = encode(objectKey)
16
- let canonicalHeaders = [
17
- Pair("host", host)
18
- Pair("x-amz-date", amzDate)
19
- Pair("x-amz-content-sha256", contentHash)
20
- ...headers
21
- ]
22
- let authenticationHeader = makeS3AuthorizationHeader(system, accessKeyId, secretAccessKey, region, bucket, encodedKey, body, canonicalHeaders, amzDate)
23
- let allHeaders = [...canonicalHeaders, Pair("Authorization", authenticationHeader)]
24
-
25
- let url = "https://" + host + "/" + encodedKey
26
- let response = system.httpClient().fetch(url, method = "PUT", headers = allHeaders, body = Some(HttpClient.bodyBuffer(body)), throw = False)
27
- system.writeLine("" + response.status())
28
- system.writeLine(response.statusText())
29
- system.writeLine(response.readText())
30
- }
31
-
32
- makeS3AuthorizationHeader(
33
- system: NodeSystem
34
- accessKeyId: String
35
- secretAccessKey: String
36
- region: String
37
- bucket: String
38
- encodedKey: String
39
- body: Buffer
40
- canonicalHeaders: List[Pair[String, String]]
41
- amzDate: String
42
- ): String {
43
- // CanonicalRequest
44
- let canonicalQuerystring = "";
45
- let contentHash = system.crypto().sha256(body).toHex()
46
- let headers = canonicalHeaders.map {_.mapFirst {_.lower()}.mapSecond {_.trim()}}.sort()
47
- let signedHeaders = headers.map {_.first}.join(";")
48
- let canonicalRequest = [
49
- "PUT"
50
- "/" + encodedKey
51
- canonicalQuerystring
52
- ...headers.map {p => p.first + ":" + p.second}
53
- ""
54
- signedHeaders
55
- contentHash
56
- ].join("\n")
57
-
58
- // StringToSign
59
- let scopeDate = amzDate.slice(0, 8) // YYYYMMDD
60
- let credentialScope = [scopeDate, region, "s3", "aws4_request"].join("/")
61
- let algorithm = "AWS4-HMAC-SHA256"
62
- let stringToSign = [
63
- algorithm
64
- amzDate
65
- credentialScope
66
- system.crypto().sha256(canonicalRequest.toBuffer()).toHex()
67
- ].join("\n")
68
-
69
- function hmacSha256(key: Buffer, message: String): Buffer {
70
- system.crypto().hmacSha256(key, message.toBuffer())
71
- }
72
-
73
- // Signature
74
- let signingKey = hmacSha256(hmacSha256(hmacSha256(hmacSha256(("AWS4" + secretAccessKey).toBuffer(), scopeDate), region), "s3"), "aws4_request")
75
- let signature = hmacSha256(signingKey, stringToSign).toHex()
76
-
77
- algorithm + " Credential=" + accessKeyId + "/" + credentialScope + ",SignedHeaders=" + signedHeaders + ",Signature=" + signature
78
- }
79
-
80
- toIsoString(instant: Instant): String
81
- target js sync """
82
- const d = new Date(instant_ * 1000);
83
- return d.toISOString().replaceAll(/[-:]|[.].../g, "");
84
- """
85
-
86
- encodeURIComponent(text: String): String
87
- target js sync """
88
- return encodeURIComponent(text_);
89
- """
90
-
91
- encode(text: String): String {
92
- text.split('/').map {encodeURIComponent(_)}.join("/")
93
- }
1
+ put(
2
+ system: NodeSystem
3
+ accessKeyId: String
4
+ secretAccessKey: String
5
+ region: String
6
+ bucket: String
7
+ objectKey: String
8
+ body: Buffer
9
+ headers: List[Pair[String, String]]
10
+ ): Unit {
11
+ let now = system.mainTask().now()
12
+ let amzDate = toIsoString(now)
13
+ let contentHash = system.crypto().sha256(body).toHex()
14
+ let host = bucket + ".s3.amazonaws.com"
15
+ let encodedKey = encode(objectKey)
16
+ let canonicalHeaders = [
17
+ Pair("host", host)
18
+ Pair("x-amz-date", amzDate)
19
+ Pair("x-amz-content-sha256", contentHash)
20
+ ...headers
21
+ ]
22
+ let authenticationHeader = makeS3AuthorizationHeader(system, accessKeyId, secretAccessKey, region, bucket, encodedKey, body, canonicalHeaders, amzDate)
23
+ let allHeaders = [...canonicalHeaders, Pair("Authorization", authenticationHeader)]
24
+
25
+ let url = "https://" + host + "/" + encodedKey
26
+ let response = system.httpClient().fetch(url, method = "PUT", headers = allHeaders, body = Some(HttpClient.bodyBuffer(body)), throw = False)
27
+ system.writeLine("" + response.status())
28
+ system.writeLine(response.statusText())
29
+ //system.writeLine(response.readText())
30
+ }
31
+
32
+ makeS3AuthorizationHeader(
33
+ system: NodeSystem
34
+ accessKeyId: String
35
+ secretAccessKey: String
36
+ region: String
37
+ bucket: String
38
+ encodedKey: String
39
+ body: Buffer
40
+ canonicalHeaders: List[Pair[String, String]]
41
+ amzDate: String
42
+ ): String {
43
+ // CanonicalRequest
44
+ let canonicalQuerystring = "";
45
+ let contentHash = system.crypto().sha256(body).toHex()
46
+ let headers = canonicalHeaders.map {_.mapFirst {_.lower()}.mapSecond {_.trim()}}.sort()
47
+ let signedHeaders = headers.map {_.first}.join(";")
48
+ let canonicalRequest = [
49
+ "PUT"
50
+ "/" + encodedKey
51
+ canonicalQuerystring
52
+ ...headers.map {p => p.first + ":" + p.second}
53
+ ""
54
+ signedHeaders
55
+ contentHash
56
+ ].join("\n")
57
+
58
+ // StringToSign
59
+ let scopeDate = amzDate.slice(0, 8) // YYYYMMDD
60
+ let credentialScope = [scopeDate, region, "s3", "aws4_request"].join("/")
61
+ let algorithm = "AWS4-HMAC-SHA256"
62
+ let stringToSign = [
63
+ algorithm
64
+ amzDate
65
+ credentialScope
66
+ system.crypto().sha256(canonicalRequest.toBuffer()).toHex()
67
+ ].join("\n")
68
+
69
+ function hmacSha256(key: Buffer, message: String): Buffer {
70
+ system.crypto().hmacSha256(key, message.toBuffer())
71
+ }
72
+
73
+ // Signature
74
+ let signingKey = hmacSha256(hmacSha256(hmacSha256(hmacSha256(("AWS4" + secretAccessKey).toBuffer(), scopeDate), region), "s3"), "aws4_request")
75
+ let signature = hmacSha256(signingKey, stringToSign).toHex()
76
+
77
+ algorithm + " Credential=" + accessKeyId + "/" + credentialScope + ",SignedHeaders=" + signedHeaders + ",Signature=" + signature
78
+ }
79
+
80
+ toIsoString(instant: Instant): String
81
+ target js sync """
82
+ const d = new Date(instant_ * 1000);
83
+ return d.toISOString().replaceAll(/[-:]|[.].../g, "");
84
+ """
85
+
86
+ encodeURIComponent(text: String): String
87
+ target js sync """
88
+ return encodeURIComponent(text_);
89
+ """
90
+
91
+ encode(text: String): String {
92
+ text.split('/').map {encodeURIComponent(_)}.join("/")
93
+ }
@@ -1,19 +1,19 @@
1
- jsSystem(): JsSystem
2
- target js async "return typeof globalThis !== 'undefined' ? globalThis : window"
3
- target js sync "return typeof globalThis !== 'undefined' ? globalThis : window"
4
-
5
- import(module: String): JsValue
6
- // Replaced by the compiler ~ hoisted top level import - dynamic imports not currently supported
7
- target js sync "throw Error('Dynamic JS imports are not currently supported.')"
8
-
9
- await[T](body: () => T): T // This is never actually called async, but will be rewitten by the compiler
10
- // Replaced by the compiler
11
- target js sync "return body_()"
12
-
13
- throwIfCancelled(): Unit
14
- // Replaced by the compiler
15
- target js sync ""
16
-
17
- cancelled(): Bool
18
- // Replaced by the compiler
19
- target js sync "return false"
1
+ jsSystem(): JsSystem
2
+ target js async "return typeof globalThis !== 'undefined' ? globalThis : window"
3
+ target js sync "return typeof globalThis !== 'undefined' ? globalThis : window"
4
+
5
+ import(module: String): JsValue
6
+ // Replaced by the compiler ~ hoisted top level import - dynamic imports not currently supported
7
+ target js sync "throw Error('Dynamic JS imports are not currently supported.')"
8
+
9
+ await[T](body: () => T): T // This is never actually called async, but will be rewitten by the compiler
10
+ // Replaced by the compiler
11
+ target js sync "return body_()"
12
+
13
+ throwIfCancelled(): Unit
14
+ // Replaced by the compiler
15
+ target js sync ""
16
+
17
+ cancelled(): Bool
18
+ // Replaced by the compiler
19
+ target js sync "return false"