@theia/core 1.53.0-next.55 → 1.53.0-next.64

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 (581) hide show
  1. package/README.md +6 -6
  2. package/i18n/nls.cs.json +582 -582
  3. package/i18n/nls.de.json +582 -582
  4. package/i18n/nls.es.json +582 -582
  5. package/i18n/nls.fr.json +582 -582
  6. package/i18n/nls.hu.json +582 -582
  7. package/i18n/nls.it.json +582 -582
  8. package/i18n/nls.ja.json +582 -582
  9. package/i18n/nls.json +582 -582
  10. package/i18n/nls.ko.json +582 -582
  11. package/i18n/nls.pl.json +582 -582
  12. package/i18n/nls.pt-br.json +582 -582
  13. package/i18n/nls.ru.json +582 -582
  14. package/i18n/nls.tr.json +582 -582
  15. package/i18n/nls.zh-cn.json +582 -582
  16. package/i18n/nls.zh-tw.json +582 -582
  17. package/lib/browser/catalog.json +54 -6
  18. package/lib/browser/common-frontend-contribution.js +3 -3
  19. package/lib/browser/common-styling-participants.js +166 -166
  20. package/lib/browser/progress-location-service.spec.js +7 -7
  21. package/lib/browser/storage-service.js +3 -3
  22. package/lib/browser/tree/tree.spec.js +75 -75
  23. package/lib/node/process-utils.spec.js +8 -8
  24. package/package.json +4 -4
  25. package/shared/ajv/index.d.ts +2 -2
  26. package/shared/markdown-it.d.ts +2 -2
  27. package/shared/markdown-it.js +1 -1
  28. package/shared/reflect-metadata/index.d.ts +1 -1
  29. package/shared/reflect-metadata/index.js +1 -1
  30. package/shared/vscode-languageserver-types/index.d.ts +1 -1
  31. package/src/browser/about-dialog.tsx +137 -137
  32. package/src/browser/authentication-service.ts +467 -467
  33. package/src/browser/breadcrumbs/breadcrumb-popup-container.ts +101 -101
  34. package/src/browser/breadcrumbs/breadcrumb-renderer.tsx +41 -41
  35. package/src/browser/breadcrumbs/breadcrumbs-constants.ts +79 -79
  36. package/src/browser/breadcrumbs/breadcrumbs-renderer.tsx +185 -185
  37. package/src/browser/breadcrumbs/breadcrumbs-service.ts +108 -108
  38. package/src/browser/breadcrumbs/index.ts +21 -21
  39. package/src/browser/browser-clipboard-service.ts +122 -122
  40. package/src/browser/browser.ts +239 -239
  41. package/src/browser/clipboard-service.ts +23 -23
  42. package/src/browser/color-application-contribution.ts +110 -110
  43. package/src/browser/color-registry.ts +60 -60
  44. package/src/browser/command-open-handler.ts +54 -54
  45. package/src/browser/common-frontend-contribution.ts +2680 -2680
  46. package/src/browser/common-styling-participants.ts +361 -361
  47. package/src/browser/connection-status-service.spec.ts +200 -200
  48. package/src/browser/connection-status-service.ts +216 -216
  49. package/src/browser/context-key-service.ts +142 -142
  50. package/src/browser/context-menu-renderer.ts +124 -124
  51. package/src/browser/core-preferences.ts +343 -343
  52. package/src/browser/credentials-service.ts +106 -106
  53. package/src/browser/decoration-style.ts +65 -65
  54. package/src/browser/decorations-service.ts +209 -209
  55. package/src/browser/dialogs/react-dialog.tsx +56 -56
  56. package/src/browser/dialogs.ts +534 -534
  57. package/src/browser/diff-uris.ts +117 -117
  58. package/src/browser/encoding-registry.ts +97 -97
  59. package/src/browser/endpoint.spec.ts +148 -148
  60. package/src/browser/endpoint.ts +136 -136
  61. package/src/browser/external-uri-service.ts +79 -79
  62. package/src/browser/file-icons-js.d.ts +20 -20
  63. package/src/browser/frontend-application-bindings.ts +62 -62
  64. package/src/browser/frontend-application-config-provider.spec.ts +45 -45
  65. package/src/browser/frontend-application-config-provider.ts +50 -50
  66. package/src/browser/frontend-application-contribution.ts +110 -110
  67. package/src/browser/frontend-application-module.ts +474 -474
  68. package/src/browser/frontend-application-state.ts +74 -74
  69. package/src/browser/frontend-application.ts +326 -326
  70. package/src/browser/hover-service.ts +218 -218
  71. package/src/browser/http-open-handler.ts +49 -49
  72. package/src/browser/i18n/i18n-frontend-module.ts +27 -27
  73. package/src/browser/i18n/language-quick-pick-service.ts +130 -130
  74. package/src/browser/icon-registry.ts +87 -87
  75. package/src/browser/icon-theme-contribution.ts +64 -64
  76. package/src/browser/icon-theme-service.ts +217 -217
  77. package/src/browser/icons/CollapseAll.svg +7 -7
  78. package/src/browser/icons/CollapseAll_inverse.svg +7 -7
  79. package/src/browser/icons/Refresh.svg +7 -7
  80. package/src/browser/icons/Refresh_inverse.svg +7 -7
  81. package/src/browser/icons/add-inverse.svg +4 -4
  82. package/src/browser/icons/add.svg +4 -4
  83. package/src/browser/icons/arrow-down-bright.svg +6 -6
  84. package/src/browser/icons/arrow-down-dark.svg +6 -6
  85. package/src/browser/icons/arrow-up-bright.svg +6 -6
  86. package/src/browser/icons/arrow-up-dark.svg +6 -6
  87. package/src/browser/icons/case-sensitive-dark.svg +16 -16
  88. package/src/browser/icons/case-sensitive.svg +16 -16
  89. package/src/browser/icons/chevron-right-dark.svg +5 -5
  90. package/src/browser/icons/chevron-right-light.svg +6 -6
  91. package/src/browser/icons/circle-bright.svg +7 -7
  92. package/src/browser/icons/circle-dark.svg +7 -7
  93. package/src/browser/icons/clear-search-results-dark.svg +7 -7
  94. package/src/browser/icons/clear-search-results.svg +7 -7
  95. package/src/browser/icons/close-all-bright.svg +7 -7
  96. package/src/browser/icons/close-all-dark.svg +7 -7
  97. package/src/browser/icons/close-bright.svg +7 -7
  98. package/src/browser/icons/close-dark.svg +7 -7
  99. package/src/browser/icons/collapse.svg +4 -4
  100. package/src/browser/icons/edit-json-dark.svg +6 -6
  101. package/src/browser/icons/edit-json.svg +6 -6
  102. package/src/browser/icons/expand.svg +4 -4
  103. package/src/browser/icons/loading-dark.svg +6 -6
  104. package/src/browser/icons/loading-light.svg +6 -6
  105. package/src/browser/icons/open-change-bright.svg +3 -3
  106. package/src/browser/icons/open-change-dark.svg +4 -4
  107. package/src/browser/icons/open-file-bright.svg +4 -4
  108. package/src/browser/icons/open-file-dark.svg +4 -4
  109. package/src/browser/icons/preview-bright.svg +3 -3
  110. package/src/browser/icons/preview-dark.svg +3 -3
  111. package/src/browser/icons/regex-dark.svg +10 -10
  112. package/src/browser/icons/regex.svg +10 -10
  113. package/src/browser/icons/remove-all-inverse.svg +4 -4
  114. package/src/browser/icons/remove-all.svg +4 -4
  115. package/src/browser/icons/replace-all-inverse.svg +13 -13
  116. package/src/browser/icons/replace-all.svg +13 -13
  117. package/src/browser/icons/replace-inverse.svg +15 -15
  118. package/src/browser/icons/replace.svg +15 -15
  119. package/src/browser/icons/whole-word-dark.svg +19 -19
  120. package/src/browser/icons/whole-word.svg +19 -19
  121. package/src/browser/index.ts +50 -50
  122. package/src/browser/json-schema-store.ts +118 -118
  123. package/src/browser/keybinding.spec.ts +554 -554
  124. package/src/browser/keybinding.ts +759 -759
  125. package/src/browser/keyboard/browser-keyboard-frontend-contribution.ts +108 -108
  126. package/src/browser/keyboard/browser-keyboard-layout-provider.spec.ts +171 -171
  127. package/src/browser/keyboard/browser-keyboard-layout-provider.ts +469 -469
  128. package/src/browser/keyboard/browser-keyboard-module.ts +30 -30
  129. package/src/browser/keyboard/index.ts +20 -20
  130. package/src/browser/keyboard/keyboard-layout-service.spec.ts +121 -121
  131. package/src/browser/keyboard/keyboard-layout-service.ts +455 -455
  132. package/src/browser/keyboard/keys.spec.ts +258 -258
  133. package/src/browser/keyboard/keys.ts +20 -20
  134. package/src/browser/keys.ts +21 -21
  135. package/src/browser/label-parser.spec.ts +165 -165
  136. package/src/browser/label-parser.ts +108 -108
  137. package/src/browser/label-provider.spec.ts +62 -62
  138. package/src/browser/label-provider.ts +385 -385
  139. package/src/browser/language-icon-provider.ts +55 -55
  140. package/src/browser/language-service.ts +77 -77
  141. package/src/browser/logger-frontend-module.ts +65 -65
  142. package/src/browser/markdown-rendering/markdown-renderer.ts +98 -98
  143. package/src/browser/menu/browser-context-menu-renderer.ts +48 -48
  144. package/src/browser/menu/browser-menu-module.ts +28 -28
  145. package/src/browser/menu/browser-menu-plugin.ts +491 -491
  146. package/src/browser/menu/context-menu-context.ts +41 -41
  147. package/src/browser/messaging/connection-source.ts +26 -26
  148. package/src/browser/messaging/frontend-id-provider.ts +37 -37
  149. package/src/browser/messaging/index.ts +18 -18
  150. package/src/browser/messaging/messaging-frontend-module.ts +41 -41
  151. package/src/browser/messaging/service-connection-provider.ts +140 -140
  152. package/src/browser/messaging/ws-connection-provider.ts +49 -49
  153. package/src/browser/messaging/ws-connection-source.ts +230 -230
  154. package/src/browser/mime-service.ts +30 -30
  155. package/src/browser/navigatable-types.ts +81 -81
  156. package/src/browser/navigatable.ts +39 -39
  157. package/src/browser/open-with-service.ts +140 -140
  158. package/src/browser/opener-service.spec.ts +49 -49
  159. package/src/browser/opener-service.ts +169 -169
  160. package/src/browser/performance/frontend-stopwatch.ts +65 -65
  161. package/src/browser/performance/index.ts +18 -18
  162. package/src/browser/performance/measurement-frontend-bindings.ts +31 -31
  163. package/src/browser/preferences/index.ts +23 -23
  164. package/src/browser/preferences/injectable-preference-proxy.ts +283 -283
  165. package/src/browser/preferences/preference-configurations.ts +82 -82
  166. package/src/browser/preferences/preference-contribution.ts +436 -436
  167. package/src/browser/preferences/preference-language-override-service.ts +111 -111
  168. package/src/browser/preferences/preference-provider.spec.ts +36 -36
  169. package/src/browser/preferences/preference-provider.ts +277 -277
  170. package/src/browser/preferences/preference-proxy.spec.ts +367 -367
  171. package/src/browser/preferences/preference-proxy.ts +367 -367
  172. package/src/browser/preferences/preference-schema-provider.spec.ts +130 -130
  173. package/src/browser/preferences/preference-scope.ts +18 -18
  174. package/src/browser/preferences/preference-service.spec.ts +613 -613
  175. package/src/browser/preferences/preference-service.ts +594 -594
  176. package/src/browser/preferences/preference-validation-service.spec.ts +334 -334
  177. package/src/browser/preferences/preference-validation-service.ts +358 -358
  178. package/src/browser/preferences/test/index.ts +19 -19
  179. package/src/browser/preferences/test/mock-preference-provider.ts +50 -50
  180. package/src/browser/preferences/test/mock-preference-proxy.ts +48 -48
  181. package/src/browser/preferences/test/mock-preference-service.ts +63 -63
  182. package/src/browser/preload/i18n-preload-contribution.ts +50 -50
  183. package/src/browser/preload/os-preload-contribution.ts +37 -37
  184. package/src/browser/preload/preload-module.ts +45 -45
  185. package/src/browser/preload/preloader.ts +37 -37
  186. package/src/browser/preload/theme-preload-contribution.ts +31 -31
  187. package/src/browser/progress-bar-factory.ts +29 -29
  188. package/src/browser/progress-bar.ts +76 -76
  189. package/src/browser/progress-client.ts +53 -53
  190. package/src/browser/progress-location-service.spec.ts +50 -50
  191. package/src/browser/progress-location-service.ts +96 -96
  192. package/src/browser/progress-status-bar-item.ts +83 -83
  193. package/src/browser/quick-input/index.ts +23 -23
  194. package/src/browser/quick-input/quick-access.ts +75 -75
  195. package/src/browser/quick-input/quick-command-frontend-contribution.ts +89 -89
  196. package/src/browser/quick-input/quick-command-service.ts +246 -246
  197. package/src/browser/quick-input/quick-help-service.ts +87 -87
  198. package/src/browser/quick-input/quick-input-frontend-contribution.ts +33 -33
  199. package/src/browser/quick-input/quick-input-service.spec.ts +176 -176
  200. package/src/browser/quick-input/quick-input-service.ts +17 -17
  201. package/src/browser/quick-input/quick-pick-service-impl.ts +69 -69
  202. package/src/browser/quick-input/quick-view-service.ts +83 -83
  203. package/src/browser/request/browser-request-module.ts +23 -23
  204. package/src/browser/request/browser-request-service.ts +172 -172
  205. package/src/browser/resource-context-key.ts +77 -77
  206. package/src/browser/saveable-service.ts +332 -332
  207. package/src/browser/saveable.ts +395 -395
  208. package/src/browser/secondary-window-handler.ts +211 -211
  209. package/src/browser/shell/additional-views-menu-widget.tsx +71 -71
  210. package/src/browser/shell/application-shell-mouse-tracker.ts +103 -103
  211. package/src/browser/shell/application-shell.ts +2271 -2271
  212. package/src/browser/shell/current-widget-command-adapter.ts +57 -57
  213. package/src/browser/shell/index.ts +23 -23
  214. package/src/browser/shell/shell-layout-restorer.ts +399 -399
  215. package/src/browser/shell/side-panel-handler.ts +794 -794
  216. package/src/browser/shell/side-panel-toolbar.ts +111 -111
  217. package/src/browser/shell/sidebar-bottom-menu-widget.tsx +39 -39
  218. package/src/browser/shell/sidebar-menu-widget.tsx +183 -183
  219. package/src/browser/shell/sidebar-top-menu-widget.tsx +26 -26
  220. package/src/browser/shell/split-panels.ts +191 -191
  221. package/src/browser/shell/tab-bar-decorator.ts +106 -106
  222. package/src/browser/shell/tab-bar-toolbar/index.ts +19 -19
  223. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar-menu-adapters.ts +31 -31
  224. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar-registry.ts +242 -242
  225. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar-types.ts +149 -149
  226. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar.spec.ts +62 -62
  227. package/src/browser/shell/tab-bar-toolbar/tab-bar-toolbar.tsx +443 -443
  228. package/src/browser/shell/tab-bars.spec.ts +63 -63
  229. package/src/browser/shell/tab-bars.ts +1468 -1468
  230. package/src/browser/shell/theia-dock-panel.ts +265 -265
  231. package/src/browser/shell/view-column-service.ts +125 -125
  232. package/src/browser/shell/view-contribution.ts +178 -178
  233. package/src/browser/source-tree/index.ts +19 -19
  234. package/src/browser/source-tree/source-tree-widget.tsx +107 -107
  235. package/src/browser/source-tree/source-tree.ts +146 -146
  236. package/src/browser/source-tree/tree-source.ts +73 -73
  237. package/src/browser/status-bar/index.ts +29 -29
  238. package/src/browser/status-bar/status-bar-types.ts +97 -97
  239. package/src/browser/status-bar/status-bar-view-model.ts +209 -209
  240. package/src/browser/status-bar/status-bar.tsx +189 -189
  241. package/src/browser/storage-service.spec.ts +76 -76
  242. package/src/browser/storage-service.ts +129 -129
  243. package/src/browser/style/about.css +36 -36
  244. package/src/browser/style/alert-messages.css +62 -62
  245. package/src/browser/style/ansi.css +88 -88
  246. package/src/browser/style/breadcrumbs.css +130 -130
  247. package/src/browser/style/dialog.css +126 -126
  248. package/src/browser/style/dockpanel.css +76 -76
  249. package/src/browser/style/hover-service.css +101 -101
  250. package/src/browser/style/icons.css +61 -61
  251. package/src/browser/style/index.css +353 -353
  252. package/src/browser/style/materialcolors.css +278 -278
  253. package/src/browser/style/menus.css +230 -230
  254. package/src/browser/style/notification.css +39 -39
  255. package/src/browser/style/os.css +87 -87
  256. package/src/browser/style/progress-bar.css +43 -43
  257. package/src/browser/style/quick-title-bar.css +45 -45
  258. package/src/browser/style/scrollbars.css +172 -172
  259. package/src/browser/style/search-box.css +123 -123
  260. package/src/browser/style/select-component.css +107 -107
  261. package/src/browser/style/sidepanel.css +367 -367
  262. package/src/browser/style/split-widget.css +38 -38
  263. package/src/browser/style/status-bar.css +127 -127
  264. package/src/browser/style/tabs.css +647 -647
  265. package/src/browser/style/tooltip.css +28 -28
  266. package/src/browser/style/tree-decorators.css +81 -81
  267. package/src/browser/style/tree.css +232 -232
  268. package/src/browser/style/view-container.css +187 -187
  269. package/src/browser/style/widget.css +19 -19
  270. package/src/browser/styling-service.ts +96 -96
  271. package/src/browser/supported-encodings.ts +262 -262
  272. package/src/browser/test/jsdom.ts +69 -69
  273. package/src/browser/test/mock-connection-status-service.ts +33 -33
  274. package/src/browser/test/mock-env-variables-server.ts +47 -47
  275. package/src/browser/test/mock-opener-service.ts +34 -34
  276. package/src/browser/test/mock-storage-service.ts +49 -49
  277. package/src/browser/theming.ts +206 -206
  278. package/src/browser/tooltip-service.tsx +96 -96
  279. package/src/browser/tree/fuzzy-search.spec.ts +99 -99
  280. package/src/browser/tree/fuzzy-search.ts +136 -136
  281. package/src/browser/tree/index.ts +29 -29
  282. package/src/browser/tree/search-box-debounce.ts +96 -96
  283. package/src/browser/tree/search-box.ts +355 -355
  284. package/src/browser/tree/test/mock-selectable-tree-model.ts +109 -109
  285. package/src/browser/tree/test/mock-tree-model.ts +136 -136
  286. package/src/browser/tree/test/tree-test-container.ts +50 -50
  287. package/src/browser/tree/tree-compression/compressed-tree-expansion-service.ts +46 -46
  288. package/src/browser/tree/tree-compression/compressed-tree-model.ts +88 -88
  289. package/src/browser/tree/tree-compression/compressed-tree-widget.tsx +203 -203
  290. package/src/browser/tree/tree-compression/index.ts +20 -20
  291. package/src/browser/tree/tree-compression/tree-compression-service.ts +125 -125
  292. package/src/browser/tree/tree-compression/tree-compression.css +28 -28
  293. package/src/browser/tree/tree-consistency.spec.ts +105 -105
  294. package/src/browser/tree/tree-container.spec.ts +45 -45
  295. package/src/browser/tree/tree-container.ts +155 -155
  296. package/src/browser/tree/tree-decorator.spec.ts +162 -162
  297. package/src/browser/tree/tree-decorator.ts +238 -238
  298. package/src/browser/tree/tree-expansion.spec.ts +173 -173
  299. package/src/browser/tree/tree-expansion.ts +165 -165
  300. package/src/browser/tree/tree-focus-service.ts +55 -55
  301. package/src/browser/tree/tree-iterator.spec.ts +170 -170
  302. package/src/browser/tree/tree-iterator.ts +256 -256
  303. package/src/browser/tree/tree-label-provider.ts +40 -40
  304. package/src/browser/tree/tree-model.ts +562 -562
  305. package/src/browser/tree/tree-navigation.ts +58 -58
  306. package/src/browser/tree/tree-preference.ts +50 -50
  307. package/src/browser/tree/tree-search.ts +128 -128
  308. package/src/browser/tree/tree-selectable.spec.ts +152 -152
  309. package/src/browser/tree/tree-selection-impl.ts +176 -176
  310. package/src/browser/tree/tree-selection-state.spec.ts +462 -462
  311. package/src/browser/tree/tree-selection-state.ts +245 -245
  312. package/src/browser/tree/tree-selection.ts +159 -159
  313. package/src/browser/tree/tree-view-welcome-widget.tsx +263 -263
  314. package/src/browser/tree/tree-widget-selection.ts +45 -45
  315. package/src/browser/tree/tree-widget.tsx +1591 -1591
  316. package/src/browser/tree/tree.spec.ts +241 -241
  317. package/src/browser/tree/tree.ts +425 -425
  318. package/src/browser/undo-redo-handler.ts +85 -85
  319. package/src/browser/user-working-directory-provider.ts +77 -77
  320. package/src/browser/view-container.ts +1640 -1640
  321. package/src/browser/widget-decoration.ts +358 -358
  322. package/src/browser/widget-manager.spec.ts +102 -102
  323. package/src/browser/widget-manager.ts +318 -318
  324. package/src/browser/widget-open-handler.ts +168 -168
  325. package/src/browser/widgets/alert-message.tsx +56 -56
  326. package/src/browser/widgets/enhanced-preview-widget.ts +27 -27
  327. package/src/browser/widgets/extractable-widget.ts +33 -33
  328. package/src/browser/widgets/index.ts +21 -21
  329. package/src/browser/widgets/previewable-widget.ts +31 -31
  330. package/src/browser/widgets/react-renderer.tsx +53 -53
  331. package/src/browser/widgets/react-widget.tsx +51 -51
  332. package/src/browser/widgets/select-component.tsx +367 -367
  333. package/src/browser/widgets/split-widget.ts +163 -163
  334. package/src/browser/widgets/widget.ts +406 -406
  335. package/src/browser/window/browser-window-module.ts +32 -32
  336. package/src/browser/window/default-secondary-window-service.ts +189 -189
  337. package/src/browser/window/default-window-service.spec.ts +78 -78
  338. package/src/browser/window/default-window-service.ts +171 -171
  339. package/src/browser/window/secondary-window-service.ts +39 -39
  340. package/src/browser/window/test/mock-window-service.ts +29 -29
  341. package/src/browser/window/window-service.ts +78 -78
  342. package/src/browser/window/window-title-service.ts +107 -107
  343. package/src/browser/window/window-title-updater.ts +95 -95
  344. package/src/browser/window-contribution.ts +64 -64
  345. package/src/browser-only/frontend-only-application-module.ts +116 -116
  346. package/src/browser-only/i18n/i18n-frontend-only-module.ts +37 -37
  347. package/src/browser-only/logger-frontend-only-module.ts +63 -63
  348. package/src/browser-only/messaging/frontend-only-service-connection-provider.ts +39 -39
  349. package/src/browser-only/messaging/messaging-frontend-only-module.ts +42 -42
  350. package/src/browser-only/preload/frontend-only-preload-module.ts +49 -49
  351. package/src/common/accessibility.ts +33 -33
  352. package/src/common/application-error.spec.ts +27 -27
  353. package/src/common/application-error.ts +76 -76
  354. package/src/common/application-protocol.ts +42 -42
  355. package/src/common/array-utils.ts +129 -129
  356. package/src/common/buffer.ts +228 -228
  357. package/src/common/cancellation.ts +163 -163
  358. package/src/common/char-code.ts +438 -438
  359. package/src/common/collections.ts +125 -125
  360. package/src/common/color.ts +103 -103
  361. package/src/common/command.spec.ts +208 -208
  362. package/src/common/command.ts +489 -489
  363. package/src/common/contribution-filter/contribution-filter-registry.ts +79 -79
  364. package/src/common/contribution-filter/contribution-filter.ts +64 -64
  365. package/src/common/contribution-filter/filter.ts +23 -23
  366. package/src/common/contribution-filter/index.ts +19 -19
  367. package/src/common/contribution-provider.ts +96 -96
  368. package/src/common/disposable.spec.ts +94 -94
  369. package/src/common/disposable.ts +188 -188
  370. package/src/common/encoding-service.ts +380 -380
  371. package/src/common/encodings.ts +24 -24
  372. package/src/common/env-variables/env-variables-protocol.ts +38 -38
  373. package/src/common/env-variables/index.ts +17 -17
  374. package/src/common/event.spec.ts +32 -32
  375. package/src/common/event.ts +493 -493
  376. package/src/common/file-uri.ts +61 -61
  377. package/src/common/frontend-application-state.ts +38 -38
  378. package/src/common/glob.ts +741 -741
  379. package/src/common/hash.ts +85 -85
  380. package/src/common/i18n/localization-server.ts +25 -25
  381. package/src/common/i18n/localization.ts +80 -80
  382. package/src/common/i18n/nls.metadata.json +34112 -34112
  383. package/src/common/index.ts +51 -51
  384. package/src/common/json-schema.ts +108 -108
  385. package/src/common/key-store.ts +26 -26
  386. package/src/common/keybinding.ts +152 -152
  387. package/src/common/keyboard/keyboard-layout-provider.ts +51 -51
  388. package/src/common/keys.ts +694 -694
  389. package/src/common/label-protocol.ts +35 -35
  390. package/src/common/logger-protocol.ts +119 -119
  391. package/src/common/logger-watcher.ts +48 -48
  392. package/src/common/logger.spec.ts +46 -46
  393. package/src/common/logger.ts +389 -389
  394. package/src/common/lsp-types.ts +34 -34
  395. package/src/common/markdown-rendering/icon-utilities.ts +30 -30
  396. package/src/common/markdown-rendering/index.ts +18 -18
  397. package/src/common/markdown-rendering/markdown-string.ts +152 -152
  398. package/src/common/menu/action-menu-node.ts +65 -65
  399. package/src/common/menu/composite-menu-node.spec.ts +67 -67
  400. package/src/common/menu/composite-menu-node.ts +114 -114
  401. package/src/common/menu/index.ts +21 -21
  402. package/src/common/menu/menu-adapter.ts +103 -103
  403. package/src/common/menu/menu-model-registry.ts +374 -374
  404. package/src/common/menu/menu-types.ts +220 -220
  405. package/src/common/menu/menu.spec.ts +101 -101
  406. package/src/common/message-rpc/channel.spec.ts +88 -88
  407. package/src/common/message-rpc/channel.ts +300 -300
  408. package/src/common/message-rpc/index.ts +22 -22
  409. package/src/common/message-rpc/message-buffer.ts +105 -105
  410. package/src/common/message-rpc/msg-pack-extension-manager.ts +70 -70
  411. package/src/common/message-rpc/rpc-message-encoder.spec.ts +65 -65
  412. package/src/common/message-rpc/rpc-message-encoder.ts +190 -190
  413. package/src/common/message-rpc/rpc-protocol.ts +255 -255
  414. package/src/common/message-rpc/uint8-array-message-buffer.spec.ts +41 -41
  415. package/src/common/message-rpc/uint8-array-message-buffer.ts +213 -213
  416. package/src/common/message-service-protocol.ts +148 -148
  417. package/src/common/message-service.ts +226 -226
  418. package/src/common/messaging/connection-error-handler.ts +73 -73
  419. package/src/common/messaging/connection-management.ts +43 -43
  420. package/src/common/messaging/handler.ts +26 -26
  421. package/src/common/messaging/index.ts +19 -19
  422. package/src/common/messaging/proxy-factory.spec.ts +108 -108
  423. package/src/common/messaging/proxy-factory.ts +336 -336
  424. package/src/common/messaging/socket-write-buffer.ts +52 -52
  425. package/src/common/messaging/web-socket-channel.ts +76 -76
  426. package/src/common/nls.ts +151 -151
  427. package/src/common/numbers.ts +21 -21
  428. package/src/common/objects.spec.ts +112 -112
  429. package/src/common/objects.ts +123 -123
  430. package/src/common/os.ts +82 -82
  431. package/src/common/path.spec.ts +415 -415
  432. package/src/common/path.ts +334 -334
  433. package/src/common/paths.ts +250 -250
  434. package/src/common/performance/index.ts +19 -19
  435. package/src/common/performance/measurement-protocol.ts +104 -104
  436. package/src/common/performance/measurement.ts +130 -130
  437. package/src/common/performance/stopwatch.ts +183 -183
  438. package/src/common/preferences/preference-schema.ts +101 -101
  439. package/src/common/preferences/preference-scope.spec.ts +48 -48
  440. package/src/common/preferences/preference-scope.ts +68 -68
  441. package/src/common/prioritizeable.ts +58 -58
  442. package/src/common/progress-service-protocol.ts +35 -35
  443. package/src/common/progress-service.ts +82 -82
  444. package/src/common/promise-util.spec.ts +102 -102
  445. package/src/common/promise-util.ts +143 -143
  446. package/src/common/quick-pick-service.ts +353 -353
  447. package/src/common/reference.spec.ts +145 -145
  448. package/src/common/reference.ts +230 -230
  449. package/src/common/resource.ts +430 -430
  450. package/src/common/selection-command-handler.ts +101 -101
  451. package/src/common/selection-service.spec.ts +43 -43
  452. package/src/common/selection-service.ts +49 -49
  453. package/src/common/selection.ts +50 -50
  454. package/src/common/severity.ts +111 -111
  455. package/src/common/stream.ts +718 -718
  456. package/src/common/strings.ts +231 -231
  457. package/src/common/telemetry.ts +45 -45
  458. package/src/common/ternary-search-tree.ts +417 -417
  459. package/src/common/test/expect.ts +34 -34
  460. package/src/common/test/mock-logger.ts +118 -118
  461. package/src/common/test/mock-menu.ts +35 -35
  462. package/src/common/test/mock-resource-provider.ts +33 -33
  463. package/src/common/theme.ts +68 -68
  464. package/src/common/types.spec.ts +86 -86
  465. package/src/common/types.ts +140 -140
  466. package/src/common/uri-command-handler.spec.ts +90 -90
  467. package/src/common/uri-command-handler.ts +148 -148
  468. package/src/common/uri.spec.ts +278 -278
  469. package/src/common/uri.ts +279 -279
  470. package/src/common/uuid.ts +45 -45
  471. package/src/common/version.ts +17 -17
  472. package/src/common/view-column.ts +33 -33
  473. package/src/common/window.ts +34 -34
  474. package/src/electron-browser/electron-clipboard-service.ts +32 -32
  475. package/src/electron-browser/electron-uri-handler.ts +42 -42
  476. package/src/electron-browser/keyboard/electron-keyboard-layout-change-notifier.ts +39 -39
  477. package/src/electron-browser/keyboard/electron-keyboard-module.ts +28 -28
  478. package/src/electron-browser/menu/electron-context-menu-renderer.ts +122 -122
  479. package/src/electron-browser/menu/electron-main-menu-factory.ts +339 -339
  480. package/src/electron-browser/menu/electron-menu-contribution.ts +506 -506
  481. package/src/electron-browser/menu/electron-menu-module.ts +40 -40
  482. package/src/electron-browser/menu/electron-menu-style.css +110 -110
  483. package/src/electron-browser/messaging/electron-frontend-id-provider.ts +25 -25
  484. package/src/electron-browser/messaging/electron-ipc-connection-source.ts +65 -65
  485. package/src/electron-browser/messaging/electron-local-ws-connection-source.ts +45 -45
  486. package/src/electron-browser/messaging/electron-messaging-frontend-module.ts +78 -78
  487. package/src/electron-browser/messaging/electron-ws-connection-source.ts +38 -38
  488. package/src/electron-browser/preload.ts +264 -264
  489. package/src/electron-browser/request/electron-browser-request-module.ts +26 -26
  490. package/src/electron-browser/token/electron-token-frontend-module.ts +22 -22
  491. package/src/electron-browser/window/electron-frontend-application-state.ts +26 -26
  492. package/src/electron-browser/window/electron-secondary-window-service.ts +35 -35
  493. package/src/electron-browser/window/electron-window-module.ts +48 -48
  494. package/src/electron-browser/window/electron-window-preferences.ts +76 -76
  495. package/src/electron-browser/window/electron-window-service.ts +109 -109
  496. package/src/electron-browser/window/external-app-open-handler.ts +42 -42
  497. package/src/electron-common/electron-api.ts +157 -157
  498. package/src/electron-common/electron-main-window-service.ts +24 -24
  499. package/src/electron-common/electron-token.ts +27 -27
  500. package/src/electron-main/electron-api-main.ts +373 -373
  501. package/src/electron-main/electron-main-application-module.ts +65 -65
  502. package/src/electron-main/electron-main-application.ts +860 -860
  503. package/src/electron-main/electron-main-constants.ts +23 -23
  504. package/src/electron-main/electron-main-window-service-impl.ts +44 -44
  505. package/src/electron-main/electron-security-token-service.ts +36 -36
  506. package/src/electron-main/event-utils.ts +36 -36
  507. package/src/electron-main/messaging/electron-connection-handler.ts +21 -21
  508. package/src/electron-main/messaging/electron-messaging-contribution.ts +143 -143
  509. package/src/electron-main/messaging/electron-messaging-service.ts +35 -35
  510. package/src/electron-main/theia-electron-window.ts +219 -219
  511. package/src/electron-node/cli/electron-backend-cli-module.ts +24 -24
  512. package/src/electron-node/cli/electron-cli-contribution.ts +35 -35
  513. package/src/electron-node/hosting/electron-backend-hosting-module.ts +24 -24
  514. package/src/electron-node/hosting/electron-ws-origin-validator.ts +37 -37
  515. package/src/electron-node/keyboard/electron-backend-keyboard-module.ts +30 -30
  516. package/src/electron-node/keyboard/electron-keyboard-layout-provider.ts +35 -35
  517. package/src/electron-node/request/electron-backend-request-module.ts +23 -23
  518. package/src/electron-node/request/electron-backend-request-service.ts +78 -78
  519. package/src/electron-node/token/electron-token-backend-contribution.ts +48 -48
  520. package/src/electron-node/token/electron-token-backend-module.ts +28 -28
  521. package/src/electron-node/token/electron-token-validator.ts +93 -93
  522. package/src/node/application-server.ts +59 -59
  523. package/src/node/backend-application-config-provider.spec.ts +29 -29
  524. package/src/node/backend-application-config-provider.ts +48 -48
  525. package/src/node/backend-application-module.ts +139 -139
  526. package/src/node/backend-application.ts +374 -374
  527. package/src/node/cli.spec.ts +94 -94
  528. package/src/node/cli.ts +63 -63
  529. package/src/node/console-logger-server.spec.ts +59 -59
  530. package/src/node/console-logger-server.ts +76 -76
  531. package/src/node/debug.ts +30 -30
  532. package/src/node/dynamic-require.ts +56 -56
  533. package/src/node/env-variables/env-variables-server.ts +123 -123
  534. package/src/node/env-variables/index.ts +17 -17
  535. package/src/node/environment-utils.spec.ts +92 -92
  536. package/src/node/environment-utils.ts +66 -66
  537. package/src/node/file-uri.spec.ts +76 -76
  538. package/src/node/filesystem-locking.ts +77 -77
  539. package/src/node/hosting/backend-application-hosts.ts +60 -60
  540. package/src/node/hosting/backend-hosting-module.ts +26 -26
  541. package/src/node/hosting/ws-origin-validator.ts +36 -36
  542. package/src/node/i18n/i18n-backend-module.ts +42 -42
  543. package/src/node/i18n/localization-contribution.ts +112 -112
  544. package/src/node/i18n/localization-provider.ts +125 -125
  545. package/src/node/i18n/localization-server.ts +52 -52
  546. package/src/node/i18n/theia-localization-contribution.ts +40 -40
  547. package/src/node/index.ts +22 -22
  548. package/src/node/key-store-server.ts +162 -162
  549. package/src/node/logger-backend-module.ts +88 -88
  550. package/src/node/logger-cli-contribution.spec.ts +245 -245
  551. package/src/node/logger-cli-contribution.ts +168 -168
  552. package/src/node/main.ts +33 -33
  553. package/src/node/messaging/binary-message-pipe.ts +168 -168
  554. package/src/node/messaging/connection-container-module.ts +96 -96
  555. package/src/node/messaging/default-messaging-service.ts +129 -129
  556. package/src/node/messaging/frontend-connection-service.ts +24 -24
  557. package/src/node/messaging/index.ts +19 -19
  558. package/src/node/messaging/ipc-bootstrap.ts +27 -27
  559. package/src/node/messaging/ipc-channel.ts +77 -77
  560. package/src/node/messaging/ipc-connection-provider.ts +107 -107
  561. package/src/node/messaging/ipc-protocol.ts +76 -76
  562. package/src/node/messaging/messaging-backend-module.ts +52 -52
  563. package/src/node/messaging/messaging-listeners.ts +52 -52
  564. package/src/node/messaging/messaging-service.ts +46 -46
  565. package/src/node/messaging/test/test-web-socket-channel.ts +61 -61
  566. package/src/node/messaging/websocket-endpoint.ts +79 -79
  567. package/src/node/messaging/websocket-frontend-connection-service.ts +186 -186
  568. package/src/node/os-backend-provider.ts +25 -25
  569. package/src/node/performance/index.ts +18 -18
  570. package/src/node/performance/measurement-backend-bindings.ts +35 -35
  571. package/src/node/performance/node-stopwatch.ts +40 -40
  572. package/src/node/process-utils.spec.ts +48 -48
  573. package/src/node/process-utils.ts +102 -102
  574. package/src/node/remote/backend-remote-service.ts +25 -25
  575. package/src/node/remote/remote-cli-contribution.ts +34 -34
  576. package/src/node/remote/remote-copy-contribution.ts +45 -45
  577. package/src/node/request/backend-request-facade.ts +39 -39
  578. package/src/node/request/backend-request-module.ts +25 -25
  579. package/src/node/request/proxy-cli-contribution.ts +65 -65
  580. package/src/node/ws-request-validators.ts +56 -56
  581. package/src/typings/native-keymap.d.ts +108 -108
