@shitiandmw/node-pty 1.1.0-agent.0

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 (288) hide show
  1. package/LICENSE +69 -0
  2. package/README.md +170 -0
  3. package/binding.gyp +111 -0
  4. package/deps/.editorconfig +2 -0
  5. package/deps/winpty/.drone.yml +17 -0
  6. package/deps/winpty/.gitattributes +19 -0
  7. package/deps/winpty/LICENSE +21 -0
  8. package/deps/winpty/Makefile +166 -0
  9. package/deps/winpty/README.md +151 -0
  10. package/deps/winpty/RELEASES.md +280 -0
  11. package/deps/winpty/VERSION.txt +1 -0
  12. package/deps/winpty/configure +167 -0
  13. package/deps/winpty/misc/BufferResizeTests.cc +90 -0
  14. package/deps/winpty/misc/ChangeScreenBuffer.cc +53 -0
  15. package/deps/winpty/misc/ClearConsole.cc +72 -0
  16. package/deps/winpty/misc/ConinMode.cc +117 -0
  17. package/deps/winpty/misc/ConinMode.ps1 +116 -0
  18. package/deps/winpty/misc/ConoutMode.cc +113 -0
  19. package/deps/winpty/misc/DebugClient.py +42 -0
  20. package/deps/winpty/misc/DebugServer.py +63 -0
  21. package/deps/winpty/misc/DumpLines.py +5 -0
  22. package/deps/winpty/misc/EnableExtendedFlags.txt +46 -0
  23. package/deps/winpty/misc/Font-Report-June2016/CP437-Consolas.txt +528 -0
  24. package/deps/winpty/misc/Font-Report-June2016/CP437-Lucida.txt +633 -0
  25. package/deps/winpty/misc/Font-Report-June2016/CP932.txt +630 -0
  26. package/deps/winpty/misc/Font-Report-June2016/CP936.txt +630 -0
  27. package/deps/winpty/misc/Font-Report-June2016/CP949.txt +630 -0
  28. package/deps/winpty/misc/Font-Report-June2016/CP950.txt +630 -0
  29. package/deps/winpty/misc/Font-Report-June2016/MinimumWindowWidths.txt +16 -0
  30. package/deps/winpty/misc/Font-Report-June2016/Results.txt +4 -0
  31. package/deps/winpty/misc/Font-Report-June2016/Windows10SetFontBugginess.txt +144 -0
  32. package/deps/winpty/misc/FontSurvey.cc +100 -0
  33. package/deps/winpty/misc/FormatChar.h +21 -0
  34. package/deps/winpty/misc/FreezePerfTest.cc +62 -0
  35. package/deps/winpty/misc/GetCh.cc +20 -0
  36. package/deps/winpty/misc/GetConsolePos.cc +41 -0
  37. package/deps/winpty/misc/GetFont.cc +261 -0
  38. package/deps/winpty/misc/IdentifyConsoleWindow.ps1 +51 -0
  39. package/deps/winpty/misc/IsNewConsole.cc +87 -0
  40. package/deps/winpty/misc/MouseInputNotes.txt +90 -0
  41. package/deps/winpty/misc/MoveConsoleWindow.cc +34 -0
  42. package/deps/winpty/misc/Notes.txt +219 -0
  43. package/deps/winpty/misc/OSVersion.cc +27 -0
  44. package/deps/winpty/misc/ScreenBufferFreezeInactive.cc +101 -0
  45. package/deps/winpty/misc/ScreenBufferTest.cc +671 -0
  46. package/deps/winpty/misc/ScreenBufferTest2.cc +151 -0
  47. package/deps/winpty/misc/SelectAllTest.cc +45 -0
  48. package/deps/winpty/misc/SetBufferSize.cc +32 -0
  49. package/deps/winpty/misc/SetCursorPos.cc +10 -0
  50. package/deps/winpty/misc/SetFont.cc +145 -0
  51. package/deps/winpty/misc/SetWindowRect.cc +36 -0
  52. package/deps/winpty/misc/ShowArgv.cc +12 -0
  53. package/deps/winpty/misc/ShowConsoleInput.cc +40 -0
  54. package/deps/winpty/misc/Spew.py +5 -0
  55. package/deps/winpty/misc/TestUtil.cc +172 -0
  56. package/deps/winpty/misc/UnicodeDoubleWidthTest.cc +102 -0
  57. package/deps/winpty/misc/UnicodeWideTest1.cc +246 -0
  58. package/deps/winpty/misc/UnicodeWideTest2.cc +130 -0
  59. package/deps/winpty/misc/UnixEcho.cc +89 -0
  60. package/deps/winpty/misc/Utf16Echo.cc +46 -0
  61. package/deps/winpty/misc/VeryLargeRead.cc +122 -0
  62. package/deps/winpty/misc/VkEscapeTest.cc +56 -0
  63. package/deps/winpty/misc/Win10ResizeWhileFrozen.cc +52 -0
  64. package/deps/winpty/misc/Win10WrapTest1.cc +57 -0
  65. package/deps/winpty/misc/Win10WrapTest2.cc +30 -0
  66. package/deps/winpty/misc/Win32Echo1.cc +26 -0
  67. package/deps/winpty/misc/Win32Echo2.cc +19 -0
  68. package/deps/winpty/misc/Win32Test1.cc +46 -0
  69. package/deps/winpty/misc/Win32Test2.cc +70 -0
  70. package/deps/winpty/misc/Win32Test3.cc +78 -0
  71. package/deps/winpty/misc/Win32Write1.cc +44 -0
  72. package/deps/winpty/misc/WindowsBugCrashReader.cc +27 -0
  73. package/deps/winpty/misc/WriteConsole.cc +106 -0
  74. package/deps/winpty/misc/build32.sh +9 -0
  75. package/deps/winpty/misc/build64.sh +9 -0
  76. package/deps/winpty/misc/color-test.sh +212 -0
  77. package/deps/winpty/misc/font-notes.txt +300 -0
  78. package/deps/winpty/misc/winbug-15048.cc +201 -0
  79. package/deps/winpty/ship/build-pty4j-libpty.bat +36 -0
  80. package/deps/winpty/ship/common_ship.py +53 -0
  81. package/deps/winpty/ship/make_msvc_package.py +165 -0
  82. package/deps/winpty/ship/ship.py +108 -0
  83. package/deps/winpty/src/agent/Agent.cc +613 -0
  84. package/deps/winpty/src/agent/Agent.h +103 -0
  85. package/deps/winpty/src/agent/AgentCreateDesktop.cc +84 -0
  86. package/deps/winpty/src/agent/AgentCreateDesktop.h +28 -0
  87. package/deps/winpty/src/agent/ConsoleFont.cc +632 -0
  88. package/deps/winpty/src/agent/ConsoleFont.h +28 -0
  89. package/deps/winpty/src/agent/ConsoleInput.cc +852 -0
  90. package/deps/winpty/src/agent/ConsoleInput.h +109 -0
  91. package/deps/winpty/src/agent/ConsoleInputReencoding.cc +121 -0
  92. package/deps/winpty/src/agent/ConsoleInputReencoding.h +36 -0
  93. package/deps/winpty/src/agent/ConsoleLine.cc +152 -0
  94. package/deps/winpty/src/agent/ConsoleLine.h +41 -0
  95. package/deps/winpty/src/agent/Coord.h +87 -0
  96. package/deps/winpty/src/agent/DebugShowInput.cc +239 -0
  97. package/deps/winpty/src/agent/DebugShowInput.h +32 -0
  98. package/deps/winpty/src/agent/DefaultInputMap.cc +422 -0
  99. package/deps/winpty/src/agent/DefaultInputMap.h +28 -0
  100. package/deps/winpty/src/agent/DsrSender.h +30 -0
  101. package/deps/winpty/src/agent/EventLoop.cc +99 -0
  102. package/deps/winpty/src/agent/EventLoop.h +47 -0
  103. package/deps/winpty/src/agent/InputMap.cc +246 -0
  104. package/deps/winpty/src/agent/InputMap.h +114 -0
  105. package/deps/winpty/src/agent/LargeConsoleRead.cc +71 -0
  106. package/deps/winpty/src/agent/LargeConsoleRead.h +68 -0
  107. package/deps/winpty/src/agent/NamedPipe.cc +378 -0
  108. package/deps/winpty/src/agent/NamedPipe.h +125 -0
  109. package/deps/winpty/src/agent/Scraper.cc +699 -0
  110. package/deps/winpty/src/agent/Scraper.h +103 -0
  111. package/deps/winpty/src/agent/SimplePool.h +75 -0
  112. package/deps/winpty/src/agent/SmallRect.h +143 -0
  113. package/deps/winpty/src/agent/Terminal.cc +535 -0
  114. package/deps/winpty/src/agent/Terminal.h +69 -0
  115. package/deps/winpty/src/agent/UnicodeEncoding.h +157 -0
  116. package/deps/winpty/src/agent/UnicodeEncodingTest.cc +189 -0
  117. package/deps/winpty/src/agent/Win32Console.cc +107 -0
  118. package/deps/winpty/src/agent/Win32Console.h +67 -0
  119. package/deps/winpty/src/agent/Win32ConsoleBuffer.cc +193 -0
  120. package/deps/winpty/src/agent/Win32ConsoleBuffer.h +99 -0
  121. package/deps/winpty/src/agent/main.cc +114 -0
  122. package/deps/winpty/src/agent/subdir.mk +61 -0
  123. package/deps/winpty/src/configurations.gypi +60 -0
  124. package/deps/winpty/src/debugserver/DebugServer.cc +117 -0
  125. package/deps/winpty/src/debugserver/subdir.mk +41 -0
  126. package/deps/winpty/src/include/winpty.h +242 -0
  127. package/deps/winpty/src/include/winpty_constants.h +131 -0
  128. package/deps/winpty/src/libwinpty/AgentLocation.cc +75 -0
  129. package/deps/winpty/src/libwinpty/AgentLocation.h +28 -0
  130. package/deps/winpty/src/libwinpty/LibWinptyException.h +54 -0
  131. package/deps/winpty/src/libwinpty/WinptyInternal.h +72 -0
  132. package/deps/winpty/src/libwinpty/subdir.mk +46 -0
  133. package/deps/winpty/src/libwinpty/winpty.cc +970 -0
  134. package/deps/winpty/src/shared/AgentMsg.h +38 -0
  135. package/deps/winpty/src/shared/BackgroundDesktop.cc +122 -0
  136. package/deps/winpty/src/shared/BackgroundDesktop.h +73 -0
  137. package/deps/winpty/src/shared/Buffer.cc +103 -0
  138. package/deps/winpty/src/shared/Buffer.h +102 -0
  139. package/deps/winpty/src/shared/DebugClient.cc +187 -0
  140. package/deps/winpty/src/shared/DebugClient.h +38 -0
  141. package/deps/winpty/src/shared/GenRandom.cc +138 -0
  142. package/deps/winpty/src/shared/GenRandom.h +55 -0
  143. package/deps/winpty/src/shared/GetCommitHash.bat +13 -0
  144. package/deps/winpty/src/shared/Mutex.h +54 -0
  145. package/deps/winpty/src/shared/OsModule.h +63 -0
  146. package/deps/winpty/src/shared/OwnedHandle.cc +36 -0
  147. package/deps/winpty/src/shared/OwnedHandle.h +45 -0
  148. package/deps/winpty/src/shared/PrecompiledHeader.h +43 -0
  149. package/deps/winpty/src/shared/StringBuilder.h +227 -0
  150. package/deps/winpty/src/shared/StringBuilderTest.cc +114 -0
  151. package/deps/winpty/src/shared/StringUtil.cc +55 -0
  152. package/deps/winpty/src/shared/StringUtil.h +80 -0
  153. package/deps/winpty/src/shared/TimeMeasurement.h +63 -0
  154. package/deps/winpty/src/shared/UnixCtrlChars.h +45 -0
  155. package/deps/winpty/src/shared/UpdateGenVersion.bat +20 -0
  156. package/deps/winpty/src/shared/WindowsSecurity.cc +460 -0
  157. package/deps/winpty/src/shared/WindowsSecurity.h +104 -0
  158. package/deps/winpty/src/shared/WindowsVersion.cc +252 -0
  159. package/deps/winpty/src/shared/WindowsVersion.h +29 -0
  160. package/deps/winpty/src/shared/WinptyAssert.cc +55 -0
  161. package/deps/winpty/src/shared/WinptyAssert.h +64 -0
  162. package/deps/winpty/src/shared/WinptyException.cc +57 -0
  163. package/deps/winpty/src/shared/WinptyException.h +43 -0
  164. package/deps/winpty/src/shared/WinptyVersion.cc +42 -0
  165. package/deps/winpty/src/shared/WinptyVersion.h +27 -0
  166. package/deps/winpty/src/shared/winpty_snprintf.h +99 -0
  167. package/deps/winpty/src/subdir.mk +5 -0
  168. package/deps/winpty/src/tests/subdir.mk +28 -0
  169. package/deps/winpty/src/tests/trivial_test.cc +158 -0
  170. package/deps/winpty/src/unix-adapter/InputHandler.cc +114 -0
  171. package/deps/winpty/src/unix-adapter/InputHandler.h +56 -0
  172. package/deps/winpty/src/unix-adapter/OutputHandler.cc +80 -0
  173. package/deps/winpty/src/unix-adapter/OutputHandler.h +53 -0
  174. package/deps/winpty/src/unix-adapter/Util.cc +86 -0
  175. package/deps/winpty/src/unix-adapter/Util.h +31 -0
  176. package/deps/winpty/src/unix-adapter/WakeupFd.cc +70 -0
  177. package/deps/winpty/src/unix-adapter/WakeupFd.h +42 -0
  178. package/deps/winpty/src/unix-adapter/main.cc +729 -0
  179. package/deps/winpty/src/unix-adapter/subdir.mk +41 -0
  180. package/deps/winpty/src/winpty.gyp +234 -0
  181. package/deps/winpty/vcbuild.bat +83 -0
  182. package/lib/conpty_console_list_agent.js +16 -0
  183. package/lib/conpty_console_list_agent.js.map +1 -0
  184. package/lib/eventEmitter2.js +47 -0
  185. package/lib/eventEmitter2.js.map +1 -0
  186. package/lib/eventEmitter2.test.js +30 -0
  187. package/lib/eventEmitter2.test.js.map +1 -0
  188. package/lib/index.js +52 -0
  189. package/lib/index.js.map +1 -0
  190. package/lib/interfaces.js +7 -0
  191. package/lib/interfaces.js.map +1 -0
  192. package/lib/shared/conout.js +11 -0
  193. package/lib/shared/conout.js.map +1 -0
  194. package/lib/terminal.js +190 -0
  195. package/lib/terminal.js.map +1 -0
  196. package/lib/terminal.test.js +139 -0
  197. package/lib/terminal.test.js.map +1 -0
  198. package/lib/testUtils.test.js +28 -0
  199. package/lib/testUtils.test.js.map +1 -0
  200. package/lib/types.js +7 -0
  201. package/lib/types.js.map +1 -0
  202. package/lib/unixTerminal.js +346 -0
  203. package/lib/unixTerminal.js.map +1 -0
  204. package/lib/unixTerminal.test.js +351 -0
  205. package/lib/unixTerminal.test.js.map +1 -0
  206. package/lib/utils.js +39 -0
  207. package/lib/utils.js.map +1 -0
  208. package/lib/windowsConoutConnection.js +125 -0
  209. package/lib/windowsConoutConnection.js.map +1 -0
  210. package/lib/windowsPtyAgent.js +320 -0
  211. package/lib/windowsPtyAgent.js.map +1 -0
  212. package/lib/windowsPtyAgent.test.js +90 -0
  213. package/lib/windowsPtyAgent.test.js.map +1 -0
  214. package/lib/windowsTerminal.js +199 -0
  215. package/lib/windowsTerminal.js.map +1 -0
  216. package/lib/windowsTerminal.test.js +219 -0
  217. package/lib/windowsTerminal.test.js.map +1 -0
  218. package/lib/worker/conoutSocketWorker.js +22 -0
  219. package/lib/worker/conoutSocketWorker.js.map +1 -0
  220. package/package.json +68 -0
  221. package/prebuilds/darwin-arm64/pty.node +0 -0
  222. package/prebuilds/darwin-arm64/spawn-helper +0 -0
  223. package/prebuilds/darwin-x64/pty.node +0 -0
  224. package/prebuilds/darwin-x64/spawn-helper +0 -0
  225. package/prebuilds/win32-arm64/conpty/OpenConsole.exe +0 -0
  226. package/prebuilds/win32-arm64/conpty/conpty.dll +0 -0
  227. package/prebuilds/win32-arm64/conpty.node +0 -0
  228. package/prebuilds/win32-arm64/conpty.pdb +0 -0
  229. package/prebuilds/win32-arm64/conpty_console_list.node +0 -0
  230. package/prebuilds/win32-arm64/conpty_console_list.pdb +0 -0
  231. package/prebuilds/win32-arm64/pty.node +0 -0
  232. package/prebuilds/win32-arm64/pty.pdb +0 -0
  233. package/prebuilds/win32-arm64/winpty-agent.exe +0 -0
  234. package/prebuilds/win32-arm64/winpty-agent.pdb +0 -0
  235. package/prebuilds/win32-arm64/winpty.dll +0 -0
  236. package/prebuilds/win32-arm64/winpty.pdb +0 -0
  237. package/prebuilds/win32-x64/conpty/OpenConsole.exe +0 -0
  238. package/prebuilds/win32-x64/conpty/conpty.dll +0 -0
  239. package/prebuilds/win32-x64/conpty.node +0 -0
  240. package/prebuilds/win32-x64/conpty.pdb +0 -0
  241. package/prebuilds/win32-x64/conpty_console_list.node +0 -0
  242. package/prebuilds/win32-x64/conpty_console_list.pdb +0 -0
  243. package/prebuilds/win32-x64/pty.node +0 -0
  244. package/prebuilds/win32-x64/pty.pdb +0 -0
  245. package/prebuilds/win32-x64/winpty-agent.exe +0 -0
  246. package/prebuilds/win32-x64/winpty-agent.pdb +0 -0
  247. package/prebuilds/win32-x64/winpty.dll +0 -0
  248. package/prebuilds/win32-x64/winpty.pdb +0 -0
  249. package/scripts/gen-compile-commands.js +8 -0
  250. package/scripts/increment-version.js +54 -0
  251. package/scripts/post-install.js +99 -0
  252. package/scripts/prebuild.js +39 -0
  253. package/scripts/sync-prebuild.js +31 -0
  254. package/scripts/verify-darwin-fd-leak.js +63 -0
  255. package/src/conpty_console_list_agent.ts +15 -0
  256. package/src/eventEmitter2.test.ts +30 -0
  257. package/src/eventEmitter2.ts +48 -0
  258. package/src/index.ts +52 -0
  259. package/src/interfaces.ts +130 -0
  260. package/src/native.d.ts +54 -0
  261. package/src/shared/conout.ts +15 -0
  262. package/src/terminal.test.ts +119 -0
  263. package/src/terminal.ts +211 -0
  264. package/src/testUtils.test.ts +23 -0
  265. package/src/tsconfig.json +22 -0
  266. package/src/types.ts +15 -0
  267. package/src/unix/pty.cc +808 -0
  268. package/src/unix/spawn-helper.cc +23 -0
  269. package/src/unixTerminal.test.ts +367 -0
  270. package/src/unixTerminal.ts +388 -0
  271. package/src/utils.ts +29 -0
  272. package/src/win/conpty.cc +583 -0
  273. package/src/win/conpty.h +41 -0
  274. package/src/win/conpty_console_list.cc +44 -0
  275. package/src/win/path_util.cc +95 -0
  276. package/src/win/path_util.h +26 -0
  277. package/src/win/winpty.cc +333 -0
  278. package/src/windowsConoutConnection.ts +82 -0
  279. package/src/windowsPtyAgent.test.ts +94 -0
  280. package/src/windowsPtyAgent.ts +321 -0
  281. package/src/windowsTerminal.test.ts +229 -0
  282. package/src/windowsTerminal.ts +203 -0
  283. package/src/worker/conoutSocketWorker.ts +22 -0
  284. package/third_party/conpty/1.23.251008001/win10-arm64/OpenConsole.exe +0 -0
  285. package/third_party/conpty/1.23.251008001/win10-arm64/conpty.dll +0 -0
  286. package/third_party/conpty/1.23.251008001/win10-x64/OpenConsole.exe +0 -0
  287. package/third_party/conpty/1.23.251008001/win10-x64/conpty.dll +0 -0
  288. package/typings/node-pty.d.ts +211 -0
