Node Single executable applications
Read about it here.
Warning
The single executable application feature currently only supports running a single embedded script using theĀ CommonJSĀ module system.
Meaning:
const mqtt = require("mqtt");
is not going to work an will give you the following error:node:internal/util/embedding:48 throw new ERR_UNKNOWN_BUILTIN_MODULE(id); ^ Error [ERR_UNKNOWN_BUILTIN_MODULE]: No such built-in module: mqtt at embedderRequire (node:internal/util/embedding:48:11) at testing/mqttServer.js:1:14 at embedderRunCjs (node:internal/util/embedding:37:10) at node:internal/main/embedding:18:8 { code: 'ERR_UNKNOWN_BUILTIN_MODULE' }
So this will work on single file apps, and maybe in the future.
steps
Okay I chose to try this on testing/mqttServer.js
.
These are just the same steps as in the node.js article
- Create a configuration file building a blob, this is the command to do it via cmd
echo '{ "main": "testing/mqttServer.js", "output": "sea-prep.blob" }' > sea-config.json
node --experimental-sea-config sea-config.json
, might cause this error.- Create a copy of theĀ
node
Ā executable and name it according to your needs:- on windows:
node -e "require('fs').copyFileSync(process.execPath, 'hello.exe')"
- on anything else:
cp $(command -v node) hello
- on windows:
- Remove the signature of the binary (macOS and Windows only):
- On macOS:
codesign --remove-signature hello
- On Windows (optional):
- signtoolĀ can be used from the installedĀ Windows SDK. If this step is skipped, ignore any signature-related warning from postject.
signtool remove /s hello.exe
- On macOS:
- Inject the blob into the copied binary by runningĀ
postject
Ā with the following options:hello
Ā /Āhello.exe
Ā - The name of the copy of theĀnode
Ā executable created in step 4.NODE_SEA_BLOB
Ā - The name of the resource / note / section in the binary where the contents of the blob will be stored.sea-prep.blob
Ā - The name of the blob created in step 1.--sentinel-fuse NODE_SEA_FUSE_fce680ab2cc467b6e072b8b5df1996b2
Ā - TheĀ fuseĀ used by the Node.js project to detect if a file has been injected.--macho-segment-name NODE_SEA
Ā (only needed on macOS) - The n- ame of the segment in the binary where the contents of the blob will be stored. To summarize, here is the required command for each platform:
- Linux:
npx postject hello NODE_SEA_BLOB sea-prep.blob \
--sentinel-fuse NODE_SEA_FUSE_fce680ab2cc467b6e072b8b5df1996b2
- On Windows - PowerShell:
npx postject hello.exe NODE_SEA_BLOB sea-prep.blob `
--sentinel-fuse NODE_SEA_FUSE_fce680ab2cc467b6e072b8b5df1996b2
- On Windows - Command Prompt:
npx postject hello.exe NODE_SEA_BLOB sea-prep.blob ^
--sentinel-fuse NODE_SEA_FUSE_fce680ab2cc467b6e072b8b5df1996b2
- on macOS:
npx postject hello NODE_SEA_BLOB sea-prep.blob \
--sentinel-fuse NODE_SEA_FUSE_fce680ab2cc467b6e072b8b5df1996b2 \
--macho-segment-name NODE_SEA
- Sign the binary (macOS and Windows only):
- On macOS:
codesign --sign - hello
- On Windows (optional):
- A certificate needs to be present for this to work. However, the unsigned binary would still be runnable.
signtool sign /fd SHA256 hello.exe
- On macOS:
- Run the binary (exe):
- Anything that is not windows:
./hello
- Windows:
.\hello.exe
- Anything that is not windows:
Congratulations, you have yourself an .exe node application.
errors
node āexperimental-sea-config sea-config.json returns:
SyntaxError: Unexpected token 'ļæ½', "ļæ½ļæ½{
"... is not valid JSON
Cannot parse JSON from sea-config.json
This explains it and helps you fix it. (atleast the vscode steps worked for me)
nexe
nexe/nexe: š create a single executable out of your node.js apps is the only package I could find that is not deprecated. - (05/06/2024)
Spoiler alert: I couldnāt get it to work.
npm i nexe -g
and Iāll be trying to build with the following command:
nexe testing/mqttServer.js --build --verbose
--verbose
is so that I can better see error messages when something goes wrong.
I also ran the following powershell because I thought I needed to do that (pretty sure I did need it, but canāt confirm):
Set-ExecutionPolicy Unrestricted -Force
iex ((New-Object System.Net.WebClient).DownloadString('https://boxstarter.org/bootstrapper.ps1'))
get-boxstarter -Force
Install-BoxstarterPackage https://raw.githubusercontent.com/nodejs/node/HEAD/tools/bootstrap/windows_boxstarter -DisableReboots
refreshenv
I did ignore an error though with the Set-ExecutionPolicy Unrestricted -Force
.
Error: vcbuild.bat nosign release x64 exited with code: 1
This is the error I was getting, and it had multiple stages, which I was able to find out because of the --verbose
Could not find NASM.
Looking for NASM
Could not find NASM, install it or build with openssl-no-asm. See BUILDING.md.
ā Compiling Node with arguments: nosign,release,x64
ā Finished in 1.117s
Error: vcbuild.bat nosign release x64 exited with code: 1
I got this error, so I googled and ended on this thread where someone told me to go install it from here.
After I ran the installer (it took a bit to show up, so I thought it didnāt work lol), I got the next error:
Failed to find a suitable Visual Studio installation.
Looking for NASM
Looking for Visual Studio 2022
Failed to find a suitable Visual Studio installation.
Try to run in a "Developer Command Prompt" or consult
https://github.com/nodejs/node/blob/HEAD/BUILDING.md#windows
ā Compiling Node with arguments: nosign,release,x64
ā Finished in 1.054s
Error: vcbuild.bat nosign release x64 exited with code: 1
So I went to the BUILDING.md where it tells me to get The .NET SDK component fromĀ Visual Studio 2022, which will be installed with the installer.
So I did that, the Community edition. This feels redundant AF.
I also signed in, for good measure, then I launched it. To try and find a Developer Command Prompt
to give me extra luck on this EPIC Journey.
Okay so I opened the folder, went under the view and selected terminal and rand the nexe testing/mqttServer.js --build --verbose
command again.
Looking for NASM
Looking for Visual Studio 2022
Failed to find a suitable Visual Studio installation.
Try to run in a "Developer Command Prompt" or consult
https://github.com/nodejs/node/blob/HEAD/BUILDING.md#windows
ā Compiling Node with arguments: nosign,release,x64
ā Finished in 1.176s
Error: vcbuild.bat nosign release x64 exited with code: 1
Well shit.
Letās try installing Visual Studio 2022 Build Tools
I already have Chocolatey installed, so I just ran the following command in powershell (with administrator):
choco install visualstudio2022buildtools
and replied Y
to all questions. (close visual studio + installer first)
If you donāt have chocolatey yet, I think this was the install command in powershell:
@"%SystemRoot%\System32\WindowsPowerShell\v1.0\powershell.exe" -NoProfile -InputFormat None -ExecutionPolicy Bypass -Command "[System.Net.ServicePointManager]::SecurityProtocol = 3072; iex ((New-Object System.Net.WebClient).DownloadString('https://community.chocolatey.org/install.ps1'))" && SET "PATH=%PATH%;%ALLUSERSPROFILE%\chocolatey\bin"
(copied from 20.00 MEAN tutorial)
Then I relaunched visual studio to try again. >> still the same error.
Letās try Visual C++ workload:
choco install visualstudio2022-workload-vctools
in powershell with administrator.
Same error.
Then it occurred to me, maybe I should restart my pc.
The console is getting spammedā¦
After waiting 10 minutes, with a few warnings in between the random spam of <random word that i assume is a filename>.cc
I end up with this:
C:\Users\Gebruiker\.nexe\21.6.0\deps\v8\src\diagnostics\x64\disasm-x64.cc(2453,20): warning C4312: 'reinterpret_cast': conversion from 'uint32_t' to 'uint8_t *' of greater size [C:\Users\Gebruiker\.nexe\21.6.0\tools\v8_gypfi
les\v8_base_without_compiler.vcxproj]
C:\Users\Gebruiker\.nexe\21.6.0\deps\v8\src\diagnostics\x64\disasm-x64.cc(2699,31): warning C4312: 'reinterpret_cast': conversion from 'int32_t' to 'uint8_t *' of greater size [C:\Users\Gebruiker\.nexe\21.6.0\tools\v8_gypfil
es\v8_base_without_compiler.vcxproj]
ā Compiling Node with arguments: nosign,release,x64
ā Finished in 643.727s
Error: vcbuild.bat nosign release x64 exited with code: 1
docker
this would be a good alternative.