@@ -1,718 +1,718 @@
1
- // *****************************************************************************
2
- // Copyright (C) 2020 TypeFox and others.
3
- //
4
- // This program and the accompanying materials are made available under the
5
- // terms of the Eclipse Public License v. 2.0 which is available at
6
- // http://www.eclipse.org/legal/epl-2.0.
7
- //
8
- // This Source Code may also be made available under the following Secondary
9
- // Licenses when the conditions for such availability set forth in the Eclipse
10
- // Public License v. 2.0 are satisfied: GNU General Public License, version 2
11
- // with the GNU Classpath Exception which is available at
12
- // https://www.gnu.org/software/classpath/license.html.
13
- //
14
- // SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
15
- // *****************************************************************************
16
- /*---------------------------------------------------------------------------------------------
17
- * Copyright (c) Microsoft Corporation. All rights reserved.
18
- * Licensed under the MIT License. See License.txt in the project root for license information.
19
- *--------------------------------------------------------------------------------------------*/
20
-
21
- // based on https://github.com/microsoft/vscode/blob/04c36be045a94fee58e5f8992d3e3fd980294a84/src/vs/base/common/stream.ts
22
-
23
- /* eslint-disable max-len */
24
- /* eslint-disable no-null/no-null */
25
- /* eslint-disable @typescript-eslint/tslint/config */
26
- /* eslint-disable @typescript-eslint/no-explicit-any */
27
-
28
- import { DisposableCollection, Disposable } from './disposable';
29
-
30
- export interface ReadableStreamEvents<T> {
31
-
32
- /**
33
- * The 'data' event is emitted whenever the stream is
34
- * relinquishing ownership of a chunk of data to a consumer.
35
- */
36
- on(event: 'data', callback: (data: T) => void): void;
37
-
38
- /**
39
- * Emitted when any error occurs.
40
- */
41
- on(event: 'error', callback: (err: Error) => void): void;
42
-
43
- /**
44
- * The 'end' event is emitted when there is no more data
45
- * to be consumed from the stream. The 'end' event will
46
- * not be emitted unless the data is completely consumed.
47
- */
48
- on(event: 'end', callback: () => void): void;
49
- }
50
-
51
- /**
52
- * A interface that emulates the API shape of a node.js readable
53
- * stream for use in desktop and web environments.
54
- */
55
- export interface ReadableStream<T> extends ReadableStreamEvents<T> {
56
-
57
- /**
58
- * Stops emitting any events until resume() is called.
59
- */
60
- pause(): void;
61
-
62
- /**
63
- * Starts emitting events again after pause() was called.
64
- */
65
- resume(): void;
66
-
67
- /**
68
- * Destroys the stream and stops emitting any event.
69
- */
70
- destroy(): void;
71
-
72
- /**
73
- * Allows to remove a listener that was previously added.
74
- */
75
- removeListener(event: string, callback: Function): void;
76
- }
77
-
78
- /**
79
- * A interface that emulates the API shape of a node.js readable
80
- * for use in desktop and web environments.
81
- */
82
- export interface Readable<T> {
83
-
84
- /**
85
- * Read data from the underlying source. Will return
86
- * null to indicate that no more data can be read.
87
- */
88
- read(): T | null;
89
- }
90
- export namespace Readable {
91
- export function fromString(value: string): Readable<string> {
92
- let done = false;
93
-
94
- return {
95
- read(): string | null {
96
- if (!done) {
97
- done = true;
98
-
99
- return value;
100
- }
101
-
102
- return null;
103
- }
104
- };
105
- }
106
- export function toString(readable: Readable<string>): string {
107
- let result = '';
108
- let chunk: string | null;
109
- while ((chunk = readable.read()) != null) {
110
- result += chunk;
111
- }
112
- return result;
113
- }
114
- }
115
-
116
- /**
117
- * A interface that emulates the API shape of a node.js writeable
118
- * stream for use in desktop and web environments.
119
- */
120
- export interface WriteableStream<T> extends ReadableStream<T> {
121
-
122
- /**
123
- * Writing data to the stream will trigger the on('data')
124
- * event listener if the stream is flowing and buffer the
125
- * data otherwise until the stream is flowing.
126
- *
127
- * If a `highWaterMark` is configured and writing to the
128
- * stream reaches this mark, a promise will be returned
129
- * that should be awaited on before writing more data.
130
- * Otherwise there is a risk of buffering a large number
131
- * of data chunks without consumer.
132
- */
133
- write(data: T): void | Promise<void>;
134
-
135
- /**
136
- * Signals an error to the consumer of the stream via the
137
- * on('error') handler if the stream is flowing.
138
- */
139
- error(error: Error): void;
140
-
141
- /**
142
- * Signals the end of the stream to the consumer. If the
143
- * result is not an error, will trigger the on('data') event
144
- * listener if the stream is flowing and buffer the data
145
- * otherwise until the stream is flowing.
146
- *
147
- * In case of an error, the on('error') event will be used
148
- * if the stream is flowing.
149
- */
150
- end(result?: T | Error): void;
151
- }
152
-
153
- /**
154
- * A stream that has a buffer already read. Returns the original stream
155
- * that was read as well as the chunks that got read.
156
- *
157
- * The `ended` flag indicates if the stream has been fully consumed.
158
- */
159
- export interface ReadableBufferedStream<T> {
160
-
161
- /**
162
- * The original stream that is being read.
163
- */
164
- stream: ReadableStream<T>;
165
-
166
- /**
167
- * An array of chunks already read from this stream.
168
- */
169
- buffer: T[];
170
-
171
- /**
172
- * Signals if the stream has ended or not. If not, consumers
173
- * should continue to read from the stream until consumed.
174
- */
175
- ended: boolean;
176
- }
177
-
178
- export function isReadableStream<T>(obj: unknown): obj is ReadableStream<T> {
179
- const candidate = obj as ReadableStream<T>;
180
-
181
- return candidate && [candidate.on, candidate.pause, candidate.resume, candidate.destroy].every(fn => typeof fn === 'function');
182
- }
183
-
184
- export function isReadableBufferedStream<T>(obj: unknown): obj is ReadableBufferedStream<T> {
185
- const candidate = obj as ReadableBufferedStream<T>;
186
-
187
- return candidate && isReadableStream(candidate.stream) && Array.isArray(candidate.buffer) && typeof candidate.ended === 'boolean';
188
- }
189
-
190
- export interface Reducer<T> {
191
- (data: T[]): T;
192
- }
193
-
194
- export interface DataTransformer<Original, Transformed> {
195
- (data: Original): Transformed;
196
- }
197
-
198
- export interface ErrorTransformer {
199
- (error: Error): Error;
200
- }
201
-
202
- export interface ITransformer<Original, Transformed> {
203
- data: DataTransformer<Original, Transformed>;
204
- error?: ErrorTransformer;
205
- }
206
-
207
- export function newWriteableStream<T>(reducer: Reducer<T>, options?: WriteableStreamOptions): WriteableStream<T> {
208
- return new WriteableStreamImpl<T>(reducer);
209
- }
210
-
211
- export interface WriteableStreamOptions {
212
-
213
- /**
214
- * The number of objects to buffer before WriteableStream#write()
215
- * signals back that the buffer is full. Can be used to reduce
216
- * the memory pressure when the stream is not flowing.
217
- */
218
- highWaterMark?: number;
219
- }
220
-
221
- class WriteableStreamImpl<T> implements WriteableStream<T> {
222
-
223
- private readonly state = {
224
- flowing: false,
225
- ended: false,
226
- destroyed: false
227
- };
228
-
229
- private readonly buffer = {
230
- data: [] as T[],
231
- error: [] as Error[]
232
- };
233
-
234
- private readonly listeners = {
235
- data: [] as { (data: T): void }[],
236
- error: [] as { (error: Error): void }[],
237
- end: [] as { (): void }[]
238
- };
239
-
240
- private readonly pendingWritePromises: Function[] = [];
241
-
242
- constructor(private reducer: Reducer<T>, private options?: WriteableStreamOptions) { }
243
-
244
- pause(): void {
245
- if (this.state.destroyed) {
246
- return;
247
- }
248
-
249
- this.state.flowing = false;
250
- }
251
-
252
- resume(): void {
253
- if (this.state.destroyed) {
254
- return;
255
- }
256
-
257
- if (!this.state.flowing) {
258
- this.state.flowing = true;
259
-
260
- // emit buffered events
261
- this.flowData();
262
- this.flowErrors();
263
- this.flowEnd();
264
- }
265
- }
266
-
267
- write(data: T): void | Promise<void> {
268
- if (this.state.destroyed) {
269
- return;
270
- }
271
-
272
- // flowing: directly send the data to listeners
273
- if (this.state.flowing) {
274
- this.listeners.data.forEach(listener => listener(data));
275
- }
276
-
277
- // not yet flowing: buffer data until flowing
278
- else {
279
- this.buffer.data.push(data);
280
-
281
- // highWaterMark: if configured, signal back when buffer reached limits
282
- if (typeof this.options?.highWaterMark === 'number' && this.buffer.data.length > this.options.highWaterMark) {
283
- return new Promise(resolve => this.pendingWritePromises.push(resolve));
284
- }
285
- }
286
- }
287
-
288
- error(error: Error): void {
289
- if (this.state.destroyed) {
290
- return;
291
- }
292
-
293
- // flowing: directly send the error to listeners
294
- if (this.state.flowing) {
295
- this.listeners.error.forEach(listener => listener(error));
296
- }
297
-
298
- // not yet flowing: buffer errors until flowing
299
- else {
300
- this.buffer.error.push(error);
301
- }
302
- }
303
-
304
- end(result?: T | Error): void {
305
- if (this.state.destroyed) {
306
- return;
307
- }
308
-
309
- // end with data or error if provided
310
- if (result instanceof Error) {
311
- this.error(result);
312
- } else if (result) {
313
- this.write(result);
314
- }
315
-
316
- // flowing: send end event to listeners
317
- if (this.state.flowing) {
318
- this.listeners.end.forEach(listener => listener());
319
-
320
- this.destroy();
321
- }
322
-
323
- // not yet flowing: remember state
324
- else {
325
- this.state.ended = true;
326
- }
327
- }
328
-
329
- on(event: 'data', callback: (data: T) => void): void;
330
- on(event: 'error', callback: (err: Error) => void): void;
331
- on(event: 'end', callback: () => void): void;
332
- on(event: 'data' | 'error' | 'end', callback: (arg0?: any) => void): void {
333
- if (this.state.destroyed) {
334
- return;
335
- }
336
-
337
- switch (event) {
338
- case 'data':
339
- this.listeners.data.push(callback);
340
-
341
- // switch into flowing mode as soon as the first 'data'
342
- // listener is added and we are not yet in flowing mode
343
- this.resume();
344
-
345
- break;
346
-
347
- case 'end':
348
- this.listeners.end.push(callback);
349
-
350
- // emit 'end' event directly if we are flowing
351
- // and the end has already been reached
352
- //
353
- // finish() when it went through
354
- if (this.state.flowing && this.flowEnd()) {
355
- this.destroy();
356
- }
357
-
358
- break;
359
-
360
- case 'error':
361
- this.listeners.error.push(callback);
362
-
363
- // emit buffered 'error' events unless done already
364
- // now that we know that we have at least one listener
365
- if (this.state.flowing) {
366
- this.flowErrors();
367
- }
368
-
369
- break;
370
- }
371
- }
372
-
373
- removeListener(event: string, callback: Function): void {
374
- if (this.state.destroyed) {
375
- return;
376
- }
377
-
378
- let listeners: unknown[] | undefined = undefined;
379
-
380
- switch (event) {
381
- case 'data':
382
- listeners = this.listeners.data;
383
- break;
384
-
385
- case 'end':
386
- listeners = this.listeners.end;
387
- break;
388
-
389
- case 'error':
390
- listeners = this.listeners.error;
391
- break;
392
- }
393
-
394
- if (listeners) {
395
- const index = listeners.indexOf(callback);
396
- if (index >= 0) {
397
- listeners.splice(index, 1);
398
- }
399
- }
400
- }
401
-
402
- private flowData(): void {
403
- if (this.buffer.data.length > 0) {
404
- const fullDataBuffer = this.reducer(this.buffer.data);
405
-
406
- this.listeners.data.forEach(listener => listener(fullDataBuffer));
407
-
408
- this.buffer.data.length = 0;
409
-
410
- // When the buffer is empty, resolve all pending writers
411
- const pendingWritePromises = [...this.pendingWritePromises];
412
- this.pendingWritePromises.length = 0;
413
- pendingWritePromises.forEach(pendingWritePromise => pendingWritePromise());
414
- }
415
- }
416
-
417
- private flowErrors(): void {
418
- if (this.listeners.error.length > 0) {
419
- for (const error of this.buffer.error) {
420
- this.listeners.error.forEach(listener => listener(error));
421
- }
422
-
423
- this.buffer.error.length = 0;
424
- }
425
- }
426
-
427
- private flowEnd(): boolean {
428
- if (this.state.ended) {
429
- this.listeners.end.forEach(listener => listener());
430
-
431
- return this.listeners.end.length > 0;
432
- }
433
-
434
- return false;
435
- }
436
-
437
- destroy(): void {
438
- if (!this.state.destroyed) {
439
- this.state.destroyed = true;
440
- this.state.ended = true;
441
-
442
- this.buffer.data.length = 0;
443
- this.buffer.error.length = 0;
444
-
445
- this.listeners.data.length = 0;
446
- this.listeners.error.length = 0;
447
- this.listeners.end.length = 0;
448
-
449
- this.pendingWritePromises.length = 0;
450
- }
451
- }
452
- }
453
-
454
- /**
455
- * Helper to fully read a T readable into a T.
456
- */
457
- export function consumeReadable<T>(readable: Readable<T>, reducer: Reducer<T>): T {
458
- const chunks: T[] = [];
459
-
460
- let chunk: T | null;
461
- while ((chunk = readable.read()) !== null) {
462
- chunks.push(chunk);
463
- }
464
-
465
- return reducer(chunks);
466
- }
467
-
468
- /**
469
- * Helper to read a T readable up to a maximum of chunks. If the limit is
470
- * reached, will return a readable instead to ensure all data can still
471
- * be read.
472
- */
473
- export function consumeReadableWithLimit<T>(readable: Readable<T>, reducer: Reducer<T>, maxChunks: number): T | Readable<T> {
474
- const chunks: T[] = [];
475
-
476
- let chunk: T | null | undefined = undefined;
477
- while ((chunk = readable.read()) !== null && chunks.length < maxChunks) {
478
- chunks.push(chunk);
479
- }
480
-
481
- // If the last chunk is null, it means we reached the end of
482
- // the readable and return all the data at once
483
- if (chunk === null && chunks.length > 0) {
484
- return reducer(chunks);
485
- }
486
-
487
- // Otherwise, we still have a chunk, it means we reached the maxChunks
488
- // value and as such we return a new Readable that first returns
489
- // the existing read chunks and then continues with reading from
490
- // the underlying readable.
491
- return {
492
- read: () => {
493
-
494
- // First consume chunks from our array
495
- if (chunks.length > 0) {
496
- return chunks.shift()!;
497
- }
498
-
499
- // Then ensure to return our last read chunk
500
- if (typeof chunk !== 'undefined') {
501
- const lastReadChunk = chunk;
502
-
503
- // explicitly use undefined here to indicate that we consumed
504
- // the chunk, which could have either been null or valued.
505
- chunk = undefined;
506
-
507
- return lastReadChunk;
508
- }
509
-
510
- // Finally delegate back to the Readable
511
- return readable.read();
512
- }
513
- };
514
- }
515
-
516
- /**
517
- * Helper to read a T readable up to a maximum of chunks. If the limit is
518
- * reached, will return a readable instead to ensure all data can still
519
- * be read.
520
- */
521
- export function peekReadable<T>(readable: Readable<T>, reducer: Reducer<T>, maxChunks: number): T | Readable<T> {
522
- const chunks: T[] = [];
523
-
524
- let chunk: T | null | undefined = undefined;
525
- while ((chunk = readable.read()) !== null && chunks.length < maxChunks) {
526
- chunks.push(chunk);
527
- }
528
-
529
- // If the last chunk is null, it means we reached the end of
530
- // the readable and return all the data at once
531
- if (chunk === null && chunks.length > 0) {
532
- return reducer(chunks);
533
- }
534
-
535
- // Otherwise, we still have a chunk, it means we reached the maxChunks
536
- // value and as such we return a new Readable that first returns
537
- // the existing read chunks and then continues with reading from
538
- // the underlying readable.
539
- return {
540
- read: () => {
541
-
542
- // First consume chunks from our array
543
- if (chunks.length > 0) {
544
- return chunks.shift()!;
545
- }
546
-
547
- // Then ensure to return our last read chunk
548
- if (typeof chunk !== 'undefined') {
549
- const lastReadChunk = chunk;
550
-
551
- // explicitly use undefined here to indicate that we consumed
552
- // the chunk, which could have either been null or valued.
553
- chunk = undefined;
554
-
555
- return lastReadChunk;
556
- }
557
-
558
- // Finally delegate back to the Readable
559
- return readable.read();
560
- }
561
- };
562
- }
563
-
564
- /**
565
- * Helper to fully read a T stream into a T.
566
- */
567
- export function consumeStream<T>(stream: ReadableStream<T>, reducer: Reducer<T>): Promise<T> {
568
- return new Promise((resolve, reject) => {
569
- const chunks: T[] = [];
570
-
571
- stream.on('data', data => chunks.push(data));
572
- stream.on('error', error => reject(error));
573
- stream.on('end', () => resolve(reducer(chunks)));
574
- });
575
- }
576
-
577
- /**
578
- * Helper to peek up to `maxChunks` into a stream. The return type signals if
579
- * the stream has ended or not. If not, caller needs to add a `data` listener
580
- * to continue reading.
581
- */
582
- export function peekStream<T>(stream: ReadableStream<T>, maxChunks: number): Promise<ReadableBufferedStream<T>> {
583
- return new Promise((resolve, reject) => {
584
- const streamListeners = new DisposableCollection();
585
-
586
- // Data Listener
587
- const buffer: T[] = [];
588
- const dataListener = (chunk: T) => {
589
-
590
- // Add to buffer
591
- buffer.push(chunk);
592
-
593
- // We reached maxChunks and thus need to return
594
- if (buffer.length > maxChunks) {
595
-
596
- // Dispose any listeners and ensure to pause the
597
- // stream so that it can be consumed again by caller
598
- streamListeners.dispose();
599
- stream.pause();
600
-
601
- return resolve({ stream, buffer, ended: false });
602
- }
603
- };
604
-
605
- streamListeners.push(Disposable.create(() => stream.removeListener('data', dataListener)));
606
- stream.on('data', dataListener);
607
-
608
- // Error Listener
609
- const errorListener = (error: Error) => reject(error);
610
-
611
- streamListeners.push(Disposable.create(() => stream.removeListener('error', errorListener)));
612
- stream.on('error', errorListener);
613
-
614
- const endListener = () => resolve({ stream, buffer, ended: true });
615
-
616
- streamListeners.push(Disposable.create(() => stream.removeListener('end', endListener)));
617
- stream.on('end', endListener);
618
- });
619
- }
620
-
621
- /**
622
- * Helper to read a T stream up to a maximum of chunks. If the limit is
623
- * reached, will return a stream instead to ensure all data can still
624
- * be read.
625
- */
626
- export function consumeStreamWithLimit<T>(stream: ReadableStream<T>, reducer: Reducer<T>, maxChunks: number): Promise<T | ReadableStream<T>> {
627
- return new Promise((resolve, reject) => {
628
- const chunks: T[] = [];
629
-
630
- let wrapperStream: WriteableStream<T> | undefined = undefined;
631
-
632
- stream.on('data', data => {
633
-
634
- // If we reach maxChunks, we start to return a stream
635
- // and make sure that any data we have already read
636
- // is in it as well
637
- if (!wrapperStream && chunks.length === maxChunks) {
638
- wrapperStream = newWriteableStream(reducer);
639
- while (chunks.length) {
640
- wrapperStream.write(chunks.shift()!);
641
- }
642
-
643
- wrapperStream.write(data);
644
-
645
- return resolve(wrapperStream);
646
- }
647
-
648
- if (wrapperStream) {
649
- wrapperStream.write(data);
650
- } else {
651
- chunks.push(data);
652
- }
653
- });
654
-
655
- stream.on('error', error => {
656
- if (wrapperStream) {
657
- wrapperStream.error(error);
658
- } else {
659
- return reject(error);
660
- }
661
- });
662
-
663
- stream.on('end', () => {
664
- if (wrapperStream) {
665
- while (chunks.length) {
666
- wrapperStream.write(chunks.shift()!);
667
- }
668
-
669
- wrapperStream.end();
670
- } else {
671
- return resolve(reducer(chunks));
672
- }
673
- });
674
- });
675
- }
676
-
677
- /**
678
- * Helper to create a readable stream from an existing T.
679
- */
680
- export function toStream<T>(t: T, reducer: Reducer<T>): ReadableStream<T> {
681
- const stream = newWriteableStream<T>(reducer);
682
-
683
- stream.end(t);
684
-
685
- return stream;
686
- }
687
-
688
- /**
689
- * Helper to convert a T into a Readable<T>.
690
- */
691
- export function toReadable<T>(t: T): Readable<T> {
692
- let consumed = false;
693
-
694
- return {
695
- read: () => {
696
- if (consumed) {
697
- return null;
698
- }
699
-
700
- consumed = true;
701
-
702
- return t;
703
- }
704
- };
705
- }
706
-
707
- /**
708
- * Helper to transform a readable stream into another stream.
709
- */
710
- export function transform<Original, Transformed>(stream: ReadableStreamEvents<Original>, transformer: ITransformer<Original, Transformed>, reducer: Reducer<Transformed>): ReadableStream<Transformed> {
711
- const target = newWriteableStream<Transformed>(reducer);
712
-
713
- stream.on('data', data => target.write(transformer.data(data)));
714
- stream.on('end', () => target.end());
715
- stream.on('error', error => target.error(transformer.error ? transformer.error(error) : error));
716
-
717
- return target;
718
- }
1
+ // *****************************************************************************
2
+ // Copyright (C) 2020 TypeFox and others.
3
+ //
4
+ // This program and the accompanying materials are made available under the
5
+ // terms of the Eclipse Public License v. 2.0 which is available at
6
+ // http://www.eclipse.org/legal/epl-2.0.
7
+ //
8
+ // This Source Code may also be made available under the following Secondary
9
+ // Licenses when the conditions for such availability set forth in the Eclipse
10
+ // Public License v. 2.0 are satisfied: GNU General Public License, version 2
11
+ // with the GNU Classpath Exception which is available at
12
+ // https://www.gnu.org/software/classpath/license.html.
13
+ //
14
+ // SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
15
+ // *****************************************************************************
16
+ /*---------------------------------------------------------------------------------------------
17
+ * Copyright (c) Microsoft Corporation. All rights reserved.
18
+ * Licensed under the MIT License. See License.txt in the project root for license information.
19
+ *--------------------------------------------------------------------------------------------*/
20
+
21
+ // based on https://github.com/microsoft/vscode/blob/04c36be045a94fee58e5f8992d3e3fd980294a84/src/vs/base/common/stream.ts
22
+
23
+ /* eslint-disable max-len */
24
+ /* eslint-disable no-null/no-null */
25
+ /* eslint-disable @typescript-eslint/tslint/config */
26
+ /* eslint-disable @typescript-eslint/no-explicit-any */
27
+
28
+ import { DisposableCollection, Disposable } from './disposable';
29
+
30
+ export interface ReadableStreamEvents<T> {
31
+
32
+ /**
33
+ * The 'data' event is emitted whenever the stream is
34
+ * relinquishing ownership of a chunk of data to a consumer.
35
+ */
36
+ on(event: 'data', callback: (data: T) => void): void;
37
+
38
+ /**
39
+ * Emitted when any error occurs.
40
+ */
41
+ on(event: 'error', callback: (err: Error) => void): void;
42
+
43
+ /**
44
+ * The 'end' event is emitted when there is no more data
45
+ * to be consumed from the stream. The 'end' event will
46
+ * not be emitted unless the data is completely consumed.
47
+ */
48
+ on(event: 'end', callback: () => void): void;
49
+ }
50
+
51
+ /**
52
+ * A interface that emulates the API shape of a node.js readable
53
+ * stream for use in desktop and web environments.
54
+ */
55
+ export interface ReadableStream<T> extends ReadableStreamEvents<T> {
56
+
57
+ /**
58
+ * Stops emitting any events until resume() is called.
59
+ */
60
+ pause(): void;
61
+
62
+ /**
63
+ * Starts emitting events again after pause() was called.
64
+ */
65
+ resume(): void;
66
+
67
+ /**
68
+ * Destroys the stream and stops emitting any event.
69
+ */
70
+ destroy(): void;
71
+
72
+ /**
73
+ * Allows to remove a listener that was previously added.
74
+ */
75
+ removeListener(event: string, callback: Function): void;
76
+ }
77
+
78
+ /**
79
+ * A interface that emulates the API shape of a node.js readable
80
+ * for use in desktop and web environments.
81
+ */
82
+ export interface Readable<T> {
83
+
84
+ /**
85
+ * Read data from the underlying source. Will return
86
+ * null to indicate that no more data can be read.
87
+ */
88
+ read(): T | null;
89
+ }
90
+ export namespace Readable {
91
+ export function fromString(value: string): Readable<string> {
92
+ let done = false;
93
+
94
+ return {
95
+ read(): string | null {
96
+ if (!done) {
97
+ done = true;
98
+
99
+ return value;
100
+ }
101
+
102
+ return null;
103
+ }
104
+ };
105
+ }
106
+ export function toString(readable: Readable<string>): string {
107
+ let result = '';
108
+ let chunk: string | null;
109
+ while ((chunk = readable.read()) != null) {
110
+ result += chunk;
111
+ }
112
+ return result;
113
+ }
114
+ }
115
+
116
+ /**
117
+ * A interface that emulates the API shape of a node.js writeable
118
+ * stream for use in desktop and web environments.
119
+ */
120
+ export interface WriteableStream<T> extends ReadableStream<T> {
121
+
122
+ /**
123
+ * Writing data to the stream will trigger the on('data')
124
+ * event listener if the stream is flowing and buffer the
125
+ * data otherwise until the stream is flowing.
126
+ *
127
+ * If a `highWaterMark` is configured and writing to the
128
+ * stream reaches this mark, a promise will be returned
129
+ * that should be awaited on before writing more data.
130
+ * Otherwise there is a risk of buffering a large number
131
+ * of data chunks without consumer.
132
+ */
133
+ write(data: T): void | Promise<void>;
134
+
135
+ /**
136
+ * Signals an error to the consumer of the stream via the
137
+ * on('error') handler if the stream is flowing.
138
+ */
139
+ error(error: Error): void;
140
+
141
+ /**
142
+ * Signals the end of the stream to the consumer. If the
143
+ * result is not an error, will trigger the on('data') event
144
+ * listener if the stream is flowing and buffer the data
145
+ * otherwise until the stream is flowing.
146
+ *
147
+ * In case of an error, the on('error') event will be used
148
+ * if the stream is flowing.
149
+ */
150
+ end(result?: T | Error): void;
151
+ }
152
+
153
+ /**
154
+ * A stream that has a buffer already read. Returns the original stream
155
+ * that was read as well as the chunks that got read.
156
+ *
157
+ * The `ended` flag indicates if the stream has been fully consumed.
158
+ */
159
+ export interface ReadableBufferedStream<T> {
160
+
161
+ /**
162
+ * The original stream that is being read.
163
+ */
164
+ stream: ReadableStream<T>;
165
+
166
+ /**
167
+ * An array of chunks already read from this stream.
168
+ */
169
+ buffer: T[];
170
+
171
+ /**
172
+ * Signals if the stream has ended or not. If not, consumers
173
+ * should continue to read from the stream until consumed.
174
+ */
175
+ ended: boolean;
176
+ }
177
+
178
+ export function isReadableStream<T>(obj: unknown): obj is ReadableStream<T> {
179
+ const candidate = obj as ReadableStream<T>;
180
+
181
+ return candidate && [candidate.on, candidate.pause, candidate.resume, candidate.destroy].every(fn => typeof fn === 'function');
182
+ }
183
+
184
+ export function isReadableBufferedStream<T>(obj: unknown): obj is ReadableBufferedStream<T> {
185
+ const candidate = obj as ReadableBufferedStream<T>;
186
+
187
+ return candidate && isReadableStream(candidate.stream) && Array.isArray(candidate.buffer) && typeof candidate.ended === 'boolean';
188
+ }
189
+
190
+ export interface Reducer<T> {
191
+ (data: T[]): T;
192
+ }
193
+
194
+ export interface DataTransformer<Original, Transformed> {
195
+ (data: Original): Transformed;
196
+ }
197
+
198
+ export interface ErrorTransformer {
199
+ (error: Error): Error;
200
+ }
201
+
202
+ export interface ITransformer<Original, Transformed> {
203
+ data: DataTransformer<Original, Transformed>;
204
+ error?: ErrorTransformer;
205
+ }
206
+
207
+ export function newWriteableStream<T>(reducer: Reducer<T>, options?: WriteableStreamOptions): WriteableStream<T> {
208
+ return new WriteableStreamImpl<T>(reducer);
209
+ }
210
+
211
+ export interface WriteableStreamOptions {
212
+
213
+ /**
214
+ * The number of objects to buffer before WriteableStream#write()
215
+ * signals back that the buffer is full. Can be used to reduce
216
+ * the memory pressure when the stream is not flowing.
217
+ */
218
+ highWaterMark?: number;
219
+ }
220
+
221
+ class WriteableStreamImpl<T> implements WriteableStream<T> {
222
+
223
+ private readonly state = {
224
+ flowing: false,
225
+ ended: false,
226
+ destroyed: false
227
+ };
228
+
229
+ private readonly buffer = {
230
+ data: [] as T[],
231
+ error: [] as Error[]
232
+ };
233
+
234
+ private readonly listeners = {
235
+ data: [] as { (data: T): void }[],
236
+ error: [] as { (error: Error): void }[],
237
+ end: [] as { (): void }[]
238
+ };
239
+
240
+ private readonly pendingWritePromises: Function[] = [];
241
+
242
+ constructor(private reducer: Reducer<T>, private options?: WriteableStreamOptions) { }
243
+
244
+ pause(): void {
245
+ if (this.state.destroyed) {
246
+ return;
247
+ }
248
+
249
+ this.state.flowing = false;
250
+ }
251
+
252
+ resume(): void {
253
+ if (this.state.destroyed) {
254
+ return;
255
+ }
256
+
257
+ if (!this.state.flowing) {
258
+ this.state.flowing = true;
259
+
260
+ // emit buffered events
261
+ this.flowData();
262
+ this.flowErrors();
263
+ this.flowEnd();
264
+ }
265
+ }
266
+
267
+ write(data: T): void | Promise<void> {
268
+ if (this.state.destroyed) {
269
+ return;
270
+ }
271
+
272
+ // flowing: directly send the data to listeners
273
+ if (this.state.flowing) {
274
+ this.listeners.data.forEach(listener => listener(data));
275
+ }
276
+
277
+ // not yet flowing: buffer data until flowing
278
+ else {
279
+ this.buffer.data.push(data);
280
+
281
+ // highWaterMark: if configured, signal back when buffer reached limits
282
+ if (typeof this.options?.highWaterMark === 'number' && this.buffer.data.length > this.options.highWaterMark) {
283
+ return new Promise(resolve => this.pendingWritePromises.push(resolve));
284
+ }
285
+ }
286
+ }
287
+
288
+ error(error: Error): void {
289
+ if (this.state.destroyed) {
290
+ return;
291
+ }
292
+
293
+ // flowing: directly send the error to listeners
294
+ if (this.state.flowing) {
295
+ this.listeners.error.forEach(listener => listener(error));
296
+ }
297
+
298
+ // not yet flowing: buffer errors until flowing
299
+ else {
300
+ this.buffer.error.push(error);
301
+ }
302
+ }
303
+
304
+ end(result?: T | Error): void {
305
+ if (this.state.destroyed) {
306
+ return;
307
+ }
308
+
309
+ // end with data or error if provided
310
+ if (result instanceof Error) {
311
+ this.error(result);
312
+ } else if (result) {
313
+ this.write(result);
314
+ }
315
+
316
+ // flowing: send end event to listeners
317
+ if (this.state.flowing) {
318
+ this.listeners.end.forEach(listener => listener());
319
+
320
+ this.destroy();
321
+ }
322
+
323
+ // not yet flowing: remember state
324
+ else {
325
+ this.state.ended = true;
326
+ }
327
+ }
328
+
329
+ on(event: 'data', callback: (data: T) => void): void;
330
+ on(event: 'error', callback: (err: Error) => void): void;
331
+ on(event: 'end', callback: () => void): void;
332
+ on(event: 'data' | 'error' | 'end', callback: (arg0?: any) => void): void {
333
+ if (this.state.destroyed) {
334
+ return;
335
+ }
336
+
337
+ switch (event) {
338
+ case 'data':
339
+ this.listeners.data.push(callback);
340
+
341
+ // switch into flowing mode as soon as the first 'data'
342
+ // listener is added and we are not yet in flowing mode
343
+ this.resume();
344
+
345
+ break;
346
+
347
+ case 'end':
348
+ this.listeners.end.push(callback);
349
+
350
+ // emit 'end' event directly if we are flowing
351
+ // and the end has already been reached
352
+ //
353
+ // finish() when it went through
354
+ if (this.state.flowing && this.flowEnd()) {
355
+ this.destroy();
356
+ }
357
+
358
+ break;
359
+
360
+ case 'error':
361
+ this.listeners.error.push(callback);
362
+
363
+ // emit buffered 'error' events unless done already
364
+ // now that we know that we have at least one listener
365
+ if (this.state.flowing) {
366
+ this.flowErrors();
367
+ }
368
+
369
+ break;
370
+ }
371
+ }
372
+
373
+ removeListener(event: string, callback: Function): void {
374
+ if (this.state.destroyed) {
375
+ return;
376
+ }
377
+
378
+ let listeners: unknown[] | undefined = undefined;
379
+
380
+ switch (event) {
381
+ case 'data':
382
+ listeners = this.listeners.data;
383
+ break;
384
+
385
+ case 'end':
386
+ listeners = this.listeners.end;
387
+ break;
388
+
389
+ case 'error':
390
+ listeners = this.listeners.error;
391
+ break;
392
+ }
393
+
394
+ if (listeners) {
395
+ const index = listeners.indexOf(callback);
396
+ if (index >= 0) {
397
+ listeners.splice(index, 1);
398
+ }
399
+ }
400
+ }
401
+
402
+ private flowData(): void {
403
+ if (this.buffer.data.length > 0) {
404
+ const fullDataBuffer = this.reducer(this.buffer.data);
405
+
406
+ this.listeners.data.forEach(listener => listener(fullDataBuffer));
407
+
408
+ this.buffer.data.length = 0;
409
+
410
+ // When the buffer is empty, resolve all pending writers
411
+ const pendingWritePromises = [...this.pendingWritePromises];
412
+ this.pendingWritePromises.length = 0;
413
+ pendingWritePromises.forEach(pendingWritePromise => pendingWritePromise());
414
+ }
415
+ }
416
+
417
+ private flowErrors(): void {
418
+ if (this.listeners.error.length > 0) {
419
+ for (const error of this.buffer.error) {
420
+ this.listeners.error.forEach(listener => listener(error));
421
+ }
422
+
423
+ this.buffer.error.length = 0;
424
+ }
425
+ }
426
+
427
+ private flowEnd(): boolean {
428
+ if (this.state.ended) {
429
+ this.listeners.end.forEach(listener => listener());
430
+
431
+ return this.listeners.end.length > 0;
432
+ }
433
+
434
+ return false;
435
+ }
436
+
437
+ destroy(): void {
438
+ if (!this.state.destroyed) {
439
+ this.state.destroyed = true;
440
+ this.state.ended = true;
441
+
442
+ this.buffer.data.length = 0;
443
+ this.buffer.error.length = 0;
444
+
445
+ this.listeners.data.length = 0;
446
+ this.listeners.error.length = 0;
447
+ this.listeners.end.length = 0;
448
+
449
+ this.pendingWritePromises.length = 0;
450
+ }
451
+ }
452
+ }
453
+
454
+ /**
455
+ * Helper to fully read a T readable into a T.
456
+ */
457
+ export function consumeReadable<T>(readable: Readable<T>, reducer: Reducer<T>): T {
458
+ const chunks: T[] = [];
459
+
460
+ let chunk: T | null;
461
+ while ((chunk = readable.read()) !== null) {
462
+ chunks.push(chunk);
463
+ }
464
+
465
+ return reducer(chunks);
466
+ }
467
+
468
+ /**
469
+ * Helper to read a T readable up to a maximum of chunks. If the limit is
470
+ * reached, will return a readable instead to ensure all data can still
471
+ * be read.
472
+ */
473
+ export function consumeReadableWithLimit<T>(readable: Readable<T>, reducer: Reducer<T>, maxChunks: number): T | Readable<T> {
474
+ const chunks: T[] = [];
475
+
476
+ let chunk: T | null | undefined = undefined;
477
+ while ((chunk = readable.read()) !== null && chunks.length < maxChunks) {
478
+ chunks.push(chunk);
479
+ }
480
+
481
+ // If the last chunk is null, it means we reached the end of
482
+ // the readable and return all the data at once
483
+ if (chunk === null && chunks.length > 0) {
484
+ return reducer(chunks);
485
+ }
486
+
487
+ // Otherwise, we still have a chunk, it means we reached the maxChunks
488
+ // value and as such we return a new Readable that first returns
489
+ // the existing read chunks and then continues with reading from
490
+ // the underlying readable.
491
+ return {
492
+ read: () => {
493
+
494
+ // First consume chunks from our array
495
+ if (chunks.length > 0) {
496
+ return chunks.shift()!;
497
+ }
498
+
499
+ // Then ensure to return our last read chunk
500
+ if (typeof chunk !== 'undefined') {
501
+ const lastReadChunk = chunk;
502
+
503
+ // explicitly use undefined here to indicate that we consumed
504
+ // the chunk, which could have either been null or valued.
505
+ chunk = undefined;
506
+
507
+ return lastReadChunk;
508
+ }
509
+
510
+ // Finally delegate back to the Readable
511
+ return readable.read();
512
+ }
513
+ };
514
+ }
515
+
516
+ /**
517
+ * Helper to read a T readable up to a maximum of chunks. If the limit is
518
+ * reached, will return a readable instead to ensure all data can still
519
+ * be read.
520
+ */
521
+ export function peekReadable<T>(readable: Readable<T>, reducer: Reducer<T>, maxChunks: number): T | Readable<T> {
522
+ const chunks: T[] = [];
523
+
524
+ let chunk: T | null | undefined = undefined;
525
+ while ((chunk = readable.read()) !== null && chunks.length < maxChunks) {
526
+ chunks.push(chunk);
527
+ }
528
+
529
+ // If the last chunk is null, it means we reached the end of
530
+ // the readable and return all the data at once
531
+ if (chunk === null && chunks.length > 0) {
532
+ return reducer(chunks);
533
+ }
534
+
535
+ // Otherwise, we still have a chunk, it means we reached the maxChunks
536
+ // value and as such we return a new Readable that first returns
537
+ // the existing read chunks and then continues with reading from
538
+ // the underlying readable.
539
+ return {
540
+ read: () => {
541
+
542
+ // First consume chunks from our array
543
+ if (chunks.length > 0) {
544
+ return chunks.shift()!;
545
+ }
546
+
547
+ // Then ensure to return our last read chunk
548
+ if (typeof chunk !== 'undefined') {
549
+ const lastReadChunk = chunk;
550
+
551
+ // explicitly use undefined here to indicate that we consumed
552
+ // the chunk, which could have either been null or valued.
553
+ chunk = undefined;
554
+
555
+ return lastReadChunk;
556
+ }
557
+
558
+ // Finally delegate back to the Readable
559
+ return readable.read();
560
+ }
561
+ };
562
+ }
563
+
564
+ /**
565
+ * Helper to fully read a T stream into a T.
566
+ */
567
+ export function consumeStream<T>(stream: ReadableStream<T>, reducer: Reducer<T>): Promise<T> {
568
+ return new Promise((resolve, reject) => {
569
+ const chunks: T[] = [];
570
+
571
+ stream.on('data', data => chunks.push(data));
572
+ stream.on('error', error => reject(error));
573
+ stream.on('end', () => resolve(reducer(chunks)));
574
+ });
575
+ }
576
+
577
+ /**
578
+ * Helper to peek up to `maxChunks` into a stream. The return type signals if
579
+ * the stream has ended or not. If not, caller needs to add a `data` listener
580
+ * to continue reading.
581
+ */
582
+ export function peekStream<T>(stream: ReadableStream<T>, maxChunks: number): Promise<ReadableBufferedStream<T>> {
583
+ return new Promise((resolve, reject) => {
584
+ const streamListeners = new DisposableCollection();
585
+
586
+ // Data Listener
587
+ const buffer: T[] = [];
588
+ const dataListener = (chunk: T) => {
589
+
590
+ // Add to buffer
591
+ buffer.push(chunk);
592
+
593
+ // We reached maxChunks and thus need to return
594
+ if (buffer.length > maxChunks) {
595
+
596
+ // Dispose any listeners and ensure to pause the
597
+ // stream so that it can be consumed again by caller
598
+ streamListeners.dispose();
599
+ stream.pause();
600
+
601
+ return resolve({ stream, buffer, ended: false });
602
+ }
603
+ };
604
+
605
+ streamListeners.push(Disposable.create(() => stream.removeListener('data', dataListener)));
606
+ stream.on('data', dataListener);
607
+
608
+ // Error Listener
609
+ const errorListener = (error: Error) => reject(error);
610
+
611
+ streamListeners.push(Disposable.create(() => stream.removeListener('error', errorListener)));
612
+ stream.on('error', errorListener);
613
+
614
+ const endListener = () => resolve({ stream, buffer, ended: true });
615
+
616
+ streamListeners.push(Disposable.create(() => stream.removeListener('end', endListener)));
617
+ stream.on('end', endListener);
618
+ });
619
+ }
620
+
621
+ /**
622
+ * Helper to read a T stream up to a maximum of chunks. If the limit is
623
+ * reached, will return a stream instead to ensure all data can still
624
+ * be read.
625
+ */
626
+ export function consumeStreamWithLimit<T>(stream: ReadableStream<T>, reducer: Reducer<T>, maxChunks: number): Promise<T | ReadableStream<T>> {
627
+ return new Promise((resolve, reject) => {
628
+ const chunks: T[] = [];
629
+
630
+ let wrapperStream: WriteableStream<T> | undefined = undefined;
631
+
632
+ stream.on('data', data => {
633
+
634
+ // If we reach maxChunks, we start to return a stream
635
+ // and make sure that any data we have already read
636
+ // is in it as well
637
+ if (!wrapperStream && chunks.length === maxChunks) {
638
+ wrapperStream = newWriteableStream(reducer);
639
+ while (chunks.length) {
640
+ wrapperStream.write(chunks.shift()!);
641
+ }
642
+
643
+ wrapperStream.write(data);
644
+
645
+ return resolve(wrapperStream);
646
+ }
647
+
648
+ if (wrapperStream) {
649
+ wrapperStream.write(data);
650
+ } else {
651
+ chunks.push(data);
652
+ }
653
+ });
654
+
655
+ stream.on('error', error => {
656
+ if (wrapperStream) {
657
+ wrapperStream.error(error);
658
+ } else {
659
+ return reject(error);
660
+ }
661
+ });
662
+
663
+ stream.on('end', () => {
664
+ if (wrapperStream) {
665
+ while (chunks.length) {
666
+ wrapperStream.write(chunks.shift()!);
667
+ }
668
+
669
+ wrapperStream.end();
670
+ } else {
671
+ return resolve(reducer(chunks));
672
+ }
673
+ });
674
+ });
675
+ }
676
+
677
+ /**
678
+ * Helper to create a readable stream from an existing T.
679
+ */
680
+ export function toStream<T>(t: T, reducer: Reducer<T>): ReadableStream<T> {
681
+ const stream = newWriteableStream<T>(reducer);
682
+
683
+ stream.end(t);
684
+
685
+ return stream;
686
+ }
687
+
688
+ /**
689
+ * Helper to convert a T into a Readable<T>.
690
+ */
691
+ export function toReadable<T>(t: T): Readable<T> {
692
+ let consumed = false;
693
+
694
+ return {
695
+ read: () => {
696
+ if (consumed) {
697
+ return null;
698
+ }
699
+
700
+ consumed = true;
701
+
702
+ return t;
703
+ }
704
+ };
705
+ }
706
+
707
+ /**
708
+ * Helper to transform a readable stream into another stream.
709
+ */
710
+ export function transform<Original, Transformed>(stream: ReadableStreamEvents<Original>, transformer: ITransformer<Original, Transformed>, reducer: Reducer<Transformed>): ReadableStream<Transformed> {
711
+ const target = newWriteableStream<Transformed>(reducer);
712
+
713
+ stream.on('data', data => target.write(transformer.data(data)));
714
+ stream.on('end', () => target.end());
715
+ stream.on('error', error => target.error(transformer.error ? transformer.error(error) : error));
716
+
717
+ return target;
718
+ }