@@ -0,0 +1,583 @@
1
+ /**
2
+ * Copyright (c) 2013-2015, Christopher Jeffrey, Peter Sunde (MIT License)
3
+ * Copyright (c) 2016, Daniel Imms (MIT License).
4
+ * Copyright (c) 2018, Microsoft Corporation (MIT License).
5
+ *
6
+ * pty.cc:
7
+ * This file is responsible for starting processes
8
+ * with pseudo-terminal file descriptors.
9
+ */
10
+
11
+ #define _WIN32_WINNT 0x600
12
+
13
+ #define NODE_ADDON_API_DISABLE_DEPRECATED
14
+ #include <node_api.h>
15
+ #include <assert.h>
16
+ #include <Shlwapi.h> // PathCombine, PathIsRelative
17
+ #include <sstream>
18
+ #include <iostream>
19
+ #include <string>
20
+ #include <thread>
21
+ #include <vector>
22
+ #include <Windows.h>
23
+ #include <strsafe.h>
24
+ #include "path_util.h"
25
+ #include "conpty.h"
26
+
27
+ // Taken from the RS5 Windows SDK, but redefined here in case we're targeting <= 17134
28
+ #ifndef PROC_THREAD_ATTRIBUTE_PSEUDOCONSOLE
29
+ #define PROC_THREAD_ATTRIBUTE_PSEUDOCONSOLE \
30
+ ProcThreadAttributeValue(22, FALSE, TRUE, FALSE)
31
+
32
+ typedef VOID* HPCON;
33
+ typedef HRESULT (__stdcall *PFNCREATEPSEUDOCONSOLE)(COORD c, HANDLE hIn, HANDLE hOut, DWORD dwFlags, HPCON* phpcon);
34
+ typedef HRESULT (__stdcall *PFNRESIZEPSEUDOCONSOLE)(HPCON hpc, COORD newSize);
35
+ typedef HRESULT (__stdcall *PFNCLEARPSEUDOCONSOLE)(HPCON hpc);
36
+ typedef void (__stdcall *PFNCLOSEPSEUDOCONSOLE)(HPCON hpc);
37
+ typedef void (__stdcall *PFNRELEASEPSEUDOCONSOLE)(HPCON hpc);
38
+
39
+ #endif
40
+
41
+ struct pty_baton {
42
+ int id;
43
+ HANDLE hIn;
44
+ HANDLE hOut;
45
+ HPCON hpc;
46
+
47
+ HANDLE hShell;
48
+
49
+ pty_baton(int _id, HANDLE _hIn, HANDLE _hOut, HPCON _hpc) : id(_id), hIn(_hIn), hOut(_hOut), hpc(_hpc) {};
50
+ };
51
+
52
+ static std::vector<std::unique_ptr<pty_baton>> ptyHandles;
53
+ static volatile LONG ptyCounter;
54
+
55
+ static pty_baton* get_pty_baton(int id) {
56
+ auto it = std::find_if(ptyHandles.begin(), ptyHandles.end(), [id](const auto& ptyHandle) {
57
+ return ptyHandle->id == id;
58
+ });
59
+ if (it != ptyHandles.end()) {
60
+ return it->get();
61
+ }
62
+ return nullptr;
63
+ }
64
+
65
+ static bool remove_pty_baton(int id) {
66
+ auto it = std::remove_if(ptyHandles.begin(), ptyHandles.end(), [id](const auto& ptyHandle) {
67
+ return ptyHandle->id == id;
68
+ });
69
+ if (it != ptyHandles.end()) {
70
+ ptyHandles.erase(it);
71
+ return true;
72
+ }
73
+ return false;
74
+ }
75
+
76
+ struct ExitEvent {
77
+ int exit_code = 0;
78
+ };
79
+
80
+ void SetupExitCallback(Napi::Env env, Napi::Function cb, pty_baton* baton) {
81
+ std::thread *th = new std::thread;
82
+ // Don't use Napi::AsyncWorker which is limited by UV_THREADPOOL_SIZE.
83
+ auto tsfn = Napi::ThreadSafeFunction::New(
84
+ env,
85
+ cb, // JavaScript function called asynchronously
86
+ "SetupExitCallback_resource", // Name
87
+ 0, // Unlimited queue
88
+ 1, // Only one thread will use this initially
89
+ [th](Napi::Env) { // Finalizer used to clean threads up
90
+ th->join();
91
+ delete th;
92
+ });
93
+ *th = std::thread([tsfn = std::move(tsfn), baton] {
94
+ auto callback = [](Napi::Env env, Napi::Function cb, ExitEvent *exit_event) {
95
+ cb.Call({Napi::Number::New(env, exit_event->exit_code)});
96
+ delete exit_event;
97
+ };
98
+
99
+ ExitEvent *exit_event = new ExitEvent;
100
+ // Wait for process to complete.
101
+ WaitForSingleObject(baton->hShell, INFINITE);
102
+ // Get process exit code.
103
+ GetExitCodeProcess(baton->hShell, (LPDWORD)(&exit_event->exit_code));
104
+ // Clean up handles
105
+ CloseHandle(baton->hShell);
106
+ assert(remove_pty_baton(baton->id));
107
+
108
+ auto status = tsfn.BlockingCall(exit_event, callback); // In main thread
109
+ switch (status) {
110
+ case napi_closing:
111
+ break;
112
+
113
+ case napi_queue_full:
114
+ Napi::Error::Fatal("SetupExitCallback", "Queue was full");
115
+
116
+ case napi_ok:
117
+ if (tsfn.Release() != napi_ok) {
118
+ Napi::Error::Fatal("SetupExitCallback", "ThreadSafeFunction.Release() failed");
119
+ }
120
+ break;
121
+
122
+ default:
123
+ Napi::Error::Fatal("SetupExitCallback", "ThreadSafeFunction.BlockingCall() failed");
124
+ }
125
+ });
126
+ }
127
+
128
+ Napi::Error errorWithCode(const Napi::CallbackInfo& info, const char* text) {
129
+ std::stringstream errorText;
130
+ errorText << text;
131
+ errorText << ", error code: " << GetLastError();
132
+ return Napi::Error::New(info.Env(), errorText.str());
133
+ }
134
+
135
+ // Returns a new server named pipe. It has not yet been connected.
136
+ bool createDataServerPipe(bool write,
137
+ std::wstring kind,
138
+ HANDLE* hServer,
139
+ std::wstring &name,
140
+ const std::wstring &pipeName)
141
+ {
142
+ *hServer = INVALID_HANDLE_VALUE;
143
+
144
+ name = L"\\\\.\\pipe\\" + pipeName + L"-" + kind;
145
+
146
+ const DWORD winOpenMode = PIPE_ACCESS_INBOUND | PIPE_ACCESS_OUTBOUND | FILE_FLAG_FIRST_PIPE_INSTANCE/* | FILE_FLAG_OVERLAPPED */;
147
+
148
+ SECURITY_ATTRIBUTES sa = {};
149
+ sa.nLength = sizeof(sa);
150
+
151
+ *hServer = CreateNamedPipeW(
152
+ name.c_str(),
153
+ /*dwOpenMode=*/winOpenMode,
154
+ /*dwPipeMode=*/PIPE_TYPE_BYTE | PIPE_READMODE_BYTE | PIPE_WAIT,
155
+ /*nMaxInstances=*/1,
156
+ /*nOutBufferSize=*/128 * 1024,
157
+ /*nInBufferSize=*/128 * 1024,
158
+ /*nDefaultTimeOut=*/30000,
159
+ &sa);
160
+
161
+ return *hServer != INVALID_HANDLE_VALUE;
162
+ }
163
+
164
+ HANDLE LoadConptyDll(const Napi::CallbackInfo& info,
165
+ const bool useConptyDll)
166
+ {
167
+ if (!useConptyDll) {
168
+ return LoadLibraryExW(L"kernel32.dll", 0, 0);
169
+ }
170
+ wchar_t currentDir[MAX_PATH];
171
+ HMODULE hModule = GetModuleHandleA("conpty.node");
172
+ if (hModule == NULL) {
173
+ throw errorWithCode(info, "Failed to get conpty.node module handle");
174
+ }
175
+ DWORD result = GetModuleFileNameW(hModule, currentDir, MAX_PATH);
176
+ if (result == 0) {
177
+ throw errorWithCode(info, "Failed to get conpty.node module file name");
178
+ }
179
+ PathRemoveFileSpecW(currentDir);
180
+ wchar_t conptyDllPath[MAX_PATH];
181
+ PathCombineW(conptyDllPath, currentDir, L"conpty\\conpty.dll");
182
+ if (!path_util::file_exists(conptyDllPath)) {
183
+ std::wstring errorMessage = L"Cannot find conpty.dll at " + std::wstring(conptyDllPath);
184
+ std::string errorMessageStr = path_util::wstring_to_string(errorMessage);
185
+ throw errorWithCode(info, errorMessageStr.c_str());
186
+ }
187
+
188
+ return LoadLibraryW(conptyDllPath);
189
+ }
190
+
191
+ HRESULT CreateNamedPipesAndPseudoConsole(const Napi::CallbackInfo& info,
192
+ COORD size,
193
+ DWORD dwFlags,
194
+ HANDLE *phInput,
195
+ HANDLE *phOutput,
196
+ HPCON* phPC,
197
+ std::wstring& inName,
198
+ std::wstring& outName,
199
+ const std::wstring& pipeName,
200
+ const bool useConptyDll)
201
+ {
202
+ HANDLE hLibrary = LoadConptyDll(info, useConptyDll);
203
+ DWORD error = GetLastError();
204
+ bool fLoadedDll = hLibrary != nullptr;
205
+ if (fLoadedDll)
206
+ {
207
+ PFNCREATEPSEUDOCONSOLE const pfnCreate = (PFNCREATEPSEUDOCONSOLE)GetProcAddress(
208
+ (HMODULE)hLibrary,
209
+ useConptyDll ? "ConptyCreatePseudoConsole" : "CreatePseudoConsole");
210
+ if (pfnCreate)
211
+ {
212
+ if (phPC == NULL || phInput == NULL || phOutput == NULL)
213
+ {
214
+ return E_INVALIDARG;
215
+ }
216
+
217
+ bool success = createDataServerPipe(true, L"in", phInput, inName, pipeName);
218
+ if (!success)
219
+ {
220
+ return HRESULT_FROM_WIN32(GetLastError());
221
+ }
222
+ success = createDataServerPipe(false, L"out", phOutput, outName, pipeName);
223
+ if (!success)
224
+ {
225
+ return HRESULT_FROM_WIN32(GetLastError());
226
+ }
227
+ return pfnCreate(size, *phInput, *phOutput, dwFlags, phPC);
228
+ }
229
+ else
230
+ {
231
+ // Failed to find CreatePseudoConsole in kernel32. This is likely because
232
+ // the user is not running a build of Windows that supports that API.
233
+ // We should fall back to winpty in this case.
234
+ return HRESULT_FROM_WIN32(GetLastError());
235
+ }
236
+ } else {
237
+ throw errorWithCode(info, "Failed to load conpty.dll");
238
+ }
239
+
240
+ // Failed to find kernel32. This is realy unlikely - honestly no idea how
241
+ // this is even possible to hit. But if it does happen, fall back to winpty.
242
+ return HRESULT_FROM_WIN32(GetLastError());
243
+ }
244
+
245
+ static Napi::Value PtyStartProcess(const Napi::CallbackInfo& info) {
246
+ Napi::Env env(info.Env());
247
+ Napi::HandleScope scope(env);
248
+
249
+ Napi::Object marshal;
250
+ std::wstring inName, outName;
251
+ BOOL fSuccess = FALSE;
252
+ std::unique_ptr<wchar_t[]> mutableCommandline;
253
+ PROCESS_INFORMATION _piClient{};
254
+
255
+ if (info.Length() != 7 ||
256
+ !info[0].IsString() ||
257
+ !info[1].IsNumber() ||
258
+ !info[2].IsNumber() ||
259
+ !info[3].IsBoolean() ||
260
+ !info[4].IsString() ||
261
+ !info[5].IsBoolean() ||
262
+ !info[6].IsBoolean()) {
263
+ throw Napi::Error::New(env, "Usage: pty.startProcess(file, cols, rows, debug, pipeName, inheritCursor, useConptyDll)");
264
+ }
265
+
266
+ const std::wstring filename(path_util::to_wstring(info[0].As<Napi::String>()));
267
+ const SHORT cols = static_cast<SHORT>(info[1].As<Napi::Number>().Uint32Value());
268
+ const SHORT rows = static_cast<SHORT>(info[2].As<Napi::Number>().Uint32Value());
269
+ const bool debug = info[3].As<Napi::Boolean>().Value();
270
+ const std::wstring pipeName(path_util::to_wstring(info[4].As<Napi::String>()));
271
+ const bool inheritCursor = info[5].As<Napi::Boolean>().Value();
272
+ const bool useConptyDll = info[6].As<Napi::Boolean>().Value();
273
+
274
+ // use environment 'Path' variable to determine location of
275
+ // the relative path that we have recieved (e.g cmd.exe)
276
+ std::wstring shellpath;
277
+ if (::PathIsRelativeW(filename.c_str())) {
278
+ shellpath = path_util::get_shell_path(filename.c_str());
279
+ } else {
280
+ shellpath = filename;
281
+ }
282
+
283
+ if (shellpath.empty() || !path_util::file_exists(shellpath)) {
284
+ std::string why;
285
+ why += "File not found: ";
286
+ why += path_util::wstring_to_string(shellpath);
287
+ throw Napi::Error::New(env, why);
288
+ }
289
+
290
+ HANDLE hIn, hOut;
291
+ HPCON hpc;
292
+ HRESULT hr = CreateNamedPipesAndPseudoConsole(info, {cols, rows}, inheritCursor ? 1/*PSEUDOCONSOLE_INHERIT_CURSOR*/ : 0, &hIn, &hOut, &hpc, inName, outName, pipeName, useConptyDll);
293
+
294
+ // Restore default handling of ctrl+c
295
+ SetConsoleCtrlHandler(NULL, FALSE);
296
+
297
+ // Set return values
298
+ marshal = Napi::Object::New(env);
299
+
300
+ if (SUCCEEDED(hr)) {
301
+ // We were able to instantiate a conpty
302
+ const int ptyId = InterlockedIncrement(&ptyCounter);
303
+ marshal.Set("pty", Napi::Number::New(env, ptyId));
304
+ ptyHandles.emplace_back(
305
+ std::make_unique<pty_baton>(ptyId, hIn, hOut, hpc));
306
+ } else {
307
+ throw Napi::Error::New(env, "Cannot launch conpty");
308
+ }
309
+
310
+ std::string inNameStr = path_util::wstring_to_string(inName);
311
+ if (inNameStr.empty()) {
312
+ throw Napi::Error::New(env, "Failed to initialize conpty conin");
313
+ }
314
+ std::string outNameStr = path_util::wstring_to_string(outName);
315
+ if (outNameStr.empty()) {
316
+ throw Napi::Error::New(env, "Failed to initialize conpty conout");
317
+ }
318
+
319
+ marshal.Set("fd", Napi::Number::New(env, -1));
320
+ marshal.Set("conin", Napi::String::New(env, inNameStr));
321
+ marshal.Set("conout", Napi::String::New(env, outNameStr));
322
+ return marshal;
323
+ }
324
+
325
+ static Napi::Value PtyConnect(const Napi::CallbackInfo& info) {
326
+ Napi::Env env(info.Env());
327
+ Napi::HandleScope scope(env);
328
+
329
+ // If we're working with conpty's we need to call ConnectNamedPipe here AFTER
330
+ // the Socket has attempted to connect to the other end, then actually
331
+ // spawn the process here.
332
+
333
+ std::stringstream errorText;
334
+ BOOL fSuccess = FALSE;
335
+
336
+ if (info.Length() != 6 ||
337
+ !info[0].IsNumber() ||
338
+ !info[1].IsString() ||
339
+ !info[2].IsString() ||
340
+ !info[3].IsArray() ||
341
+ !info[4].IsBoolean() ||
342
+ !info[5].IsFunction()) {
343
+ throw Napi::Error::New(env, "Usage: pty.connect(id, cmdline, cwd, env, useConptyDll, exitCallback)");
344
+ }
345
+
346
+ const int id = info[0].As<Napi::Number>().Int32Value();
347
+ const std::wstring cmdline(path_util::to_wstring(info[1].As<Napi::String>()));
348
+ const std::wstring cwd(path_util::to_wstring(info[2].As<Napi::String>()));
349
+ const Napi::Array envValues = info[3].As<Napi::Array>();
350
+ const bool useConptyDll = info[4].As<Napi::Boolean>().Value();
351
+ Napi::Function exitCallback = info[5].As<Napi::Function>();
352
+
353
+ // Fetch pty handle from ID and start process
354
+ pty_baton* handle = get_pty_baton(id);
355
+ if (!handle) {
356
+ throw Napi::Error::New(env, "Invalid pty handle");
357
+ }
358
+
359
+ // Prepare command line
360
+ std::unique_ptr<wchar_t[]> mutableCommandline = std::make_unique<wchar_t[]>(cmdline.length() + 1);
361
+ HRESULT hr = StringCchCopyW(mutableCommandline.get(), cmdline.length() + 1, cmdline.c_str());
362
+
363
+ // Prepare cwd
364
+ std::unique_ptr<wchar_t[]> mutableCwd = std::make_unique<wchar_t[]>(cwd.length() + 1);
365
+ hr = StringCchCopyW(mutableCwd.get(), cwd.length() + 1, cwd.c_str());
366
+
367
+ // Prepare environment
368
+ std::wstring envStr;
369
+ if (!envValues.IsEmpty()) {
370
+ std::wstring envBlock;
371
+ for(uint32_t i = 0; i < envValues.Length(); i++) {
372
+ envBlock += path_util::to_wstring(envValues.Get(i).As<Napi::String>());
373
+ envBlock += L'\0';
374
+ }
375
+ envBlock += L'\0';
376
+ envStr = std::move(envBlock);
377
+ }
378
+ std::vector<wchar_t> envV(envStr.cbegin(), envStr.cend());
379
+ LPWSTR envArg = envV.empty() ? nullptr : envV.data();
380
+
381
+ ConnectNamedPipe(handle->hIn, nullptr);
382
+ ConnectNamedPipe(handle->hOut, nullptr);
383
+
384
+ // Attach the pseudoconsole to the client application we're creating
385
+ STARTUPINFOEXW siEx{0};
386
+ siEx.StartupInfo.cb = sizeof(STARTUPINFOEXW);
387
+ siEx.StartupInfo.dwFlags |= STARTF_USESTDHANDLES;
388
+ siEx.StartupInfo.hStdError = nullptr;
389
+ siEx.StartupInfo.hStdInput = nullptr;
390
+ siEx.StartupInfo.hStdOutput = nullptr;
391
+
392
+ SIZE_T size = 0;
393
+ InitializeProcThreadAttributeList(NULL, 1, 0, &size);
394
+ BYTE *attrList = new BYTE[size];
395
+ siEx.lpAttributeList = reinterpret_cast<PPROC_THREAD_ATTRIBUTE_LIST>(attrList);
396
+
397
+ fSuccess = InitializeProcThreadAttributeList(siEx.lpAttributeList, 1, 0, &size);
398
+ if (!fSuccess) {
399
+ throw errorWithCode(info, "InitializeProcThreadAttributeList failed");
400
+ }
401
+ fSuccess = UpdateProcThreadAttribute(siEx.lpAttributeList,
402
+ 0,
403
+ PROC_THREAD_ATTRIBUTE_PSEUDOCONSOLE,
404
+ handle->hpc,
405
+ sizeof(HPCON),
406
+ NULL,
407
+ NULL);
408
+ if (!fSuccess) {
409
+ throw errorWithCode(info, "UpdateProcThreadAttribute failed");
410
+ }
411
+
412
+ PROCESS_INFORMATION piClient{};
413
+ fSuccess = !!CreateProcessW(
414
+ nullptr,
415
+ mutableCommandline.get(),
416
+ nullptr, // lpProcessAttributes
417
+ nullptr, // lpThreadAttributes
418
+ false, // bInheritHandles VERY IMPORTANT that this is false
419
+ EXTENDED_STARTUPINFO_PRESENT | CREATE_UNICODE_ENVIRONMENT, // dwCreationFlags
420
+ envArg, // lpEnvironment
421
+ mutableCwd.get(), // lpCurrentDirectory
422
+ &siEx.StartupInfo, // lpStartupInfo
423
+ &piClient // lpProcessInformation
424
+ );
425
+ if (!fSuccess) {
426
+ throw errorWithCode(info, "Cannot create process");
427
+ }
428
+
429
+ HANDLE hLibrary = LoadConptyDll(info, useConptyDll);
430
+ bool fLoadedDll = hLibrary != nullptr;
431
+ if (useConptyDll && fLoadedDll)
432
+ {
433
+ PFNRELEASEPSEUDOCONSOLE const pfnReleasePseudoConsole = (PFNRELEASEPSEUDOCONSOLE)GetProcAddress(
434
+ (HMODULE)hLibrary, "ConptyReleasePseudoConsole");
435
+ if (pfnReleasePseudoConsole)
436
+ {
437
+ pfnReleasePseudoConsole(handle->hpc);
438
+ }
439
+ }
440
+
441
+ // Update handle
442
+ handle->hShell = piClient.hProcess;
443
+
444
+ // Close the thread handle to avoid resource leak
445
+ CloseHandle(piClient.hThread);
446
+ // Close the input read and output write handle of the pseudoconsole
447
+ CloseHandle(handle->hIn);
448
+ CloseHandle(handle->hOut);
449
+
450
+ SetupExitCallback(env, exitCallback, handle);
451
+
452
+ // Return
453
+ auto marshal = Napi::Object::New(env);
454
+ marshal.Set("pid", Napi::Number::New(env, piClient.dwProcessId));
455
+ return marshal;
456
+ }
457
+
458
+ static Napi::Value PtyResize(const Napi::CallbackInfo& info) {
459
+ Napi::Env env(info.Env());
460
+ Napi::HandleScope scope(env);
461
+
462
+ if (info.Length() != 4 ||
463
+ !info[0].IsNumber() ||
464
+ !info[1].IsNumber() ||
465
+ !info[2].IsNumber() ||
466
+ !info[3].IsBoolean()) {
467
+ throw Napi::Error::New(env, "Usage: pty.resize(id, cols, rows, useConptyDll)");
468
+ }
469
+
470
+ int id = info[0].As<Napi::Number>().Int32Value();
471
+ SHORT cols = static_cast<SHORT>(info[1].As<Napi::Number>().Uint32Value());
472
+ SHORT rows = static_cast<SHORT>(info[2].As<Napi::Number>().Uint32Value());
473
+ const bool useConptyDll = info[3].As<Napi::Boolean>().Value();
474
+
475
+ const pty_baton* handle = get_pty_baton(id);
476
+
477
+ if (handle != nullptr) {
478
+ HANDLE hLibrary = LoadConptyDll(info, useConptyDll);
479
+ bool fLoadedDll = hLibrary != nullptr;
480
+ if (fLoadedDll)
481
+ {
482
+ PFNRESIZEPSEUDOCONSOLE const pfnResizePseudoConsole = (PFNRESIZEPSEUDOCONSOLE)GetProcAddress(
483
+ (HMODULE)hLibrary,
484
+ useConptyDll ? "ConptyResizePseudoConsole" : "ResizePseudoConsole");
485
+ if (pfnResizePseudoConsole)
486
+ {
487
+ COORD size = {cols, rows};
488
+ pfnResizePseudoConsole(handle->hpc, size);
489
+ }
490
+ }
491
+ }
492
+
493
+ return env.Undefined();
494
+ }
495
+
496
+ static Napi::Value PtyClear(const Napi::CallbackInfo& info) {
497
+ Napi::Env env(info.Env());
498
+ Napi::HandleScope scope(env);
499
+
500
+ if (info.Length() != 2 ||
501
+ !info[0].IsNumber() ||
502
+ !info[1].IsBoolean()) {
503
+ throw Napi::Error::New(env, "Usage: pty.clear(id, useConptyDll)");
504
+ }
505
+
506
+ int id = info[0].As<Napi::Number>().Int32Value();
507
+ const bool useConptyDll = info[1].As<Napi::Boolean>().Value();
508
+
509
+ // This API is only supported for conpty.dll as it was introduced in a later version of Windows.
510
+ // We could hook it up to point at >= a version of Windows only, but the future is conpty.dll
511
+ // anyway.
512
+ if (!useConptyDll) {
513
+ return env.Undefined();
514
+ }
515
+
516
+ const pty_baton* handle = get_pty_baton(id);
517
+
518
+ if (handle != nullptr) {
519
+ HANDLE hLibrary = LoadConptyDll(info, useConptyDll);
520
+ bool fLoadedDll = hLibrary != nullptr;
521
+ if (fLoadedDll)
522
+ {
523
+ PFNCLEARPSEUDOCONSOLE const pfnClearPseudoConsole = (PFNCLEARPSEUDOCONSOLE)GetProcAddress((HMODULE)hLibrary, "ConptyClearPseudoConsole");
524
+ if (pfnClearPseudoConsole)
525
+ {
526
+ pfnClearPseudoConsole(handle->hpc);
527
+ }
528
+ }
529
+ }
530
+
531
+ return env.Undefined();
532
+ }
533
+
534
+ static Napi::Value PtyKill(const Napi::CallbackInfo& info) {
535
+ Napi::Env env(info.Env());
536
+ Napi::HandleScope scope(env);
537
+
538
+ if (info.Length() != 2 ||
539
+ !info[0].IsNumber() ||
540
+ !info[1].IsBoolean()) {
541
+ throw Napi::Error::New(env, "Usage: pty.kill(id, useConptyDll)");
542
+ }
543
+
544
+ int id = info[0].As<Napi::Number>().Int32Value();
545
+ const bool useConptyDll = info[1].As<Napi::Boolean>().Value();
546
+
547
+ const pty_baton* handle = get_pty_baton(id);
548
+
549
+ if (handle != nullptr) {
550
+ HANDLE hLibrary = LoadConptyDll(info, useConptyDll);
551
+ bool fLoadedDll = hLibrary != nullptr;
552
+ if (fLoadedDll)
553
+ {
554
+ PFNCLOSEPSEUDOCONSOLE const pfnClosePseudoConsole = (PFNCLOSEPSEUDOCONSOLE)GetProcAddress(
555
+ (HMODULE)hLibrary,
556
+ useConptyDll ? "ConptyClosePseudoConsole" : "ClosePseudoConsole");
557
+ if (pfnClosePseudoConsole)
558
+ {
559
+ pfnClosePseudoConsole(handle->hpc);
560
+ }
561
+ }
562
+ if (useConptyDll) {
563
+ TerminateProcess(handle->hShell, 1);
564
+ }
565
+ }
566
+
567
+ return env.Undefined();
568
+ }
569
+
570
+ /**
571
+ * Init
572
+ */
573
+
574
+ Napi::Object init(Napi::Env env, Napi::Object exports) {
575
+ exports.Set("startProcess", Napi::Function::New(env, PtyStartProcess));
576
+ exports.Set("connect", Napi::Function::New(env, PtyConnect));
577
+ exports.Set("resize", Napi::Function::New(env, PtyResize));
578
+ exports.Set("clear", Napi::Function::New(env, PtyClear));
579
+ exports.Set("kill", Napi::Function::New(env, PtyKill));
580
+ return exports;
581
+ };
582
+
583
+ NODE_API_MODULE(NODE_GYP_MODULE_NAME, init);
@@ -0,0 +1,41 @@
1
+ // Copyright (c) Microsoft Corporation.
2
+ // Licensed under the MIT license.
3
+
4
+ // This header prototypes the Pseudoconsole symbols from conpty.lib with their original names.
5
+ // This is required because we cannot import __imp_CreatePseudoConsole from a static library
6
+ // as it doesn't produce an import lib.
7
+ // We can't use an /ALTERNATENAME trick because it seems that that name is only resolved when the
8
+ // linker cannot otherwise find the symbol.
9
+
10
+ #pragma once
11
+
12
+ #include <consoleapi.h>
13
+
14
+ #ifndef CONPTY_IMPEXP
15
+ #define CONPTY_IMPEXP __declspec(dllimport)
16
+ #endif
17
+
18
+ #ifndef CONPTY_EXPORT
19
+ #ifdef __cplusplus
20
+ #define CONPTY_EXPORT extern "C" CONPTY_IMPEXP
21
+ #else
22
+ #define CONPTY_EXPORT extern CONPTY_IMPEXP
23
+ #endif
24
+ #endif
25
+
26
+ #define PSEUDOCONSOLE_RESIZE_QUIRK (2u)
27
+ #define PSEUDOCONSOLE_PASSTHROUGH_MODE (8u)
28
+
29
+ CONPTY_EXPORT HRESULT WINAPI ConptyCreatePseudoConsole(COORD size, HANDLE hInput, HANDLE hOutput, DWORD dwFlags, HPCON* phPC);
30
+ CONPTY_EXPORT HRESULT WINAPI ConptyCreatePseudoConsoleAsUser(HANDLE hToken, COORD size, HANDLE hInput, HANDLE hOutput, DWORD dwFlags, HPCON* phPC);
31
+
32
+ CONPTY_EXPORT HRESULT WINAPI ConptyResizePseudoConsole(HPCON hPC, COORD size);
33
+ CONPTY_EXPORT HRESULT WINAPI ConptyClearPseudoConsole(HPCON hPC);
34
+ CONPTY_EXPORT HRESULT WINAPI ConptyShowHidePseudoConsole(HPCON hPC, bool show);
35
+ CONPTY_EXPORT HRESULT WINAPI ConptyReparentPseudoConsole(HPCON hPC, HWND newParent);
36
+ CONPTY_EXPORT HRESULT WINAPI ConptyReleasePseudoConsole(HPCON hPC);
37
+
38
+ CONPTY_EXPORT VOID WINAPI ConptyClosePseudoConsole(HPCON hPC);
39
+ CONPTY_EXPORT VOID WINAPI ConptyClosePseudoConsoleTimeout(HPCON hPC, DWORD dwMilliseconds);
40
+
41
+ CONPTY_EXPORT HRESULT WINAPI ConptyPackPseudoConsole(HANDLE hServerProcess, HANDLE hRef, HANDLE hSignal, HPCON* phPC);
@@ -0,0 +1,44 @@
1
+ /**
2
+ * Copyright (c) 2019, Microsoft Corporation (MIT License).
3
+ */
4
+
5
+ #define NODE_ADDON_API_DISABLE_DEPRECATED
6
+ #include <napi.h>
7
+ #include <windows.h>
8
+
9
+ static Napi::Value ApiConsoleProcessList(const Napi::CallbackInfo& info) {
10
+ Napi::Env env(info.Env());
11
+ if (info.Length() != 1 ||
12
+ !info[0].IsNumber()) {
13
+ throw Napi::Error::New(env, "Usage: getConsoleProcessList(shellPid)");
14
+ }
15
+
16
+ const DWORD pid = info[0].As<Napi::Number>().Uint32Value();
17
+
18
+ if (!FreeConsole()) {
19
+ throw Napi::Error::New(env, "FreeConsole failed");
20
+ }
21
+ if (!AttachConsole(pid)) {
22
+ throw Napi::Error::New(env, "AttachConsole failed");
23
+ }
24
+ auto processList = std::vector<DWORD>(64);
25
+ auto processCount = GetConsoleProcessList(&processList[0], static_cast<DWORD>(processList.size()));
26
+ if (processList.size() < processCount) {
27
+ processList.resize(processCount);
28
+ processCount = GetConsoleProcessList(&processList[0], static_cast<DWORD>(processList.size()));
29
+ }
30
+ FreeConsole();
31
+
32
+ Napi::Array result = Napi::Array::New(env);
33
+ for (DWORD i = 0; i < processCount; i++) {
34
+ result.Set(i, Napi::Number::New(env, processList[i]));
35
+ }
36
+ return result;
37
+ }
38
+
39
+ Napi::Object init(Napi::Env env, Napi::Object exports) {
40
+ exports.Set("getConsoleProcessList", Napi::Function::New(env, ApiConsoleProcessList));
41
+ return exports;
42
+ };
43
+
44
+ NODE_API_MODULE(NODE_GYP_MODULE_NAME, init);