mirror of
https://github.com/balena-io/balena-cli.git
synced 2025-06-24 18:45:07 +00:00
Compare commits
279 Commits
Author | SHA1 | Date | |
---|---|---|---|
3f9288e9d3 | |||
0efa745628 | |||
bddad252f7 | |||
a1a0e4f028 | |||
de74baa2ff | |||
6c12f755c5 | |||
f80b8e63b1 | |||
b32514f5af | |||
935f8d2549 | |||
b2de857ef1 | |||
78f1471bf4 | |||
d47abf072d | |||
8502c4db4b | |||
dd2c5c40d7 | |||
d23b253ac5 | |||
0b0e24c9b2 | |||
f9656cbe91 | |||
e174f7db4c | |||
a7a408a5c7 | |||
e5877c7de9 | |||
ecb8b3ae6b | |||
fd20516f69 | |||
5ccee0e4f1 | |||
7bb13a551c | |||
19d287aefc | |||
8d10c1af2a | |||
ebfabdba6a | |||
b0cbe43708 | |||
b7f1469912 | |||
3396ba5a97 | |||
78ffff83bc | |||
ae13d584a3 | |||
14f12d17eb | |||
45eb0ad4b1 | |||
21fd8a3307 | |||
b545bd00ad | |||
3eda7938f9 | |||
2bccabfc38 | |||
1b42d08567 | |||
55b69be987 | |||
7c8ce1b1a9 | |||
31ddacec4c | |||
dad26328b9 | |||
98197eef47 | |||
0f8ce47ec6 | |||
33d3be326a | |||
3eb397de1b | |||
457b81a597 | |||
8eb1777437 | |||
bbc08dcfc5 | |||
48cee061f4 | |||
37e96e5d67 | |||
a5c865b7f9 | |||
3fb3dd5819 | |||
daf5c518fb | |||
4fcedd0607 | |||
42d9cbb48d | |||
408efa91c1 | |||
a2209ffe56 | |||
3f27db811b | |||
839a3050fb | |||
c8ea9cfcdb | |||
776115ef5d | |||
f031ec1dea | |||
fe42438090 | |||
b616fbdd79 | |||
81edfbbae1 | |||
663e83c3b8 | |||
b650f8ff6d | |||
58234f17e1 | |||
77905f4a74 | |||
30076fabe6 | |||
28703bb5ae | |||
37b3c6abe9 | |||
b4e473e4d4 | |||
0d4e411777 | |||
7e6f2189e8 | |||
3903daf8a8 | |||
18bc0d61e7 | |||
7f2daeebb0 | |||
813e9cb82e | |||
3bcb3c1b2e | |||
20d76556c2 | |||
e829068725 | |||
650e896f70 | |||
a9042124ea | |||
d24d78dac7 | |||
42c50ef8ae | |||
ba4b9bd447 | |||
02c0ea5b59 | |||
bc3558dd8e | |||
aad62d1ccd | |||
ecc6f80164 | |||
c0fd1e3886 | |||
9d3120b144 | |||
ed0e03ddb2 | |||
8fe6d6c026 | |||
727033ae14 | |||
c19ce6a905 | |||
1a33029738 | |||
043bc48a1c | |||
a10156a441 | |||
4f665f43d2 | |||
9f097a96f5 | |||
64d1943804 | |||
666ce876e6 | |||
e01184080f | |||
93039b010d | |||
795259bf30 | |||
fa134d2d39 | |||
bef5221ed8 | |||
72d6db796c | |||
e848eb63ee | |||
6f0f7350cf | |||
07a88c700e | |||
9cae66bd92 | |||
cddea24cef | |||
b1c246c0b4 | |||
00b4d57a03 | |||
2cba82e914 | |||
1352c5c823 | |||
c86eb97010 | |||
53be743b9d | |||
d9f21b4c3f | |||
261ab398dd | |||
f28a9992e4 | |||
29e7827eb1 | |||
1d77cf3665 | |||
017c767f61 | |||
7d79c4e24b | |||
60bc5092e0 | |||
a33a794931 | |||
f0ede6fca2 | |||
dbe177e766 | |||
09f80730a8 | |||
327d28c103 | |||
56ab785a82 | |||
305d65d5ed | |||
c4d3686a34 | |||
ce06854b55 | |||
8db05cc8a7 | |||
7a22c987d2 | |||
45efbcdfe3 | |||
d6a9b78b3e | |||
e8ac3ea960 | |||
0ffa0f85a2 | |||
5e7479f60e | |||
07365c45f2 | |||
e5076434c6 | |||
5d687f5a55 | |||
e192767156 | |||
5a8d2fad5f | |||
45f482fad1 | |||
c0e7ae9c91 | |||
36077cacda | |||
82b9983450 | |||
703dbd01c9 | |||
602e63c8a9 | |||
2ab635f49a | |||
322736a145 | |||
c347b67b25 | |||
4022beeb56 | |||
ccf97cfc9f | |||
9c5fe14f2e | |||
38e29251e7 | |||
bfc7a14646 | |||
610db81fcb | |||
d1f7d6d07f | |||
694eb78aaa | |||
1caccafbcd | |||
61d4d1f1e7 | |||
a01c85bc15 | |||
5d7b7cfc6f | |||
92fd9e0883 | |||
24273b5ac0 | |||
6155509f4c | |||
735af9f6a9 | |||
d7c60e6dea | |||
bcb42c8a21 | |||
04f5e0fa2b | |||
8cb5804848 | |||
91c3fced49 | |||
99a94eafbb | |||
b4cff78588 | |||
8577bb6281 | |||
e0f081623b | |||
2887ab8200 | |||
aaf4625abb | |||
6f30dc0550 | |||
6565ef4392 | |||
60b46192a2 | |||
bb101de96e | |||
ca6344bf4c | |||
48596fa318 | |||
7fb28ddb21 | |||
6dd6f43d64 | |||
07a7bd76fe | |||
4b3fdcf99c | |||
d023d0af91 | |||
de1821d7ac | |||
12923c9b84 | |||
8be069dbdb | |||
9d3f9128a8 | |||
61ebf9e4fd | |||
84985022e5 | |||
c5d8f73263 | |||
5db0c71bb3 | |||
c7a06f7259 | |||
bc66febc50 | |||
bb80311700 | |||
3dee7bd6f6 | |||
3251f04287 | |||
35dce4579a | |||
71ef00534d | |||
b6f8be27ec | |||
13110cca45 | |||
d4b554da1b | |||
1275c11573 | |||
593233a99f | |||
ec92f21b70 | |||
5adc43bcbd | |||
1ee9a68288 | |||
56e5dafb20 | |||
f0e0c0d728 | |||
afd14794f5 | |||
37e08e4667 | |||
61af57acc9 | |||
19be0fec1a | |||
2b656c23b3 | |||
2bf7b81645 | |||
746b6fa439 | |||
f52e218290 | |||
8655b89313 | |||
09d52c504d | |||
e5cee648f2 | |||
98a6b431d9 | |||
f305d5d9a0 | |||
5cb5ac80a6 | |||
3eb3b3b584 | |||
8ee5ede34d | |||
d6d08cc7c9 | |||
76c63d4b20 | |||
bff5897047 | |||
22c9fd399e | |||
65222b5fc9 | |||
d554658eee | |||
8ded517dd9 | |||
b0d8b021d5 | |||
2eb5d7f6b3 | |||
f1924bba6b | |||
44082e22e4 | |||
2d15530f61 | |||
a7c612f7de | |||
147ce8067a | |||
5e7ac09dd0 | |||
4c9fc89a6b | |||
9b1eb57973 | |||
76c08b6c13 | |||
3f82f42652 | |||
69d820878a | |||
aaaee47e0c | |||
e7761a616b | |||
9559d5cba3 | |||
c1649dd828 | |||
8e9b992a59 | |||
b77f266bd7 | |||
3c9ac76982 | |||
e483d06d2b | |||
24d2d19d33 | |||
ba5bb7b12c | |||
9082e7b3f7 | |||
0716544042 | |||
e1858aa69d | |||
082cce332a | |||
218e0a1b6b | |||
4065c5775c | |||
fd966df1f0 | |||
5eba175bf1 | |||
bd1b71bf2f |
@ -1,2 +0,0 @@
|
|||||||
/completion/*
|
|
||||||
/bin/*
|
|
21
.eslintrc.js
21
.eslintrc.js
@ -1,21 +0,0 @@
|
|||||||
module.exports = {
|
|
||||||
extends: ['./node_modules/@balena/lint/config/.eslintrc.js'],
|
|
||||||
parserOptions: {
|
|
||||||
project: 'tsconfig.dev.json',
|
|
||||||
},
|
|
||||||
root: true,
|
|
||||||
rules: {
|
|
||||||
ignoreDefinitionFiles: 0,
|
|
||||||
// to avoid the `warning Forbidden non-null assertion @typescript-eslint/no-non-null-assertion`
|
|
||||||
'@typescript-eslint/no-non-null-assertion': 'off',
|
|
||||||
'@typescript-eslint/no-shadow': 'off',
|
|
||||||
'@typescript-eslint/no-var-requires': 'off',
|
|
||||||
'no-restricted-imports': [
|
|
||||||
'error',
|
|
||||||
{
|
|
||||||
paths: ['resin-cli-visuals', 'chalk', 'common-tags', 'resin-cli-form'],
|
|
||||||
},
|
|
||||||
],
|
|
||||||
'@typescript-eslint/no-unused-vars': ['error', { argsIgnorePattern: '^_' }],
|
|
||||||
},
|
|
||||||
};
|
|
2
.github/ISSUE_TEMPLATE.md
vendored
2
.github/ISSUE_TEMPLATE.md
vendored
@ -67,7 +67,7 @@ fixed it.
|
|||||||
- **Cloud backend: openBalena or balenaCloud?** If unsure, it will be balenaCloud
|
- **Cloud backend: openBalena or balenaCloud?** If unsure, it will be balenaCloud
|
||||||
- **Operating system version:** e.g. Windows 10, Ubuntu 18.04, macOS 10.14.5
|
- **Operating system version:** e.g. Windows 10, Ubuntu 18.04, macOS 10.14.5
|
||||||
- **32/64 bit OS and processor:** e.g. 32-bit Windows on 64-bit Intel processor
|
- **32/64 bit OS and processor:** e.g. 32-bit Windows on 64-bit Intel processor
|
||||||
- **Install method:** npm or zip package or executable installer
|
- **Install method:** npm or standalone package or executable installer
|
||||||
- **If npm install, Node.js and npm version:** e.g. Node v8.16.0 and npm v6.4.1
|
- **If npm install, Node.js and npm version:** e.g. Node v8.16.0 and npm v6.4.1
|
||||||
|
|
||||||
# Additional References
|
# Additional References
|
||||||
|
18
.github/actions/publish/action.yml
vendored
18
.github/actions/publish/action.yml
vendored
@ -18,7 +18,7 @@ inputs:
|
|||||||
default: 'accounts+apple@balena.io'
|
default: 'accounts+apple@balena.io'
|
||||||
NODE_VERSION:
|
NODE_VERSION:
|
||||||
type: string
|
type: string
|
||||||
default: '20.x'
|
default: '22.x'
|
||||||
VERBOSE:
|
VERBOSE:
|
||||||
type: string
|
type: string
|
||||||
default: 'true'
|
default: 'true'
|
||||||
@ -28,7 +28,7 @@ runs:
|
|||||||
using: 'composite'
|
using: 'composite'
|
||||||
steps:
|
steps:
|
||||||
- name: Download custom source artifact
|
- name: Download custom source artifact
|
||||||
uses: actions/download-artifact@fa0a91b85d4f404e444e00e005971372dc801d16 # v4.1.8
|
uses: actions/download-artifact@cc203385981b70ca67e1cc392babf9cc229d5806 # v4.1.9
|
||||||
with:
|
with:
|
||||||
name: custom-${{ github.event.pull_request.head.sha || github.event.head_commit.id }}-${{ runner.os }}-${{ runner.arch }}
|
name: custom-${{ github.event.pull_request.head.sha || github.event.head_commit.id }}-${{ runner.os }}-${{ runner.arch }}
|
||||||
path: ${{ runner.temp }}
|
path: ${{ runner.temp }}
|
||||||
@ -39,7 +39,7 @@ runs:
|
|||||||
run: tar -xf ${{ runner.temp }}/custom.tgz
|
run: tar -xf ${{ runner.temp }}/custom.tgz
|
||||||
|
|
||||||
- name: Setup Node.js
|
- name: Setup Node.js
|
||||||
uses: actions/setup-node@1e60f620b9541d16bece96c5465dc8ee9832be0b # v4
|
uses: actions/setup-node@cdca7365b2dadb8aad0a33bc7601856ffabcc48e # v4
|
||||||
with:
|
with:
|
||||||
node-version: ${{ inputs.NODE_VERSION }}
|
node-version: ${{ inputs.NODE_VERSION }}
|
||||||
cache: npm
|
cache: npm
|
||||||
@ -48,7 +48,7 @@ runs:
|
|||||||
if: runner.os == 'macOS'
|
if: runner.os == 'macOS'
|
||||||
uses: actions/setup-python@65d7f2d534ac1bc67fcd62888c5f4f3d2cb2b236 # v4
|
uses: actions/setup-python@65d7f2d534ac1bc67fcd62888c5f4f3d2cb2b236 # v4
|
||||||
with:
|
with:
|
||||||
python-version: "3.11"
|
python-version: '3.11'
|
||||||
|
|
||||||
- name: Install additional tools
|
- name: Install additional tools
|
||||||
if: runner.os == 'Windows'
|
if: runner.os == 'Windows'
|
||||||
@ -94,7 +94,7 @@ runs:
|
|||||||
runner_arch="$(echo "${RUNNER_ARCH}" | tr '[:upper:]' '[:lower:]')"
|
runner_arch="$(echo "${RUNNER_ARCH}" | tr '[:upper:]' '[:lower:]')"
|
||||||
|
|
||||||
if [[ $runner_os =~ darwin|macos|osx ]]; then
|
if [[ $runner_os =~ darwin|macos|osx ]]; then
|
||||||
CSC_KEY_PASSWORD=${{ fromJSON(inputs.secrets).APPLE_SIGNING_PASSWORD }}
|
CSC_KEY_PASSWORD='${{ fromJSON(inputs.secrets).APPLE_SIGNING_PASSWORD }}'
|
||||||
CSC_KEYCHAIN=signing_temp
|
CSC_KEYCHAIN=signing_temp
|
||||||
CSC_LINK=${{ fromJSON(inputs.secrets).APPLE_SIGNING }}
|
CSC_LINK=${{ fromJSON(inputs.secrets).APPLE_SIGNING }}
|
||||||
|
|
||||||
@ -112,8 +112,8 @@ runs:
|
|||||||
PATH="/c/Program Files/DigiCert/DigiCert One Signing Manager Tools:${PATH}"
|
PATH="/c/Program Files/DigiCert/DigiCert One Signing Manager Tools:${PATH}"
|
||||||
smksp_registrar.exe list
|
smksp_registrar.exe list
|
||||||
smctl.exe keypair ls
|
smctl.exe keypair ls
|
||||||
|
smctl.exe windows certsync
|
||||||
/c/Windows/System32/certutil.exe -csp "DigiCert Signing Manager KSP" -key -user
|
/c/Windows/System32/certutil.exe -csp "DigiCert Signing Manager KSP" -key -user
|
||||||
smksp_cert_sync.exe
|
|
||||||
|
|
||||||
# (signtool.exe) https://github.com/actions/runner-images/blob/main/images/win/Windows2019-Readme.md#installed-windows-sdks
|
# (signtool.exe) https://github.com/actions/runner-images/blob/main/images/win/Windows2019-Readme.md#installed-windows-sdks
|
||||||
PATH="/c/Program Files (x86)/Windows Kits/10/bin/${runner_arch}:${PATH}"
|
PATH="/c/Program Files (x86)/Windows Kits/10/bin/${runner_arch}:${PATH}"
|
||||||
@ -135,9 +135,11 @@ runs:
|
|||||||
XCODE_APP_LOADER_TEAM_ID: ${{ inputs.XCODE_APP_LOADER_TEAM_ID }}
|
XCODE_APP_LOADER_TEAM_ID: ${{ inputs.XCODE_APP_LOADER_TEAM_ID }}
|
||||||
|
|
||||||
- name: Upload artifacts
|
- name: Upload artifacts
|
||||||
uses: actions/upload-artifact@50769540e7f4bd5e21e526ee35c689e35e0d6874 # v4
|
uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4
|
||||||
with:
|
with:
|
||||||
name: gh-release-${{ github.event.pull_request.head.sha || github.event.head_commit.id }}-${{ strategy.job-index }}
|
name: gh-release-${{ github.event.pull_request.head.sha || github.event.head_commit.id }}-${{ strategy.job-index }}
|
||||||
path: dist
|
path: |
|
||||||
|
dist
|
||||||
|
!dist/balena
|
||||||
retention-days: 1
|
retention-days: 1
|
||||||
if-no-files-found: error
|
if-no-files-found: error
|
||||||
|
6
.github/actions/test/action.yml
vendored
6
.github/actions/test/action.yml
vendored
@ -15,7 +15,7 @@ inputs:
|
|||||||
# --- custom environment
|
# --- custom environment
|
||||||
NODE_VERSION:
|
NODE_VERSION:
|
||||||
type: string
|
type: string
|
||||||
default: '20.x'
|
default: '22.x'
|
||||||
VERBOSE:
|
VERBOSE:
|
||||||
type: string
|
type: string
|
||||||
default: "true"
|
default: "true"
|
||||||
@ -26,7 +26,7 @@ runs:
|
|||||||
steps:
|
steps:
|
||||||
# https://github.com/actions/setup-node#caching-global-packages-data
|
# https://github.com/actions/setup-node#caching-global-packages-data
|
||||||
- name: Setup Node.js
|
- name: Setup Node.js
|
||||||
uses: actions/setup-node@1e60f620b9541d16bece96c5465dc8ee9832be0b # v4
|
uses: actions/setup-node@cdca7365b2dadb8aad0a33bc7601856ffabcc48e # v4
|
||||||
with:
|
with:
|
||||||
node-version: ${{ inputs.NODE_VERSION }}
|
node-version: ${{ inputs.NODE_VERSION }}
|
||||||
cache: npm
|
cache: npm
|
||||||
@ -58,7 +58,7 @@ runs:
|
|||||||
run: tar --exclude-vcs -acf ${{ runner.temp }}/custom.tgz .
|
run: tar --exclude-vcs -acf ${{ runner.temp }}/custom.tgz .
|
||||||
|
|
||||||
- name: Upload custom artifact
|
- name: Upload custom artifact
|
||||||
uses: actions/upload-artifact@50769540e7f4bd5e21e526ee35c689e35e0d6874 # v4
|
uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4
|
||||||
with:
|
with:
|
||||||
name: custom-${{ github.event.pull_request.head.sha || github.event.head_commit.id }}-${{ runner.os }}-${{ runner.arch }}
|
name: custom-${{ github.event.pull_request.head.sha || github.event.head_commit.id }}-${{ runner.os }}-${{ runner.arch }}
|
||||||
path: ${{ runner.temp }}/custom.tgz
|
path: ${{ runner.temp }}/custom.tgz
|
||||||
|
4
.github/workflows/flowzone.yml
vendored
4
.github/workflows/flowzone.yml
vendored
@ -26,7 +26,7 @@ jobs:
|
|||||||
"os": [
|
"os": [
|
||||||
["self-hosted", "X64"],
|
["self-hosted", "X64"],
|
||||||
["self-hosted", "ARM64"],
|
["self-hosted", "ARM64"],
|
||||||
["macos-12"],
|
["macos-13"],
|
||||||
["windows-2019"],
|
["windows-2019"],
|
||||||
["macos-latest-xlarge"]
|
["macos-latest-xlarge"]
|
||||||
]
|
]
|
||||||
@ -36,7 +36,7 @@ jobs:
|
|||||||
"os": [
|
"os": [
|
||||||
["self-hosted", "X64"],
|
["self-hosted", "X64"],
|
||||||
["self-hosted", "ARM64"],
|
["self-hosted", "ARM64"],
|
||||||
["macos-12"],
|
["macos-13"],
|
||||||
["windows-2019"],
|
["windows-2019"],
|
||||||
["macos-latest-xlarge"]
|
["macos-latest-xlarge"]
|
||||||
]
|
]
|
||||||
|
@ -2,7 +2,7 @@ module.exports = {
|
|||||||
reporter: 'spec',
|
reporter: 'spec',
|
||||||
require: 'ts-node/register/transpile-only',
|
require: 'ts-node/register/transpile-only',
|
||||||
file: './tests/config-tests',
|
file: './tests/config-tests',
|
||||||
timeout: 12000,
|
timeout: 48000,
|
||||||
// To test only, say, 'push.spec.ts', do it as follows so that
|
// To test only, say, 'push.spec.ts', do it as follows so that
|
||||||
// requests are authenticated:
|
// requests are authenticated:
|
||||||
// spec: ['tests/auth/*.spec.ts', 'tests/**/deploy.spec.ts'],
|
// spec: ['tests/auth/*.spec.ts', 'tests/**/deploy.spec.ts'],
|
||||||
|
File diff suppressed because it is too large
Load Diff
542
CHANGELOG.md
542
CHANGELOG.md
@ -4,6 +4,548 @@ All notable changes to this project will be documented in this file
|
|||||||
automatically by Versionist. DO NOT EDIT THIS FILE MANUALLY!
|
automatically by Versionist. DO NOT EDIT THIS FILE MANUALLY!
|
||||||
This project adheres to [Semantic Versioning](http://semver.org/).
|
This project adheres to [Semantic Versioning](http://semver.org/).
|
||||||
|
|
||||||
|
## 22.1.1 - 2025-06-19
|
||||||
|
|
||||||
|
* Deploy: Limit the submitted error_message of images that fail to build to 1000 characters [Thodoris Greasidis]
|
||||||
|
|
||||||
|
## 22.1.0 - 2025-06-09
|
||||||
|
|
||||||
|
* Add support for node 22 [Otavio Jacobi]
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary> Bump etcher-sdk to v10.0.0 [Otavio Jacobi] </summary>
|
||||||
|
|
||||||
|
> ### etcher-sdk-10.0.0 - 2025-06-02
|
||||||
|
>
|
||||||
|
> * Drop support to node18 and add support to node 22 & 24 [Otavio Jacobi]
|
||||||
|
>
|
||||||
|
|
||||||
|
</details>
|
||||||
|
|
||||||
|
## 22.0.6 - 2025-06-02
|
||||||
|
|
||||||
|
* Remove `request` dependency [myarmolinsky]
|
||||||
|
* Replace `request` usage with `got` [myarmolinsky]
|
||||||
|
|
||||||
|
## 22.0.5 - 2025-05-29
|
||||||
|
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary> Bump etcher-sdk to v9.1.4 [Otavio Jacobi] </summary>
|
||||||
|
|
||||||
|
> ### etcher-sdk-9.1.4 - 2025-05-29
|
||||||
|
>
|
||||||
|
> * Run `npm audit fix` which should only do non-breaking changes [Otavio Jacobi]
|
||||||
|
>
|
||||||
|
> ### etcher-sdk-9.1.3 - 2025-02-17
|
||||||
|
>
|
||||||
|
> * Embed config.json with a fixed timestamp to enable consistent checksums [Pagan Gazzard]
|
||||||
|
>
|
||||||
|
> ### etcher-sdk-9.1.2 - 2024-10-09
|
||||||
|
>
|
||||||
|
> * Update dependency unzip-stream to v0.3.2 [SECURITY] [Self-hosted Renovate Bot]
|
||||||
|
>
|
||||||
|
> ### etcher-sdk-9.1.1 - 2024-10-09
|
||||||
|
>
|
||||||
|
> * patch: add EXLOCK flag for windows [Talha Can Havadar]
|
||||||
|
>
|
||||||
|
|
||||||
|
</details>
|
||||||
|
|
||||||
|
## 22.0.4 - 2025-05-29
|
||||||
|
|
||||||
|
* tests: Replace request with got [Otavio Jacobi]
|
||||||
|
* deploy-legacy: Replace request with got [Otavio Jacobi]
|
||||||
|
|
||||||
|
## 22.0.3 - 2025-05-29
|
||||||
|
|
||||||
|
* Bump sentry to v9 [Otavio Jacobi]
|
||||||
|
|
||||||
|
## 22.0.2 - 2025-05-28
|
||||||
|
|
||||||
|
* Fix balena build to work with --nologs [Otavio Jacobi]
|
||||||
|
|
||||||
|
## 22.0.1 - 2025-05-28
|
||||||
|
|
||||||
|
* DeviceAPI: Move away from `request` in favor of BalenaSdk request [myarmolinsky]
|
||||||
|
* Update `nock` to 14.0.4 [myarmolinsky]
|
||||||
|
|
||||||
|
## 22.0.0 - 2025-05-26
|
||||||
|
|
||||||
|
* Add migration guide to v22 [Otavio Jacobi]
|
||||||
|
* Build standalone without pkg [Otavio Jacobi]
|
||||||
|
|
||||||
|
## 21.1.14 - 2025-05-21
|
||||||
|
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary> Bump balena-preload to 18.0.4 [Otavio Jacobi] </summary>
|
||||||
|
|
||||||
|
> ### balena-preload-18.0.4 - 2025-05-21
|
||||||
|
>
|
||||||
|
> * Fix balena-preload pip deps [Otavio Jacobi]
|
||||||
|
>
|
||||||
|
|
||||||
|
</details>
|
||||||
|
|
||||||
|
## 21.1.13 - 2025-05-21
|
||||||
|
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary> Bump balena-preload to 18.0.3 [Otavio Jacobi] </summary>
|
||||||
|
|
||||||
|
> ### balena-preload-18.0.3 - 2025-03-19
|
||||||
|
>
|
||||||
|
> * Update dependency sh to v1.14.3 [balena-renovate[bot]]
|
||||||
|
>
|
||||||
|
> ### balena-preload-18.0.2 - 2025-03-19
|
||||||
|
>
|
||||||
|
> * Update alpine Docker tag to v3.21 [balena-renovate[bot]]
|
||||||
|
>
|
||||||
|
|
||||||
|
</details>
|
||||||
|
|
||||||
|
## 21.1.12 - 2025-05-16
|
||||||
|
|
||||||
|
* Update @balena/compose dependency to fix error with build secrets. [Ken Bannister]
|
||||||
|
|
||||||
|
## 21.1.11 - 2025-05-06
|
||||||
|
|
||||||
|
* Fix typos in `os configure` and `config generate` help messages [myarmolinsky]
|
||||||
|
|
||||||
|
## 21.1.10 - 2025-05-05
|
||||||
|
|
||||||
|
* patch: fix windows signing [Edwin Joassart]
|
||||||
|
|
||||||
|
## 21.1.9 - 2025-04-07
|
||||||
|
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary> Update balena-config-json to rely on the balena-image-fs helpers [Thodoris Greasidis] </summary>
|
||||||
|
|
||||||
|
> ### balena-config-json-4.2.7 - 2025-04-02
|
||||||
|
>
|
||||||
|
> * Fix getBootPartition always warning that the boot partitions were not found [Thodoris Greasidis]
|
||||||
|
>
|
||||||
|
> ### balena-config-json-4.2.6 - 2025-04-01
|
||||||
|
>
|
||||||
|
> * write: Allow undefined as a value for the deprecated type parameter [Thodoris Greasidis]
|
||||||
|
>
|
||||||
|
> <details>
|
||||||
|
> <summary> Use the balena-image-fs findPartition() helper to find the boot partition [Thodoris Greasidis] </summary>
|
||||||
|
>
|
||||||
|
>> #### balena-image-fs-7.5.0 - 2025-03-26
|
||||||
|
>>
|
||||||
|
>> * Add function to find a partition by name/label [Ken Bannister]
|
||||||
|
>>
|
||||||
|
>> #### balena-image-fs-7.4.1 - 2025-02-21
|
||||||
|
>>
|
||||||
|
>> * bump ext2fs to 4.2.4 [Ryan Cooke]
|
||||||
|
>>
|
||||||
|
>
|
||||||
|
> </details>
|
||||||
|
>
|
||||||
|
>
|
||||||
|
> ### balena-config-json-4.2.5 - 2025-03-26
|
||||||
|
>
|
||||||
|
> * Update @balena/lint to 9.1.4 [Thodoris Greasidis]
|
||||||
|
>
|
||||||
|
> ### balena-config-json-4.2.4 - 2025-03-26
|
||||||
|
>
|
||||||
|
> * Switch use ts-mocha instead of manually compiling the tests [Thodoris Greasidis]
|
||||||
|
> * Update TypeScript to 5.8.2 [Thodoris Greasidis]
|
||||||
|
>
|
||||||
|
> ### balena-config-json-4.2.3 - 2025-03-26
|
||||||
|
>
|
||||||
|
> * Update chai to v5 [balena-renovate[bot]]
|
||||||
|
>
|
||||||
|
> ### balena-image-fs-7.5.2 - 2025-04-02
|
||||||
|
>
|
||||||
|
> * Update dependency jsdoc-to-markdown to v9 [balena-renovate[bot]]
|
||||||
|
>
|
||||||
|
> ### balena-image-fs-7.5.1 - 2025-03-26
|
||||||
|
>
|
||||||
|
> * Update @balena/lint to v9 [Thodoris Greasidis]
|
||||||
|
> * Remove the no longer needed gpt detection helper [Thodoris Greasidis]
|
||||||
|
> * Update TypeScript to 5.8.2 [Thodoris Greasidis]
|
||||||
|
> * Drop the package-lock.json [Thodoris Greasidis]
|
||||||
|
>
|
||||||
|
|
||||||
|
</details>
|
||||||
|
|
||||||
|
## 21.1.8 - 2025-04-03
|
||||||
|
|
||||||
|
* Update actions/download-artifact action to v4.1.9 [balena-renovate[bot]]
|
||||||
|
|
||||||
|
## 21.1.7 - 2025-04-03
|
||||||
|
|
||||||
|
* Update actions/upload-artifact digest to ea165f8 [balena-renovate[bot]]
|
||||||
|
|
||||||
|
## 21.1.6 - 2025-04-03
|
||||||
|
|
||||||
|
* Update actions/setup-node digest to cdca736 [balena-renovate[bot]]
|
||||||
|
|
||||||
|
## 21.1.5 - 2025-04-03
|
||||||
|
|
||||||
|
* Update dockerode/docker-modem dependencies for fixes [Ken Bannister]
|
||||||
|
|
||||||
|
## 21.1.4 - 2025-04-02
|
||||||
|
|
||||||
|
* Add comment with secure boot signature file example for preload [Ken Bannister]
|
||||||
|
|
||||||
|
## 21.1.3 - 2025-03-28
|
||||||
|
|
||||||
|
* Fix device detail for open balena [Otavio Jacobi]
|
||||||
|
|
||||||
|
## 21.1.2 - 2025-03-27
|
||||||
|
|
||||||
|
* Deny preload for an image with secure boot enabled [Ken Bannister]
|
||||||
|
|
||||||
|
## 21.1.1 - 2025-03-26
|
||||||
|
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary> Bump balena-sdk to 21.3.0 [Otavio Jacobi] </summary>
|
||||||
|
|
||||||
|
> ### balena-sdk-21.3.0 - 2025-03-26
|
||||||
|
>
|
||||||
|
> * device: add `changed_api_heartbeat_state_on__date` to typings [Otavio Jacobi]
|
||||||
|
>
|
||||||
|
> ### balena-sdk-21.2.2 - 2025-03-26
|
||||||
|
>
|
||||||
|
> * fix linting [Otavio Jacobi]
|
||||||
|
>
|
||||||
|
|
||||||
|
</details>
|
||||||
|
|
||||||
|
## 21.1.0 - 2025-03-12
|
||||||
|
|
||||||
|
* Add support for new requirement labels feature [Felipe Lalanne]
|
||||||
|
|
||||||
|
## 21.0.0 - 2025-03-11
|
||||||
|
|
||||||
|
* Drop support for OS versions <2.14.0 [myarmolinsky]
|
||||||
|
* api-key generate: Add required argument `expiryDate` [myarmolinsky]
|
||||||
|
* Update `balena-preload` to 18.0.1 [myarmolinsky]
|
||||||
|
* Add dependency `date-fns` [myarmolinsky]
|
||||||
|
* Update `balena-sdk` to 21.2.1 [myarmolinsky]
|
||||||
|
|
||||||
|
## 20.2.10 - 2025-03-10
|
||||||
|
|
||||||
|
* Update TypeScript to 5.8.2 [Thodoris Greasidis]
|
||||||
|
|
||||||
|
## 20.2.9 - 2025-02-26
|
||||||
|
|
||||||
|
* Fix CORS issue with X-Balena-Client header [Thodoris Greasidis]
|
||||||
|
|
||||||
|
## 20.2.8 - 2025-02-26
|
||||||
|
|
||||||
|
* Update balena-config-json dependency and fix test [Ken Bannister]
|
||||||
|
|
||||||
|
## 20.2.7 - 2025-02-25
|
||||||
|
|
||||||
|
* Use the CLI version in the X-Balena-Client header [Thodoris Greasidis]
|
||||||
|
|
||||||
|
## 20.2.6 - 2025-02-25
|
||||||
|
|
||||||
|
* Update actions/upload-artifact digest to 4cec3d8 [balena-renovate[bot]]
|
||||||
|
|
||||||
|
## 20.2.5 - 2025-02-25
|
||||||
|
|
||||||
|
* Update actions/setup-node digest to 1d0ff46 [balena-renovate[bot]]
|
||||||
|
|
||||||
|
## 20.2.4 - 2025-02-25
|
||||||
|
|
||||||
|
* Pin docker-modem and dockerode to avoid regression [Ken Bannister]
|
||||||
|
|
||||||
|
## 20.2.3 - 2025-01-15
|
||||||
|
|
||||||
|
* Remove unused old eslint version files [Otavio Jacobi]
|
||||||
|
|
||||||
|
## 20.2.2 - 2025-01-12
|
||||||
|
|
||||||
|
* Use the promises namespace of balena-image-fs [Thodoris Greasidis]
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary> Update balena-device-init to 8.1.3 & balena-image-fs to 7.3.0 [Thodoris Greasidis] </summary>
|
||||||
|
|
||||||
|
> ### balena-image-fs-7.3.0 - 2025-01-06
|
||||||
|
>
|
||||||
|
> * Drop Bluebird from devDependencies [Thodoris Greasidis]
|
||||||
|
> * flowzone: Remove empty with clause [Thodoris Greasidis]
|
||||||
|
> * Add the promises namespace as part of the exposed fs [Thodoris Greasidis]
|
||||||
|
>
|
||||||
|
> ### balena-image-fs-7.2.2 - 2024-01-02
|
||||||
|
>
|
||||||
|
> * Update dependency @types/node to v20 [Self-hosted Renovate Bot]
|
||||||
|
>
|
||||||
|
> ### balena-image-fs-7.2.1 - 2023-12-19
|
||||||
|
>
|
||||||
|
> * Remove repo config from flowzone.yml [Kyle Harding]
|
||||||
|
>
|
||||||
|
> ### balena-image-fs-7.2.0 - 2023-01-20
|
||||||
|
>
|
||||||
|
> * Add support for Node 18 [Akis Kesoglou]
|
||||||
|
>
|
||||||
|
> ### balena-image-fs-7.1.2 - 2023-01-05
|
||||||
|
>
|
||||||
|
> * Update dependencies [ab77]
|
||||||
|
>
|
||||||
|
> ### balena-image-fs-7.1.1 - 2022-12-20
|
||||||
|
>
|
||||||
|
> * Update dependency jsdoc-to-markdown to 8.0.0 [Renovate Bot]
|
||||||
|
>
|
||||||
|
> ### balena-image-fs-7.1.0 - 2022-12-13
|
||||||
|
>
|
||||||
|
> * update dependencies [Zane Hitchcox]
|
||||||
|
>
|
||||||
|
> ### balena-device-init-8.1.3 - 2025-01-09
|
||||||
|
>
|
||||||
|
> * README: Remove the travisci badge [Thodoris Greasidis]
|
||||||
|
>
|
||||||
|
> ### balena-device-init-8.1.2 - 2025-01-09
|
||||||
|
>
|
||||||
|
>
|
||||||
|
> <details>
|
||||||
|
> <summary> Use the promises namespace of balena-image-fs [Thodoris Greasidis] </summary>
|
||||||
|
>
|
||||||
|
>> #### balena-image-fs-7.3.0 - 2025-01-06
|
||||||
|
>>
|
||||||
|
>> * Drop Bluebird from devDependencies [Thodoris Greasidis]
|
||||||
|
>> * flowzone: Remove empty with clause [Thodoris Greasidis]
|
||||||
|
>> * Add the promises namespace as part of the exposed fs [Thodoris Greasidis]
|
||||||
|
>>
|
||||||
|
>> #### balena-image-fs-7.2.2 - 2024-01-02
|
||||||
|
>>
|
||||||
|
>> * Update dependency @types/node to v20 [Self-hosted Renovate Bot]
|
||||||
|
>>
|
||||||
|
>> #### balena-image-fs-7.2.1 - 2023-12-19
|
||||||
|
>>
|
||||||
|
>> * Remove repo config from flowzone.yml [Kyle Harding]
|
||||||
|
>>
|
||||||
|
>> #### balena-image-fs-7.2.0 - 2023-01-20
|
||||||
|
>>
|
||||||
|
>> * Add support for Node 18 [Akis Kesoglou]
|
||||||
|
>>
|
||||||
|
>> #### balena-image-fs-7.1.2 - 2023-01-05
|
||||||
|
>>
|
||||||
|
>> * Update dependencies [ab77]
|
||||||
|
>>
|
||||||
|
>> #### balena-image-fs-7.1.1 - 2022-12-20
|
||||||
|
>>
|
||||||
|
>> * Update dependency jsdoc-to-markdown to 8.0.0 [Renovate Bot]
|
||||||
|
>>
|
||||||
|
>> #### balena-image-fs-7.1.0 - 2022-12-13
|
||||||
|
>>
|
||||||
|
>> * update dependencies [Zane Hitchcox]
|
||||||
|
>>
|
||||||
|
>
|
||||||
|
> </details>
|
||||||
|
>
|
||||||
|
>
|
||||||
|
> ### balena-device-init-8.1.1 - 2025-01-06
|
||||||
|
>
|
||||||
|
> * Convert some parts to async await and simplify [Thodoris Greasidis]
|
||||||
|
> * Avoid unnecessary destructuring [Thodoris Greasidis]
|
||||||
|
>
|
||||||
|
|
||||||
|
</details>
|
||||||
|
|
||||||
|
## 20.2.1 - 2025-01-01
|
||||||
|
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary> Update balena-preload to 17.0.0 [Thodoris Greasidis] </summary>
|
||||||
|
|
||||||
|
> ### balena-preload-17.0.0 - 2024-10-21
|
||||||
|
>
|
||||||
|
> * Improve typings [Thodoris Greasidis]
|
||||||
|
> * Stop returning Bluebird promises & drop it from the dependencies [Thodoris Greasidis]
|
||||||
|
>
|
||||||
|
|
||||||
|
</details>
|
||||||
|
|
||||||
|
## 20.2.0 - 2024-12-31
|
||||||
|
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary> os configure: Give precedence to the boot partition located in the image over the device-type.json contents [Thodoris Greasidis] </summary>
|
||||||
|
|
||||||
|
> ### balena-device-init-8.1.0 - Invalid date
|
||||||
|
>
|
||||||
|
> * Try to find the boot partition by inspecting the image [Thodoris Greasidis]
|
||||||
|
>
|
||||||
|
> ### balena-device-init-8.0.1 - 2024-12-19
|
||||||
|
>
|
||||||
|
> * Drop the unnecessary eslint.config.js [Thodoris Greasidis]
|
||||||
|
> * packacke.json: Explicitly set type commonjs [Thodoris Greasidis]
|
||||||
|
>
|
||||||
|
|
||||||
|
</details>
|
||||||
|
|
||||||
|
## 20.1.6 - 2024-12-30
|
||||||
|
|
||||||
|
* Add more realistic os configure tests [Thodoris Greasidis]
|
||||||
|
|
||||||
|
## 20.1.5 - 2024-12-20
|
||||||
|
|
||||||
|
* Update shrinkwrapped express to v4.21.2 [Oskar Williams]
|
||||||
|
|
||||||
|
## 20.1.4 - 2024-12-20
|
||||||
|
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary> Update balena-device-init to 8.0.0 [Thodoris Greasidis] </summary>
|
||||||
|
|
||||||
|
> ### balena-device-init-8.0.0 - 2024-12-18
|
||||||
|
>
|
||||||
|
> * Avoid running linting in the custom tests [Thodoris Greasidis]
|
||||||
|
> * Stop returning Bluebird promises [Thodoris Greasidis]
|
||||||
|
> * package: Publish only the build & typings folders [Thodoris Greasidis]
|
||||||
|
> * Convert to TypeScript with es6 module notation [Thodoris Greasidis]
|
||||||
|
> * Replace gulp, coffeelint & mochainon with tsc, @balena/lint, mocha, chai & sinon [Thodoris Greasidis]
|
||||||
|
> * Drop support for node <20.6.0 [Thodoris Greasidis]
|
||||||
|
>
|
||||||
|
> ### balena-device-init-7.0.2 - 2024-12-17
|
||||||
|
>
|
||||||
|
> * flowzone: Update runner versions [Thodoris Greasidis]
|
||||||
|
> * Pin etcher-sdk to 9.0.8 to match resin-device-operations and fix tests [Thodoris Greasidis]
|
||||||
|
>
|
||||||
|
|
||||||
|
</details>
|
||||||
|
|
||||||
|
## 20.1.3 - 2024-12-20
|
||||||
|
|
||||||
|
* Update oclif to 4.17.0 and @oclif/core 4.1.0 [Otavio Jacobi]
|
||||||
|
|
||||||
|
## 20.1.2 - 2024-12-17
|
||||||
|
|
||||||
|
* Remove unnecessary `Promise.resolve` and `Promise.reject` [Pagan Gazzard]
|
||||||
|
|
||||||
|
## 20.1.1 - 2024-12-16
|
||||||
|
|
||||||
|
* Update @balena/lint to v9.1.3 [Otavio Jacobi]
|
||||||
|
|
||||||
|
## 20.1.0 - 2024-12-12
|
||||||
|
|
||||||
|
* `device os-update`: Add handling for updates that require takeover [myarmolinsky]
|
||||||
|
* Update `balena-sdk` [myarmolinsky]
|
||||||
|
* Update `@balena/compose` [myarmolinsky]
|
||||||
|
|
||||||
|
## 20.0.9 - 2024-12-05
|
||||||
|
|
||||||
|
* Update shrinkwrapped express to v4.21.1 [Oskar Williams]
|
||||||
|
|
||||||
|
## 20.0.8 - 2024-12-04
|
||||||
|
|
||||||
|
* Run test and publish with macos-13 [Otavio Jacobi]
|
||||||
|
|
||||||
|
## 20.0.7 - 2024-11-23
|
||||||
|
|
||||||
|
* Update TypeScript to 5.7.2 [Thodoris Greasidis]
|
||||||
|
|
||||||
|
## 20.0.6 - 2024-11-08
|
||||||
|
|
||||||
|
* Refactor balena build for clarity [Thodoris Greasidis]
|
||||||
|
|
||||||
|
## 20.0.5 - 2024-11-05
|
||||||
|
|
||||||
|
* Update actions/upload-artifact digest to b4b15b8 [balena-renovate[bot]]
|
||||||
|
|
||||||
|
## 20.0.4 - 2024-11-05
|
||||||
|
|
||||||
|
* Update actions/setup-node digest to 39370e3 [balena-renovate[bot]]
|
||||||
|
|
||||||
|
## 20.0.3 - 2024-11-05
|
||||||
|
|
||||||
|
* api-key generate: Display a descriptive error when the generation fails due to a stale JWT [Thodoris Greasidis]
|
||||||
|
|
||||||
|
## 20.0.2 - 2024-10-29
|
||||||
|
|
||||||
|
* Restore ability to cat key into `ssh-key add` [myarmolinsky]
|
||||||
|
|
||||||
|
## 20.0.1 - 2024-10-29
|
||||||
|
|
||||||
|
* Fix sending input to some aliases not working [myarmolinsky]
|
||||||
|
|
||||||
|
## 20.0.0 - 2024-10-25
|
||||||
|
|
||||||
|
* `device update`: Use detached HUP for os updates [jaomaloy]
|
||||||
|
* Drop `-h` flag for help and stop manually adding `help` per command in favor of oclif automatically adding it [myarmolinsky]
|
||||||
|
* Stop checking for very old, long-removed commands [myarmolinsky]
|
||||||
|
* Deprecate `devices supported` command in favor of `device-type list` [myarmolinsky]
|
||||||
|
* Deprecate `notes` command in favor of `device note` [myarmolinsky]
|
||||||
|
* Deprecate `tunnel` command in favor of `device tunnel` [myarmolinsky]
|
||||||
|
* Deprecate `env add` in favor of `env set` [myarmolinsky]
|
||||||
|
* Deprecate `ssh` command in favor of `device ssh` [myarmolinsky]
|
||||||
|
* Deprecate `logs` command in favor of `device logs` [myarmolinsky]
|
||||||
|
* Deprecate `scan` command in favor of `device detect` [myarmolinsky]
|
||||||
|
* Deprecate `orgs` command in favor of `organization list` [myarmolinsky]
|
||||||
|
* Deprecate `tags` command in favor of `tag list` [myarmolinsky]
|
||||||
|
* Deprecate `envs` command in favor of `env list` [myarmolinsky]
|
||||||
|
* Deprecate `key` commands in favor of `ssh-key` [myarmolinsky]
|
||||||
|
* Deprecate `keys` command in favor of `key list` [myarmolinsky]
|
||||||
|
* Deprecate `releases` command in favor of `release list` [myarmolinsky]
|
||||||
|
* Deprecate `fleets` command in favor of `fleet list` [myarmolinsky]
|
||||||
|
* Deprecate `api-keys` command in favor of `api-key list` [myarmolinsky]
|
||||||
|
* Deprecate `devices` command in favor of `device list` [myarmolinsky]
|
||||||
|
* Docs: Show whether an alias is deprecated [myarmolinsky]
|
||||||
|
* Update `balena-preload` to 16.0.0 [myarmolinsky]
|
||||||
|
* Tests: Drop unused `my_application` resource mock [myarmolinsky]
|
||||||
|
* Rename `device.overall_status` values: `IDLE` to `OPERATIONAL`, `OFFLINE` to `DISCONNECTED`; add `REDUCED_FUNCTIONALITY` [myarmolinsky]
|
||||||
|
* Update `@balena/compose` to 5.0.0 [myarmolinsky]
|
||||||
|
* Bump balena-sdk to 20.3.0 [myarmolinsky]
|
||||||
|
|
||||||
|
## 19.16.0 - 2024-10-23
|
||||||
|
|
||||||
|
* device-type list: Add `--all` flag for including no longer supported device types in the list [myarmolinsky]
|
||||||
|
* Add alias `device-type list` for command `devices supported` [myarmolinsky]
|
||||||
|
|
||||||
|
## 19.15.0 - 2024-10-23
|
||||||
|
|
||||||
|
* Add alias `device note` for command `notes` [myarmolinsky]
|
||||||
|
|
||||||
|
## 19.14.0 - 2024-10-22
|
||||||
|
|
||||||
|
* Add alias `device tunnel` for command `tunnel` [myarmolinsky]
|
||||||
|
|
||||||
|
## 19.13.1 - 2024-10-21
|
||||||
|
|
||||||
|
* Update dependency chalk to v4 [Self-hosted Renovate Bot]
|
||||||
|
|
||||||
|
## 19.13.0 - 2024-10-21
|
||||||
|
|
||||||
|
* Add alias `env set` for command `env add` [myarmolinsky]
|
||||||
|
|
||||||
|
## 19.12.1 - 2024-10-21
|
||||||
|
|
||||||
|
* Update `@oclif/core` [myarmolinsky]
|
||||||
|
|
||||||
|
## 19.12.0 - 2024-10-21
|
||||||
|
|
||||||
|
* Add alias `device ssh` for `ssh` command [myarmolinsky]
|
||||||
|
|
||||||
|
## 19.11.1 - 2024-10-21
|
||||||
|
|
||||||
|
* Update dependency sinon to v19 [Self-hosted Renovate Bot]
|
||||||
|
|
||||||
|
## 19.11.0 - 2024-10-21
|
||||||
|
|
||||||
|
* Add alias `device logs` for `logs` [myarmolinsky]
|
||||||
|
* git mv `logs/index` to `device/logs` [myarmolinsky]
|
||||||
|
|
||||||
|
## 19.10.0 - 2024-10-21
|
||||||
|
|
||||||
|
* Add alias `device detect` for `scan` [myarmolinsky]
|
||||||
|
|
||||||
|
## 19.9.0 - 2024-10-18
|
||||||
|
|
||||||
|
* Add alias `organization list` for `orgs` command [myarmolinsky]
|
||||||
|
|
||||||
|
## 19.8.0 - 2024-10-18
|
||||||
|
|
||||||
|
* Add alias `tag list` for `tags` command [myarmolinsky]
|
||||||
|
|
||||||
## 19.7.0 - 2024-10-18
|
## 19.7.0 - 2024-10-18
|
||||||
|
|
||||||
* Add alias `env list` for command `envs` [myarmolinsky]
|
* Add alias `env list` for command `envs` [myarmolinsky]
|
||||||
|
@ -14,7 +14,7 @@ The balena CLI is an open source project and your contribution is welcome!
|
|||||||
In order to ease development:
|
In order to ease development:
|
||||||
|
|
||||||
* `npm run build:fast` skips some of the build steps for interactive testing, or
|
* `npm run build:fast` skips some of the build steps for interactive testing, or
|
||||||
* `npm run test:source` skips testing the standalone zip packages (which is rather slow)
|
* `npm run test:source` skips testing the standalone packages (which is rather slow)
|
||||||
* `./bin/balena-dev` uses `ts-node/register` to transpile on the fly.
|
* `./bin/balena-dev` uses `ts-node/register` to transpile on the fly.
|
||||||
|
|
||||||
Before opening a PR, test your changes with `npm test`. Keep compatibility in mind, as the CLI is
|
Before opening a PR, test your changes with `npm test`. Keep compatibility in mind, as the CLI is
|
||||||
|
@ -8,8 +8,8 @@ There are 3 options to choose from to install balena's CLI:
|
|||||||
|
|
||||||
* [Executable Installer](#executable-installer): the easiest method on Windows and macOS, using the
|
* [Executable Installer](#executable-installer): the easiest method on Windows and macOS, using the
|
||||||
traditional graphical desktop application installers.
|
traditional graphical desktop application installers.
|
||||||
* [Standalone Zip Package](#standalone-zip-package): these are plain zip files with the balena CLI
|
* [Standalone tar.gz Package](#standalone-targz-package): these are plain tar.gz files with the balena CLI
|
||||||
executable in them: extract and run. Available for all platforms: Linux, Windows, macOS.
|
bundled within. Available for all platforms: Linux, Windows, macOS.
|
||||||
Recommended also for scripted installation in CI (continuous integration) environments.
|
Recommended also for scripted installation in CI (continuous integration) environments.
|
||||||
* [NPM Installation](#npm-installation): recommended for Node.js developers who may be interested
|
* [NPM Installation](#npm-installation): recommended for Node.js developers who may be interested
|
||||||
in integrating the balena CLI in their existing projects or workflow.
|
in integrating the balena CLI in their existing projects or workflow.
|
||||||
@ -30,9 +30,9 @@ instructions:
|
|||||||
> If you would like to use WSL, follow the [installations instructions for
|
> If you would like to use WSL, follow the [installations instructions for
|
||||||
> Linux](./INSTALL-LINUX.md) rather than Windows, as WSL consists of a Linux environment.
|
> Linux](./INSTALL-LINUX.md) rather than Windows, as WSL consists of a Linux environment.
|
||||||
|
|
||||||
If you had previously installed the CLI using a standalone zip package, it may be a good idea to
|
If you had previously installed the CLI using a standalone tar package, it may be a good idea to
|
||||||
check your system's `PATH` environment variable for duplicate entries, as the terminal will use the
|
check your system's `PATH` environment variable for duplicate entries, as the terminal will use the
|
||||||
entry that comes first. Check the [Standalone Zip Package](#standalone-zip-package) instructions
|
entry that comes first. Check the [Standalone tar.gz Package](#standalone-targz-package) instructions
|
||||||
for how to modify the PATH variable.
|
for how to modify the PATH variable.
|
||||||
|
|
||||||
By default, the CLI is installed to the following folders:
|
By default, the CLI is installed to the following folders:
|
||||||
@ -42,18 +42,17 @@ OS | Folders
|
|||||||
Windows: | `C:\Program Files\balena-cli\`
|
Windows: | `C:\Program Files\balena-cli\`
|
||||||
macOS: | `/usr/local/src/balena-cli/` <br> `/usr/local/bin/balena`
|
macOS: | `/usr/local/src/balena-cli/` <br> `/usr/local/bin/balena`
|
||||||
|
|
||||||
## Standalone Zip Package
|
## Standalone tar.gz Package
|
||||||
|
|
||||||
1. Download the latest zip file from the [releases page](https://github.com/balena-io/balena-cli/releases).
|
1. Download the latest tar.gz file from the [releases page](https://github.com/balena-io/balena-cli/releases).
|
||||||
Look for a file name that ends with the word "standalone", for example:
|
Look for a file name that ends with the word "standalone", for example:
|
||||||
`balena-cli-vX.Y.Z-linux-x64-standalone.zip` ← _also for the Windows Subsystem for Linux_
|
`balena-cli-vX.Y.Z-linux-x64-standalone.tar.gz` ← _also for the Windows Subsystem for Linux_
|
||||||
`balena-cli-vX.Y.Z-macOS-x64-standalone.zip`
|
`balena-cli-vX.Y.Z-macOS-x64-standalone.tar.gz`
|
||||||
`balena-cli-vX.Y.Z-windows-x64-standalone.zip`
|
`balena-cli-vX.Y.Z-windows-x64-standalone.tar.gz`
|
||||||
|
|
||||||
2. Extract the zip file contents to any folder you choose. The extracted contents will include a
|
2. Extract the tar.gz file contents to any folder you choose. The extracted contents will be a `balena` folder containing a `bin` subdirectory.
|
||||||
`balena-cli` folder.
|
|
||||||
|
|
||||||
3. Add the `balena-cli` folder to the system's `PATH` environment variable.
|
3. Add the `balena/bin` folder to the system's `PATH` environment variable.
|
||||||
See instructions for:
|
See instructions for:
|
||||||
[Linux](https://stackoverflow.com/questions/14637979/how-to-permanently-set-path-on-linux-unix) |
|
[Linux](https://stackoverflow.com/questions/14637979/how-to-permanently-set-path-on-linux-unix) |
|
||||||
[macOS](https://www.architectryan.com/2012/10/02/add-to-the-path-on-mac-os-x-mountain-lion/#.Uydjga1dXDg) |
|
[macOS](https://www.architectryan.com/2012/10/02/add-to-the-path-on-mac-os-x-mountain-lion/#.Uydjga1dXDg) |
|
||||||
@ -61,14 +60,14 @@ macOS: | `/usr/local/src/balena-cli/` <br> `/usr/local/bin/balena`
|
|||||||
|
|
||||||
> * If you are using macOS 10.15 or later (Catalina, Big Sur), [check this known issue and
|
> * If you are using macOS 10.15 or later (Catalina, Big Sur), [check this known issue and
|
||||||
> workaround](https://github.com/balena-io/balena-cli/issues/2244).
|
> workaround](https://github.com/balena-io/balena-cli/issues/2244).
|
||||||
> * **Linux Alpine** and **Busybox:** the standalone zip package is not currently compatible with
|
> * **Linux Alpine** and **Busybox:** the standalone tar.gz package is not currently compatible with
|
||||||
> these "compact" Linux distributions, because of the alternative C libraries they ship with.
|
> these "compact" Linux distributions, because of the alternative C libraries they ship with.
|
||||||
> For these, consider the [NPM Installation](#npm-installation) option.
|
> For these, consider the [NPM Installation](#npm-installation) option.
|
||||||
> * Note that moving the `balena` executable out of the extracted `balena-cli` folder on its own
|
> * Note that moving the `balena/bin/balena` executable out of the extracted `balena` folder on its own
|
||||||
> (e.g. moving it to `/usr/local/bin/balena`) will **not** work, as it depends on the other
|
> (e.g. moving it to `/usr/local/bin/balena`) will **not** work, as it depends on the other
|
||||||
> folders and files also present in the `balena-cli` folder.
|
> folders and files also present in the `balena` folder.
|
||||||
|
|
||||||
To update the CLI to a new version, download a new release zip file and replace the previous
|
To update the CLI to a new version, download a new release tar.gz file and replace the previous
|
||||||
installation folder. To uninstall, simply delete the folder and edit the PATH environment variable
|
installation folder. To uninstall, simply delete the folder and edit the PATH environment variable
|
||||||
as described above.
|
as described above.
|
||||||
|
|
||||||
@ -78,8 +77,8 @@ If you are a Node.js developer, you may wish to install the balena CLI via [npm]
|
|||||||
The npm installation involves building native (platform-specific) binary modules, which require
|
The npm installation involves building native (platform-specific) binary modules, which require
|
||||||
some development tools to be installed first, as follows.
|
some development tools to be installed first, as follows.
|
||||||
|
|
||||||
> **The balena CLI currently requires Node.js version ^20.6.0**
|
> **The balena CLI currently requires Node.js version >=20.6.0**
|
||||||
> **Versions 21 and later are not yet fully supported.**
|
> **Versions 23 and later are not yet fully supported.**
|
||||||
|
|
||||||
### Install development tools
|
### Install development tools
|
||||||
|
|
||||||
@ -89,7 +88,7 @@ some development tools to be installed first, as follows.
|
|||||||
$ sudo apt-get update && sudo apt-get -y install curl python3 git make g++
|
$ sudo apt-get update && sudo apt-get -y install curl python3 git make g++
|
||||||
$ curl -o- https://raw.githubusercontent.com/nvm-sh/nvm/v0.38.0/install.sh | bash
|
$ curl -o- https://raw.githubusercontent.com/nvm-sh/nvm/v0.38.0/install.sh | bash
|
||||||
$ . ~/.bashrc
|
$ . ~/.bashrc
|
||||||
$ nvm install 20
|
$ nvm install 22
|
||||||
```
|
```
|
||||||
|
|
||||||
The `curl` command line above uses
|
The `curl` command line above uses
|
||||||
@ -106,7 +105,7 @@ recommended.
|
|||||||
```sh
|
```sh
|
||||||
$ curl -o- https://raw.githubusercontent.com/nvm-sh/nvm/v0.38.0/install.sh | bash
|
$ curl -o- https://raw.githubusercontent.com/nvm-sh/nvm/v0.38.0/install.sh | bash
|
||||||
$ . ~/.bashrc
|
$ . ~/.bashrc
|
||||||
$ nvm install 20
|
$ nvm install 22
|
||||||
```
|
```
|
||||||
|
|
||||||
#### **Windows** (not WSL)
|
#### **Windows** (not WSL)
|
||||||
@ -114,7 +113,7 @@ $ nvm install 20
|
|||||||
Install:
|
Install:
|
||||||
|
|
||||||
* If you'd like the ability to switch between Node.js versions, install
|
* If you'd like the ability to switch between Node.js versions, install
|
||||||
- Node.js v20 from the [Nodejs.org releases page](https://nodejs.org/en/download/releases/).
|
- Node.js v22 from the [Nodejs.org releases page](https://nodejs.org/en/download/releases/).
|
||||||
[nvm-windows](https://github.com/coreybutler/nvm-windows#node-version-manager-nvm-for-windows)
|
[nvm-windows](https://github.com/coreybutler/nvm-windows#node-version-manager-nvm-for-windows)
|
||||||
instead.
|
instead.
|
||||||
* The [MSYS2 shell](https://www.msys2.org/), which provides `git`, `make`, `g++` and more:
|
* The [MSYS2 shell](https://www.msys2.org/), which provides `git`, `make`, `g++` and more:
|
||||||
@ -145,7 +144,7 @@ container) in order to allow npm scripts like `postinstall` to be executed.
|
|||||||
|
|
||||||
## Additional Dependencies
|
## Additional Dependencies
|
||||||
|
|
||||||
The `balena ssh`, `scan`, `build`, `deploy` and `preload` commands may require
|
The `balena device ssh`, `device detect`, `build`, `deploy` and `preload` commands may require
|
||||||
additional software to be installed. Check the Additional Dependencies sections for each operating
|
additional software to be installed. Check the Additional Dependencies sections for each operating
|
||||||
system:
|
system:
|
||||||
|
|
||||||
|
@ -8,15 +8,15 @@ method.
|
|||||||
|
|
||||||
Selected operating system: **Linux**
|
Selected operating system: **Linux**
|
||||||
|
|
||||||
1. Download the latest zip file from the [latest release
|
1. Download the latest tar.gz file from the [latest release
|
||||||
page](https://github.com/balena-io/balena-cli/releases/latest). Look for a file name that ends
|
page](https://github.com/balena-io/balena-cli/releases/latest). Look for a file name that ends
|
||||||
with "-standalone.zip", for example:
|
with "-standalone.tar.gz", for example:
|
||||||
`balena-cli-vX.Y.Z-linux-x64-standalone.zip`
|
`balena-cli-vX.Y.Z-linux-x64-standalone.tar.gz`
|
||||||
|
|
||||||
2. Extract the zip file contents to any folder you choose, for example `/home/james`.
|
2. Extract the tar.gz file contents to any folder you choose, for example `/home/james`.
|
||||||
The extracted contents will include a `balena-cli` folder.
|
The extracted contents will include a `balena/bin` folder.
|
||||||
|
|
||||||
3. Add that folder (e.g. `/home/james/balena-cli`) to the `PATH` environment variable.
|
3. Add that folder (e.g. `/home/james/balena/bin`) to the `PATH` environment variable.
|
||||||
Check this [StackOverflow
|
Check this [StackOverflow
|
||||||
post](https://stackoverflow.com/questions/14637979/how-to-permanently-set-path-on-linux-unix)
|
post](https://stackoverflow.com/questions/14637979/how-to-permanently-set-path-on-linux-unix)
|
||||||
for instructions. Close and reopen the terminal window so that the changes to `PATH`
|
for instructions. Close and reopen the terminal window so that the changes to `PATH`
|
||||||
@ -27,13 +27,13 @@ Selected operating system: **Linux**
|
|||||||
* `balena version` - should print the CLI's version
|
* `balena version` - should print the CLI's version
|
||||||
* `balena help` - should print a list of available commands
|
* `balena help` - should print a list of available commands
|
||||||
|
|
||||||
To update the balena CLI to a new version, download a new release zip file and replace the previous
|
To update the balena CLI to a new version, download a new release tar.gz file and replace the previous
|
||||||
installation folder. To uninstall, simply delete the folder and edit the PATH environment variable
|
installation folder. To uninstall, simply delete the folder and edit the PATH environment variable
|
||||||
as described above.
|
as described above.
|
||||||
|
|
||||||
## sudo configuration
|
## sudo configuration
|
||||||
|
|
||||||
A few CLI commands require execution through sudo, e.g. `sudo balena scan`.
|
A few CLI commands require execution through sudo, e.g. `sudo balena device detect`.
|
||||||
If your Linux distribution has an `/etc/sudoers` file that defines a `secure_path`
|
If your Linux distribution has an `/etc/sudoers` file that defines a `secure_path`
|
||||||
setting, run `sudo visudo` to edit it and add the balena CLI's installation folder to
|
setting, run `sudo visudo` to edit it and add the balena CLI's installation folder to
|
||||||
the ***pre-existing*** `secure_path` setting, for example:
|
the ***pre-existing*** `secure_path` setting, for example:
|
||||||
@ -61,19 +61,19 @@ instructions](https://docs.docker.com/install/overview/) to install Docker on th
|
|||||||
workstation as the balena CLI. The [advanced installation
|
workstation as the balena CLI. The [advanced installation
|
||||||
options](./INSTALL-ADVANCED.md#additional-dependencies) document describes other possibilities.
|
options](./INSTALL-ADVANCED.md#additional-dependencies) document describes other possibilities.
|
||||||
|
|
||||||
### balena ssh
|
### balena device ssh
|
||||||
|
|
||||||
The `balena ssh` command requires the `ssh` command-line tool to be available. Most Linux
|
The `balena device ssh` command requires the `ssh` command-line tool to be available. Most Linux
|
||||||
distributions will already have it installed. Otherwise, `sudo apt-get install openssh-client`
|
distributions will already have it installed. Otherwise, `sudo apt-get install openssh-client`
|
||||||
should do the trick on Debian or Ubuntu.
|
should do the trick on Debian or Ubuntu.
|
||||||
|
|
||||||
The `balena ssh` command also requires an SSH key to be added to your balena account: see [SSH
|
The `balena device ssh` command also requires an SSH key to be added to your balena account: see [SSH
|
||||||
Access documentation](https://www.balena.io/docs/learn/manage/ssh-access/). The `balena key*`
|
Access documentation](https://www.balena.io/docs/learn/manage/ssh-access/). The `balena key*`
|
||||||
command set can also be used to list and manage SSH keys: see `balena help -v`.
|
command set can also be used to list and manage SSH keys: see `balena help -v`.
|
||||||
|
|
||||||
### balena scan
|
### balena device detect
|
||||||
|
|
||||||
The `balena scan` command requires a multicast DNS (mDNS) service like
|
The `balena device detect` command requires a multicast DNS (mDNS) service like
|
||||||
[Avahi](https://en.wikipedia.org/wiki/Avahi_(software)), which is installed by default on most
|
[Avahi](https://en.wikipedia.org/wiki/Avahi_(software)), which is installed by default on most
|
||||||
desktop Linux distributions. Otherwise, on Debian or Ubuntu, the installation command would be
|
desktop Linux distributions. Otherwise, on Debian or Ubuntu, the installation command would be
|
||||||
`sudo apt-get install avahi-daemon`.
|
`sudo apt-get install avahi-daemon`.
|
||||||
|
@ -7,8 +7,8 @@ Selected operating system: **macOS**
|
|||||||
|
|
||||||
1. Download the installer from the [latest release
|
1. Download the installer from the [latest release
|
||||||
page](https://github.com/balena-io/balena-cli/releases/latest).
|
page](https://github.com/balena-io/balena-cli/releases/latest).
|
||||||
Look for a file name that ends with "-installer.pkg":
|
Look for a file name that ends with "-installer.pkg":
|
||||||
`balena-cli-vX.Y.Z-macOS-x64-installer.pkg`
|
`balena-cli-vX.Y.Z-macOS-x64-installer.pkg`
|
||||||
|
|
||||||
2. Double click on the downloaded file to run the installer and follow the installer's
|
2. Double click on the downloaded file to run the installer and follow the installer's
|
||||||
instructions.
|
instructions.
|
||||||
@ -19,7 +19,7 @@ Selected operating system: **macOS**
|
|||||||
- On the terminal prompt, type `balena version` and hit Enter. It should display
|
- On the terminal prompt, type `balena version` and hit Enter. It should display
|
||||||
the version of the balena CLI that you have installed.
|
the version of the balena CLI that you have installed.
|
||||||
|
|
||||||
No further steps are required to run most CLI commands. The `balena ssh`, `build`, `deploy`
|
No further steps are required to run most CLI commands. The `balena device ssh`, `build`, `deploy`
|
||||||
and `preload` commands may require additional software to be installed, as described
|
and `preload` commands may require additional software to be installed, as described
|
||||||
in the next section.
|
in the next section.
|
||||||
|
|
||||||
@ -41,9 +41,9 @@ instructions](https://docs.docker.com/install/overview/) to install Docker on th
|
|||||||
workstation as the balena CLI. The [advanced installation
|
workstation as the balena CLI. The [advanced installation
|
||||||
options](./INSTALL-ADVANCED.md#additional-dependencies) document describes other possibilities.
|
options](./INSTALL-ADVANCED.md#additional-dependencies) document describes other possibilities.
|
||||||
|
|
||||||
### balena ssh
|
### balena device ssh
|
||||||
|
|
||||||
The `balena ssh` command requires the `ssh` command-line tool to be available. To check whether
|
The `balena device ssh` command requires the `ssh` command-line tool to be available. To check whether
|
||||||
it is already installed, run `ssh` on a Terminal window. If it is not yet installed, the options
|
it is already installed, run `ssh` on a Terminal window. If it is not yet installed, the options
|
||||||
include:
|
include:
|
||||||
|
|
||||||
@ -52,7 +52,7 @@ include:
|
|||||||
Components → Command Line Tools → Install.
|
Components → Command Line Tools → Install.
|
||||||
* Or, install [Homebrew](https://brew.sh/), then `brew install openssh`
|
* Or, install [Homebrew](https://brew.sh/), then `brew install openssh`
|
||||||
|
|
||||||
The `balena ssh` command also requires an SSH key to be added to your balena account: see [SSH
|
The `balena device ssh` command also requires an SSH key to be added to your balena account: see [SSH
|
||||||
Access documentation](https://www.balena.io/docs/learn/manage/ssh-access/). The `balena key*`
|
Access documentation](https://www.balena.io/docs/learn/manage/ssh-access/). The `balena key*`
|
||||||
command set can also be used to list and manage SSH keys: see `balena help -v`.
|
command set can also be used to list and manage SSH keys: see `balena help -v`.
|
||||||
|
|
||||||
|
@ -8,7 +8,7 @@ Selected operating system: **Windows**
|
|||||||
1. Download the installer from the [latest release
|
1. Download the installer from the [latest release
|
||||||
page](https://github.com/balena-io/balena-cli/releases/latest).
|
page](https://github.com/balena-io/balena-cli/releases/latest).
|
||||||
Look for a file name that ends with "-installer.exe":
|
Look for a file name that ends with "-installer.exe":
|
||||||
`balena-cli-vX.Y.Z-windows-x64-installer.exe`
|
`balena-cli-vX.Y.Z-windows-x64-installer.exe`
|
||||||
|
|
||||||
2. Double click on the downloaded file to run the installer and follow the installer's
|
2. Double click on the downloaded file to run the installer and follow the installer's
|
||||||
instructions.
|
instructions.
|
||||||
@ -19,7 +19,7 @@ Selected operating system: **Windows**
|
|||||||
- On the command prompt, type `balena version` and hit Enter. It should display
|
- On the command prompt, type `balena version` and hit Enter. It should display
|
||||||
the version of the balena CLI that you have installed.
|
the version of the balena CLI that you have installed.
|
||||||
|
|
||||||
No further steps are required to run most CLI commands. The `balena ssh`, `scan`, `build`,
|
No further steps are required to run most CLI commands. The `balena device ssh`, `device detect`, `build`,
|
||||||
`deploy` and `preload` commands may require additional software to be installed, as
|
`deploy` and `preload` commands may require additional software to be installed, as
|
||||||
described below.
|
described below.
|
||||||
|
|
||||||
@ -34,9 +34,9 @@ instructions](https://docs.docker.com/install/overview/) to install Docker on th
|
|||||||
workstation as the balena CLI. The [advanced installation
|
workstation as the balena CLI. The [advanced installation
|
||||||
options](./INSTALL-ADVANCED.md#additional-dependencies) document describes other possibilities.
|
options](./INSTALL-ADVANCED.md#additional-dependencies) document describes other possibilities.
|
||||||
|
|
||||||
### balena ssh
|
### balena device ssh
|
||||||
|
|
||||||
The `balena ssh` command requires the `ssh` command-line tool to be available. Microsoft started
|
The `balena device ssh` command requires the `ssh` command-line tool to be available. Microsoft started
|
||||||
distributing an SSH client with Windows 10, which is automatically installed through Windows
|
distributing an SSH client with Windows 10, which is automatically installed through Windows
|
||||||
Update. To check whether it is installed, run `ssh` on a Windows Command Prompt or PowerShell. It
|
Update. To check whether it is installed, run `ssh` on a Windows Command Prompt or PowerShell. It
|
||||||
can also be [manually
|
can also be [manually
|
||||||
@ -44,13 +44,13 @@ installed](https://docs.microsoft.com/en-us/windows-server/administration/openss
|
|||||||
if needed. For older versions of Windows, there are several ssh/OpenSSH clients provided by 3rd
|
if needed. For older versions of Windows, there are several ssh/OpenSSH clients provided by 3rd
|
||||||
parties.
|
parties.
|
||||||
|
|
||||||
The `balena ssh` command also requires an SSH key to be added to your balena account: see [SSH
|
The `balena device ssh` command also requires an SSH key to be added to your balena account: see [SSH
|
||||||
Access documentation](https://www.balena.io/docs/learn/manage/ssh-access/). The `balena key*`
|
Access documentation](https://www.balena.io/docs/learn/manage/ssh-access/). The `balena key*`
|
||||||
command set can also be used to list and manage SSH keys: see `balena help -v`.
|
command set can also be used to list and manage SSH keys: see `balena help -v`.
|
||||||
|
|
||||||
### balena scan
|
### balena device detect
|
||||||
|
|
||||||
The `balena scan` command requires a multicast DNS (mDNS) service like Apple's Bonjour.
|
The `balena device detect` command requires a multicast DNS (mDNS) service like Apple's Bonjour.
|
||||||
Many Windows machines will already have this service installed, as it is bundled in popular
|
Many Windows machines will already have this service installed, as it is bundled in popular
|
||||||
applications such as Skype (Wikipedia lists [several others](https://en.wikipedia.org/wiki/Bonjour_(software))).
|
applications such as Skype (Wikipedia lists [several others](https://en.wikipedia.org/wiki/Bonjour_(software))).
|
||||||
Otherwise, Bonjour for Windows can be downloaded and installed from: https://support.apple.com/kb/DL999
|
Otherwise, Bonjour for Windows can be downloaded and installed from: https://support.apple.com/kb/DL999
|
||||||
|
45
MIGRATIONS.md
Normal file
45
MIGRATIONS.md
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
## Migrating to balena CLI v22
|
||||||
|
|
||||||
|
This guide outlines the changes introduced in balena CLI v22 and provides instructions for when and how to migrate.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### For Installer Users (Windows .exe, macOS .pkg)
|
||||||
|
|
||||||
|
If you are using the Windows executable (.exe) or macOS package (.pkg) installers, **no changes** are required for this update. You can continue to use the installers as before.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### For npm Installation Users
|
||||||
|
|
||||||
|
If you installed balena CLI via npm, **no changes** are required for this update. Your existing installation and update process remains the same.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### For Standalone Installation Users
|
||||||
|
|
||||||
|
Users of the standalone balena CLI will need to make the following adjustments:
|
||||||
|
|
||||||
|
1. **Archive Format Change**: The distribution archive format has changed from `.zip` to `.tar.gz`. You will need to use the `tar` command instead of `unzip` to extract the CLI.
|
||||||
|
|
||||||
|
* **Previous command (v21.x.x and older):**
|
||||||
|
```bash
|
||||||
|
unzip balena-cli-v21.1.12-linux-x64-standalone.zip
|
||||||
|
```
|
||||||
|
* **New command (v22.0.0 and newer):**
|
||||||
|
```bash
|
||||||
|
tar -xzf balena-cli-v22.0.0-linux-x64-standalone.tar.gz
|
||||||
|
```
|
||||||
|
|
||||||
|
2. **Executable Path Change**: The path to the balena CLI executable within the extracted folder has been updated.
|
||||||
|
|
||||||
|
* **Previous path (v21.x.x and older):**
|
||||||
|
```
|
||||||
|
balena-cli/balena
|
||||||
|
```
|
||||||
|
* **New path (v22.0.0 and newer):**
|
||||||
|
```
|
||||||
|
balena/bin/balena
|
||||||
|
```
|
||||||
|
|
||||||
|
Please update your scripts and any aliases to reflect these changes if you are using the standalone version.
|
@ -20,6 +20,8 @@ GitHub](https://github.com/balena-io/balena-cli/), and your contribution is also
|
|||||||
Check the [balena CLI installation instructions on
|
Check the [balena CLI installation instructions on
|
||||||
GitHub](https://github.com/balena-io/balena-cli/blob/master/INSTALL.md).
|
GitHub](https://github.com/balena-io/balena-cli/blob/master/INSTALL.md).
|
||||||
|
|
||||||
|
Note for v22 Migration: If you are upgrading to balena CLI v22 from a previous standalone installation, please [see the v22 Migration Guide](https://github.com/balena-io/balena-cli/blob/master/MIGRATIONS.md) for installation changes.
|
||||||
|
|
||||||
## Choosing a shell (command prompt/terminal)
|
## Choosing a shell (command prompt/terminal)
|
||||||
|
|
||||||
On **Windows,** the standard Command Prompt (`cmd.exe`) and
|
On **Windows,** the standard Command Prompt (`cmd.exe`) and
|
||||||
@ -88,9 +90,9 @@ HTTP(S) proxies can be configured through any of the following methods, in prece
|
|||||||
* The `HTTPS_PROXY` and/or `HTTP_PROXY` environment variables, in the same URL format as
|
* The `HTTPS_PROXY` and/or `HTTP_PROXY` environment variables, in the same URL format as
|
||||||
`BALENARC_PROXY`.
|
`BALENARC_PROXY`.
|
||||||
|
|
||||||
### Proxy setup for balena ssh
|
### Proxy setup for balena device ssh
|
||||||
|
|
||||||
In order to work behind a proxy server, the `balena ssh` command requires the
|
In order to work behind a proxy server, the `balena device ssh` command requires the
|
||||||
[`proxytunnel`](http://proxytunnel.sourceforge.net/) package (command-line tool) to be installed.
|
[`proxytunnel`](http://proxytunnel.sourceforge.net/) package (command-line tool) to be installed.
|
||||||
`proxytunnel` is available for Linux distributions like Ubuntu/Debian (`apt install proxytunnel`),
|
`proxytunnel` is available for Linux distributions like Ubuntu/Debian (`apt install proxytunnel`),
|
||||||
and for macOS through [Homebrew](https://brew.sh/). Windows support is limited to the [Windows
|
and for macOS through [Homebrew](https://brew.sh/). Windows support is limited to the [Windows
|
||||||
@ -110,7 +112,7 @@ The `BALENARC_NO_PROXY` variable may be used to exclude specified destinations f
|
|||||||
> * This feature requires CLI version 11.30.8 or later. In the case of the npm [installation
|
> * This feature requires CLI version 11.30.8 or later. In the case of the npm [installation
|
||||||
> option](https://github.com/balena-io/balena-cli/blob/master/INSTALL.md), it also requires
|
> option](https://github.com/balena-io/balena-cli/blob/master/INSTALL.md), it also requires
|
||||||
> Node.js version 10.16.0 or later.
|
> Node.js version 10.16.0 or later.
|
||||||
> * To exclude a `balena ssh` target from proxying (IP address or `.local` hostname), the
|
> * To exclude a `balena device ssh` target from proxying (IP address or `.local` hostname), the
|
||||||
> `--noproxy` option should be specified in addition to the `BALENARC_NO_PROXY` variable.
|
> `--noproxy` option should be specified in addition to the `BALENARC_NO_PROXY` variable.
|
||||||
|
|
||||||
By default (if `BALENARC_NO_PROXY` is not defined), all [private IPv4
|
By default (if `BALENARC_NO_PROXY` is not defined), all [private IPv4
|
||||||
|
@ -31,7 +31,7 @@ command again.
|
|||||||
|
|
||||||
Check whether the SD card is locked (a physical switch on the side of the card).
|
Check whether the SD card is locked (a physical switch on the side of the card).
|
||||||
|
|
||||||
## I get `connect ETIMEDOUT` with `balena tunnel`
|
## I get `connect ETIMEDOUT` with `balena device tunnel`
|
||||||
|
|
||||||
Please update the CLI to the latest version. This issue was fixed in v12.38.5.
|
Please update the CLI to the latest version. This issue was fixed in v12.38.5.
|
||||||
For more details, see: https://github.com/balena-io/balena-cli/issues/2172
|
For more details, see: https://github.com/balena-io/balena-cli/issues/2172
|
||||||
@ -79,10 +79,10 @@ Try resetting the ownership by running:
|
|||||||
$ sudo chown -R <user> $HOME/.balena
|
$ sudo chown -R <user> $HOME/.balena
|
||||||
```
|
```
|
||||||
|
|
||||||
## Broken line wrapping / cursor behavior with `balena ssh`
|
## Broken line wrapping / cursor behavior with `balena device ssh`
|
||||||
|
|
||||||
Users sometimes come across broken line wrapping or cursor behavior in text terminals, for example
|
Users sometimes come across broken line wrapping or cursor behavior in text terminals, for example
|
||||||
when long command lines are typed in a `balena ssh` session, or when using text editors like `vim`
|
when long command lines are typed in a `balena device ssh` session, or when using text editors like `vim`
|
||||||
or `nano`. This is not something specific to the balena CLI, being also a commonly reported issue
|
or `nano`. This is not something specific to the balena CLI, being also a commonly reported issue
|
||||||
with standard remote terminal tools like `ssh` or `telnet`. It is often a remote shell
|
with standard remote terminal tools like `ssh` or `telnet`. It is often a remote shell
|
||||||
configuration issue (files like `/etc/profile`, `~/.bash_profile`, `~/.bash_login`, `~/.profile`
|
configuration issue (files like `/etc/profile`, `~/.bash_profile`, `~/.bash_login`, `~/.profile`
|
||||||
@ -115,7 +115,7 @@ If nothing seems to help, consider also using a different client-side terminal a
|
|||||||
## "Docker seems to be unavailable" error when using Windows Subsystem for Linux (WSL)
|
## "Docker seems to be unavailable" error when using Windows Subsystem for Linux (WSL)
|
||||||
|
|
||||||
When running on WSL, the recommendation is to install a CLI release for Linux, like the standalone
|
When running on WSL, the recommendation is to install a CLI release for Linux, like the standalone
|
||||||
zip package for Linux. However, commands like "balena build" will, by default, attempt to reach the
|
tar.gz package for Linux. However, commands like "balena build" will, by default, attempt to reach the
|
||||||
Docker daemon at the Unix socket path `/var/run/docker.sock`, while Docker Desktop for Windows uses
|
Docker daemon at the Unix socket path `/var/run/docker.sock`, while Docker Desktop for Windows uses
|
||||||
a Windows named pipe at `//./pipe/docker_engine` (which the Linux CLI on WSL cannot use). A
|
a Windows named pipe at `//./pipe/docker_engine` (which the Linux CLI on WSL cannot use). A
|
||||||
solution is:
|
solution is:
|
||||||
|
@ -15,29 +15,17 @@
|
|||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import type { JsonVersions } from '../src/commands/version/index';
|
|
||||||
|
|
||||||
import { run as oclifRun } from '@oclif/core';
|
import { run as oclifRun } from '@oclif/core';
|
||||||
import * as archiver from 'archiver';
|
|
||||||
import { exec, execFile } from 'child_process';
|
import { exec, execFile } from 'child_process';
|
||||||
import * as filehound from 'filehound';
|
|
||||||
import type { Stats } from 'fs';
|
import type { Stats } from 'fs';
|
||||||
import * as fs from 'fs-extra';
|
import * as fs from 'fs-extra';
|
||||||
import * as klaw from 'klaw';
|
import * as klaw from 'klaw';
|
||||||
import * as path from 'path';
|
import * as path from 'path';
|
||||||
import * as rimraf from 'rimraf';
|
import * as rimraf from 'rimraf';
|
||||||
import * as semver from 'semver';
|
|
||||||
import { promisify } from 'util';
|
import { promisify } from 'util';
|
||||||
import { notarize } from '@electron/notarize';
|
import { notarize } from '@electron/notarize';
|
||||||
|
|
||||||
import { stripIndent } from '../build/utils/lazy';
|
import { loadPackageJson, ROOT, whichSpawn } from './utils';
|
||||||
import {
|
|
||||||
diffLines,
|
|
||||||
loadPackageJson,
|
|
||||||
ROOT,
|
|
||||||
StdOutTap,
|
|
||||||
whichSpawn,
|
|
||||||
} from './utils';
|
|
||||||
|
|
||||||
const execFileAsync = promisify(execFile);
|
const execFileAsync = promisify(execFile);
|
||||||
const execAsync = promisify(exec);
|
const execAsync = promisify(exec);
|
||||||
@ -55,12 +43,6 @@ interface PathByPlatform {
|
|||||||
[platform: string]: string;
|
[platform: string]: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
const standaloneZips: PathByPlatform = {
|
|
||||||
linux: dPath(`balena-cli-${version}-linux-${arch}-standalone.zip`),
|
|
||||||
darwin: dPath(`balena-cli-${version}-macOS-${arch}-standalone.zip`),
|
|
||||||
win32: dPath(`balena-cli-${version}-windows-${arch}-standalone.zip`),
|
|
||||||
};
|
|
||||||
|
|
||||||
const getOclifInstallersOriginalNames = async (): Promise<PathByPlatform> => {
|
const getOclifInstallersOriginalNames = async (): Promise<PathByPlatform> => {
|
||||||
const { stdout } = await execAsync('git rev-parse --short HEAD');
|
const { stdout } = await execAsync('git rev-parse --short HEAD');
|
||||||
const sha = stdout.trim();
|
const sha = stdout.trim();
|
||||||
@ -75,260 +57,28 @@ const renamedOclifInstallers: PathByPlatform = {
|
|||||||
win32: dPath(`balena-cli-${version}-windows-${arch}-installer.exe`),
|
win32: dPath(`balena-cli-${version}-windows-${arch}-installer.exe`),
|
||||||
};
|
};
|
||||||
|
|
||||||
export const finalReleaseAssets: { [platform: string]: string[] } = {
|
const getOclifStandaloneOriginalNames = async (): Promise<PathByPlatform> => {
|
||||||
win32: [standaloneZips['win32'], renamedOclifInstallers['win32']],
|
const { stdout } = await execAsync('git rev-parse --short HEAD');
|
||||||
darwin: [standaloneZips['darwin'], renamedOclifInstallers['darwin']],
|
const sha = stdout.trim();
|
||||||
linux: [standaloneZips['linux']],
|
return {
|
||||||
|
linux: dPath(`balena-${version}-${sha}-linux-${arch}.tar.gz`),
|
||||||
|
darwin: dPath(`balena-${version}-${sha}-darwin-${arch}.tar.gz`),
|
||||||
|
win32: dPath(`balena-${version}-${sha}-win32-${arch}.tar.gz`),
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
const renamedOclifStandalone: PathByPlatform = {
|
||||||
* Given the output of `pkg` as a string (containing warning messages),
|
linux: dPath(`balena-cli-${version}-linux-${arch}-standalone.tar.gz`),
|
||||||
* diff it against previously saved output of known "safe" warnings.
|
darwin: dPath(`balena-cli-${version}-macOS-${arch}-standalone.tar.gz`),
|
||||||
* Throw an error if the diff is not empty.
|
win32: dPath(`balena-cli-${version}-windows-${arch}-standalone.tar.gz`),
|
||||||
*/
|
};
|
||||||
async function diffPkgOutput(pkgOut: string) {
|
|
||||||
const { monochrome } = await import('../tests/helpers');
|
|
||||||
const relSavedPath = path.join(
|
|
||||||
'tests',
|
|
||||||
'test-data',
|
|
||||||
'pkg',
|
|
||||||
`expected-warnings-${process.platform}-${arch}.txt`,
|
|
||||||
);
|
|
||||||
const absSavedPath = path.join(ROOT, relSavedPath);
|
|
||||||
const ignoreStartsWith = [
|
|
||||||
'> pkg@',
|
|
||||||
'> Fetching base Node.js binaries',
|
|
||||||
' fetched-',
|
|
||||||
'prebuild-install WARN install No prebuilt binaries found',
|
|
||||||
];
|
|
||||||
const modulesRE =
|
|
||||||
process.platform === 'win32'
|
|
||||||
? /(?<=[ '])([A-Z]:)?\\.+?\\node_modules(?=\\)/
|
|
||||||
: /(?<=[ '])\/.+?\/node_modules(?=\/)/;
|
|
||||||
const buildRE =
|
|
||||||
process.platform === 'win32'
|
|
||||||
? /(?<=[ '])([A-Z]:)?\\.+\\build(?=\\)/
|
|
||||||
: /(?<=[ '])\/.+\/build(?=\/)/;
|
|
||||||
|
|
||||||
const cleanLines = (chunks: string | string[]) => {
|
|
||||||
const lines = typeof chunks === 'string' ? chunks.split('\n') : chunks;
|
|
||||||
return lines
|
|
||||||
.map((line: string) => monochrome(line)) // remove ASCII colors
|
|
||||||
.filter((line: string) => !/^\s*$/.test(line)) // blank lines
|
|
||||||
.filter((line: string) =>
|
|
||||||
ignoreStartsWith.every((i) => !line.startsWith(i)),
|
|
||||||
)
|
|
||||||
.map((line: string) => {
|
|
||||||
// replace absolute paths with relative paths
|
|
||||||
let replaced = line.replace(modulesRE, 'node_modules');
|
|
||||||
if (replaced === line) {
|
|
||||||
replaced = line.replace(buildRE, 'build');
|
|
||||||
}
|
|
||||||
return replaced;
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
pkgOut = cleanLines(pkgOut).join('\n');
|
|
||||||
const { readFile } = (await import('fs')).promises;
|
|
||||||
const expectedOut = cleanLines(await readFile(absSavedPath, 'utf8')).join(
|
|
||||||
'\n',
|
|
||||||
);
|
|
||||||
if (expectedOut !== pkgOut) {
|
|
||||||
const sep =
|
|
||||||
'================================================================================';
|
|
||||||
const diff = diffLines(expectedOut, pkgOut);
|
|
||||||
const msg = `pkg output does not match expected output from "${relSavedPath}"
|
|
||||||
Diff:
|
|
||||||
${sep}
|
|
||||||
${diff}
|
|
||||||
${sep}
|
|
||||||
Check whether the new or changed pkg warnings are safe to ignore, then update
|
|
||||||
"${relSavedPath}"
|
|
||||||
and share the result of your investigation as comments on the pull request.
|
|
||||||
Hint: the fix is often a matter of updating the 'pkg.scripts' or 'pkg.assets'
|
|
||||||
sections in the CLI's 'package.json' file, or a matter of updating the
|
|
||||||
'buildPkg' function in 'automation/build-bin.ts'. Sometimes it requires
|
|
||||||
patching dependencies: See for example 'patches/all/open+7.0.2.patch'.
|
|
||||||
${sep}
|
|
||||||
`;
|
|
||||||
throw new Error(msg);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Call `pkg.exec` to generate the standalone zip file, capturing its warning
|
|
||||||
* messages (stdout and stderr) in order to call diffPkgOutput().
|
|
||||||
*/
|
|
||||||
async function execPkg(...args: any[]) {
|
|
||||||
const { exec: pkgExec } = await import('@yao-pkg/pkg');
|
|
||||||
const outTap = new StdOutTap(true);
|
|
||||||
try {
|
|
||||||
outTap.tap();
|
|
||||||
await (pkgExec as any)(...args);
|
|
||||||
} catch (err) {
|
|
||||||
outTap.untap();
|
|
||||||
console.log(outTap.stdoutBuf.join(''));
|
|
||||||
console.error(outTap.stderrBuf.join(''));
|
|
||||||
throw err;
|
|
||||||
}
|
|
||||||
outTap.untap();
|
|
||||||
await diffPkgOutput(outTap.allBuf.join(''));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Use the 'pkg' module to create a single large executable file with
|
|
||||||
* the contents of 'node_modules' and the CLI's javascript code.
|
|
||||||
* Also copy a number of native modules (binary '.node' files) that are
|
|
||||||
* compiled during 'npm install' to the 'build-bin' folder, alongside
|
|
||||||
* the single large executable file created by pkg. (This is necessary
|
|
||||||
* because of a pkg limitation that does not allow binary executables
|
|
||||||
* to be directly executed from inside another binary executable.)
|
|
||||||
*/
|
|
||||||
async function buildPkg() {
|
|
||||||
// https://github.com/vercel/pkg#targets
|
|
||||||
let targets = `linux-${arch}`;
|
|
||||||
if (process.platform === 'darwin') {
|
|
||||||
targets = `macos-${arch}`;
|
|
||||||
}
|
|
||||||
// TBC: not yet possible to build for Windows arm64 on x64 nodes
|
|
||||||
if (process.platform === 'win32') {
|
|
||||||
targets = `win-x64`;
|
|
||||||
}
|
|
||||||
const args = [
|
|
||||||
'--targets',
|
|
||||||
targets,
|
|
||||||
'--output',
|
|
||||||
'build-bin/balena',
|
|
||||||
'package.json',
|
|
||||||
];
|
|
||||||
console.log('=======================================================');
|
|
||||||
console.log(`execPkg ${args.join(' ')}`);
|
|
||||||
console.log(`cwd="${process.cwd()}" ROOT="${ROOT}"`);
|
|
||||||
console.log('=======================================================');
|
|
||||||
|
|
||||||
await execPkg(args);
|
|
||||||
|
|
||||||
const paths: Array<[string, string[], string[]]> = [
|
|
||||||
// [platform, [source path], [destination path]]
|
|
||||||
['*', ['open', 'xdg-open'], ['xdg-open']],
|
|
||||||
['darwin', ['denymount', 'bin', 'denymount'], ['denymount']],
|
|
||||||
];
|
|
||||||
await Promise.all(
|
|
||||||
paths.map(([platform, source, dest]) => {
|
|
||||||
if (platform === '*' || platform === process.platform) {
|
|
||||||
// eg copy from node_modules/open/xdg-open to build-bin/xdg-open
|
|
||||||
return fs.copy(
|
|
||||||
path.join(ROOT, 'node_modules', ...source),
|
|
||||||
path.join(ROOT, 'build-bin', ...dest),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
const nativeExtensionPaths: string[] = await filehound
|
|
||||||
.create()
|
|
||||||
.paths(path.join(ROOT, 'node_modules'))
|
|
||||||
.ext(['node', 'dll'])
|
|
||||||
.find();
|
|
||||||
|
|
||||||
console.log(`\nCopying to build-bin:\n${nativeExtensionPaths.join('\n')}`);
|
|
||||||
|
|
||||||
await Promise.all(
|
|
||||||
nativeExtensionPaths.map((extPath) =>
|
|
||||||
fs.copy(
|
|
||||||
extPath,
|
|
||||||
extPath.replace(
|
|
||||||
path.join(ROOT, 'node_modules'),
|
|
||||||
path.join(ROOT, 'build-bin'),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Run some basic tests on the built pkg executable.
|
|
||||||
* TODO: test more than just `balena version -j`; integrate with the
|
|
||||||
* existing mocha/chai CLI command testing.
|
|
||||||
*/
|
|
||||||
async function testPkg() {
|
|
||||||
const pkgBalenaPath = path.join(
|
|
||||||
ROOT,
|
|
||||||
'build-bin',
|
|
||||||
process.platform === 'win32' ? 'balena.exe' : 'balena',
|
|
||||||
);
|
|
||||||
console.log(`Testing standalone package "${pkgBalenaPath}"...`);
|
|
||||||
// Run `balena version -j`, parse its stdout as JSON, and check that the
|
|
||||||
// reported Node.js major version matches semver.major(process.version)
|
|
||||||
let { stdout, stderr } = await execFileAsync(pkgBalenaPath, [
|
|
||||||
'version',
|
|
||||||
'-j',
|
|
||||||
]);
|
|
||||||
const { filterCliOutputForTests } = await import('../tests/helpers');
|
|
||||||
const filtered = filterCliOutputForTests({
|
|
||||||
err: stderr.split(/\r?\n/),
|
|
||||||
out: stdout.split(/\r?\n/),
|
|
||||||
});
|
|
||||||
stdout = filtered.out.join('\n');
|
|
||||||
stderr = filtered.err.join('\n');
|
|
||||||
let pkgNodeVersion = '';
|
|
||||||
let pkgNodeMajorVersion = 0;
|
|
||||||
try {
|
|
||||||
const balenaVersions: JsonVersions = JSON.parse(stdout);
|
|
||||||
pkgNodeVersion = balenaVersions['Node.js'];
|
|
||||||
pkgNodeMajorVersion = semver.major(pkgNodeVersion);
|
|
||||||
} catch (err) {
|
|
||||||
throw new Error(stripIndent`
|
|
||||||
Error parsing JSON output of "balena version -j": ${err}
|
|
||||||
Original output: "${stdout}"`);
|
|
||||||
}
|
|
||||||
if (semver.major(process.version) !== pkgNodeMajorVersion) {
|
|
||||||
throw new Error(
|
|
||||||
`Mismatched major version: built-in pkg Node version="${pkgNodeVersion}" vs process.version="${process.version}"`,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
if (filtered.err.length > 0) {
|
|
||||||
const err = filtered.err.join('\n');
|
|
||||||
throw new Error(`"${pkgBalenaPath}": non-empty stderr "${err}"`);
|
|
||||||
}
|
|
||||||
console.log('Success! (standalone package test successful)');
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create the zip file for the standalone 'pkg' bundle previously created
|
|
||||||
* by the buildPkg() function in 'build-bin.ts'.
|
|
||||||
*/
|
|
||||||
async function zipPkg() {
|
|
||||||
const outputFile = standaloneZips[process.platform];
|
|
||||||
if (!outputFile) {
|
|
||||||
throw new Error(
|
|
||||||
`Standalone installer unavailable for platform "${process.platform}"`,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
await fs.mkdirp(path.dirname(outputFile));
|
|
||||||
await new Promise((resolve, reject) => {
|
|
||||||
console.log(`Zipping standalone package to "${outputFile}"...`);
|
|
||||||
|
|
||||||
const archive = archiver('zip', {
|
|
||||||
zlib: { level: 7 },
|
|
||||||
});
|
|
||||||
archive.directory(path.join(ROOT, 'build-bin'), 'balena-cli');
|
|
||||||
|
|
||||||
const outputStream = fs.createWriteStream(outputFile);
|
|
||||||
|
|
||||||
outputStream.on('close', resolve);
|
|
||||||
outputStream.on('error', reject);
|
|
||||||
|
|
||||||
archive.on('error', reject);
|
|
||||||
archive.on('warning', console.warn);
|
|
||||||
|
|
||||||
archive.pipe(outputStream);
|
|
||||||
archive.finalize().catch(reject);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function signFilesForNotarization() {
|
export async function signFilesForNotarization() {
|
||||||
console.log('Signing files for notarization');
|
console.log('Signing files for notarization');
|
||||||
if (process.platform !== 'darwin') {
|
// If signFilesForNotarization is called on the test CI environment (which will not set CSC_LINK)
|
||||||
|
// then we skip the signing process.
|
||||||
|
if (process.platform !== 'darwin' || !process.env.CSC_LINK) {
|
||||||
|
console.log('Skipping signing for notarization');
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
console.log('Deleting unneeded zip files...');
|
console.log('Deleting unneeded zip files...');
|
||||||
@ -416,20 +166,39 @@ export async function signFilesForNotarization() {
|
|||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function buildStandaloneZip() {
|
export async function buildStandalone() {
|
||||||
console.log(`Building standalone zip package for CLI ${version}`);
|
console.log(`Building standalone tarball for CLI ${version}`);
|
||||||
|
fs.rmSync('./tmp', { recursive: true, force: true });
|
||||||
|
fs.rmSync('./dist', { recursive: true, force: true });
|
||||||
|
fs.mkdirSync('./dist');
|
||||||
try {
|
try {
|
||||||
await buildPkg();
|
let packOpts = ['-r', ROOT, '--no-xz'];
|
||||||
await testPkg();
|
if (process.platform === 'darwin') {
|
||||||
await zipPkg();
|
packOpts = packOpts.concat('--targets', `darwin-${arch}`);
|
||||||
console.log(`Standalone zip package build completed`);
|
} else if (process.platform === 'win32') {
|
||||||
|
packOpts = packOpts.concat('--targets', 'win32-x64');
|
||||||
|
} else if (process.platform === 'linux') {
|
||||||
|
packOpts = packOpts.concat('--targets', `linux-${arch}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log(`Building oclif installer for CLI ${version}`);
|
||||||
|
const packCmd = `pack:tarballs`;
|
||||||
|
console.log('=======================================================');
|
||||||
|
console.log(`oclif ${packCmd} ${packOpts.join(' ')}`);
|
||||||
|
console.log(`cwd="${process.cwd()}" ROOT="${ROOT}"`);
|
||||||
|
console.log('=======================================================');
|
||||||
|
const oclifPath = path.join(ROOT, 'node_modules', 'oclif');
|
||||||
|
await oclifRun([packCmd].concat(...packOpts), oclifPath);
|
||||||
|
await renameStandalone();
|
||||||
|
|
||||||
|
console.log(`Standalone tarball package build completed`);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error(`Error creating or testing standalone zip package`);
|
console.error(`Error creating or testing standalone tarball package`);
|
||||||
throw error;
|
throw error;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async function renameInstallerFiles() {
|
async function renameInstallers() {
|
||||||
const oclifInstallers = await getOclifInstallersOriginalNames();
|
const oclifInstallers = await getOclifInstallersOriginalNames();
|
||||||
if (await fs.pathExists(oclifInstallers[process.platform])) {
|
if (await fs.pathExists(oclifInstallers[process.platform])) {
|
||||||
await fs.rename(
|
await fs.rename(
|
||||||
@ -439,6 +208,16 @@ async function renameInstallerFiles() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function renameStandalone() {
|
||||||
|
const oclifStandalone = await getOclifStandaloneOriginalNames();
|
||||||
|
if (await fs.pathExists(oclifStandalone[process.platform])) {
|
||||||
|
await fs.rename(
|
||||||
|
oclifStandalone[process.platform],
|
||||||
|
renamedOclifStandalone[process.platform],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* If the CSC_LINK and CSC_KEY_PASSWORD env vars are set, digitally sign the
|
* If the CSC_LINK and CSC_KEY_PASSWORD env vars are set, digitally sign the
|
||||||
* executable installer using Microsoft SignTool.exe (Sign Tool)
|
* executable installer using Microsoft SignTool.exe (Sign Tool)
|
||||||
@ -446,7 +225,7 @@ async function renameInstallerFiles() {
|
|||||||
*/
|
*/
|
||||||
async function signWindowsInstaller() {
|
async function signWindowsInstaller() {
|
||||||
if (process.env.SM_CODE_SIGNING_CERT_SHA1_HASH) {
|
if (process.env.SM_CODE_SIGNING_CERT_SHA1_HASH) {
|
||||||
const exeName = renamedOclifInstallers[process.platform];
|
const exeName = (await getOclifInstallersOriginalNames())[process.platform];
|
||||||
console.log(`Signing installer "${exeName}"`);
|
console.log(`Signing installer "${exeName}"`);
|
||||||
// trust ...
|
// trust ...
|
||||||
await execFileAsync('signtool.exe', [
|
await execFileAsync('signtool.exe', [
|
||||||
@ -480,12 +259,14 @@ async function notarizeMacInstaller(): Promise<void> {
|
|||||||
const appleId =
|
const appleId =
|
||||||
process.env.XCODE_APP_LOADER_EMAIL || 'accounts+apple@balena.io';
|
process.env.XCODE_APP_LOADER_EMAIL || 'accounts+apple@balena.io';
|
||||||
const appleIdPassword = process.env.XCODE_APP_LOADER_PASSWORD;
|
const appleIdPassword = process.env.XCODE_APP_LOADER_PASSWORD;
|
||||||
|
const appPath = (await getOclifInstallersOriginalNames())[process.platform];
|
||||||
|
console.log(`Notarizing file "${appPath}"`);
|
||||||
|
|
||||||
if (appleIdPassword && teamId) {
|
if (appleIdPassword && teamId) {
|
||||||
await notarize({
|
await notarize({
|
||||||
tool: 'notarytool',
|
tool: 'notarytool',
|
||||||
teamId,
|
teamId,
|
||||||
appPath: renamedOclifInstallers.darwin,
|
appPath,
|
||||||
appleId,
|
appleId,
|
||||||
appleIdPassword,
|
appleIdPassword,
|
||||||
});
|
});
|
||||||
@ -525,7 +306,6 @@ export async function buildOclifInstaller() {
|
|||||||
console.log('=======================================================');
|
console.log('=======================================================');
|
||||||
const oclifPath = path.join(ROOT, 'node_modules', 'oclif');
|
const oclifPath = path.join(ROOT, 'node_modules', 'oclif');
|
||||||
await oclifRun([packCmd].concat(...packOpts), oclifPath);
|
await oclifRun([packCmd].concat(...packOpts), oclifPath);
|
||||||
await renameInstallerFiles();
|
|
||||||
// The Windows installer is explicitly signed here (oclif doesn't do it).
|
// The Windows installer is explicitly signed here (oclif doesn't do it).
|
||||||
// The macOS installer is automatically signed by oclif (which runs the
|
// The macOS installer is automatically signed by oclif (which runs the
|
||||||
// `pkgbuild` tool), using the certificate name given in package.json
|
// `pkgbuild` tool), using the certificate name given in package.json
|
||||||
@ -537,6 +317,7 @@ export async function buildOclifInstaller() {
|
|||||||
await notarizeMacInstaller(); // Notarize
|
await notarizeMacInstaller(); // Notarize
|
||||||
console.log('Package notarized.');
|
console.log('Package notarized.');
|
||||||
}
|
}
|
||||||
|
await renameInstallers();
|
||||||
console.log(`oclif installer build completed`);
|
console.log(`oclif installer build completed`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -572,4 +353,5 @@ export async function testShrinkwrap(): Promise<void> {
|
|||||||
if (process.platform !== 'win32') {
|
if (process.platform !== 'win32') {
|
||||||
await whichSpawn(path.resolve(__dirname, 'test-lock-deduplicated.sh'));
|
await whichSpawn(path.resolve(__dirname, 'test-lock-deduplicated.sh'));
|
||||||
}
|
}
|
||||||
|
await Promise.resolve();
|
||||||
}
|
}
|
||||||
|
@ -58,12 +58,9 @@ const commandHeadings: { [key: string]: string } = {
|
|||||||
env: 'Environment Variables',
|
env: 'Environment Variables',
|
||||||
help: 'Help and Version',
|
help: 'Help and Version',
|
||||||
'ssh-key': 'SSH Keys',
|
'ssh-key': 'SSH Keys',
|
||||||
orgs: 'Organizations',
|
organization: 'Organizations',
|
||||||
os: 'OS',
|
os: 'OS',
|
||||||
util: 'Utilities',
|
util: 'Utilities',
|
||||||
ssh: 'Network',
|
|
||||||
scan: 'Network',
|
|
||||||
tunnel: 'Network',
|
|
||||||
build: 'Deploy',
|
build: 'Deploy',
|
||||||
join: 'Platform',
|
join: 'Platform',
|
||||||
leave: 'Platform',
|
leave: 'Platform',
|
||||||
@ -148,7 +145,7 @@ export async function getCapitanoDoc(): Promise<typeof capitanoDoc> {
|
|||||||
throw new Error(`Error parsing section title`);
|
throw new Error(`Error parsing section title`);
|
||||||
}
|
}
|
||||||
// match[1] has the title, match[2] has the rest
|
// match[1] has the title, match[2] has the rest
|
||||||
return match && match[2];
|
return match?.[2];
|
||||||
}),
|
}),
|
||||||
mdParser.getSectionOfTitle('Installation'),
|
mdParser.getSectionOfTitle('Installation'),
|
||||||
mdParser.getSectionOfTitle('Choosing a shell (command prompt/terminal)'),
|
mdParser.getSectionOfTitle('Choosing a shell (command prompt/terminal)'),
|
||||||
|
@ -25,7 +25,14 @@ function renderOclifCommand(command: Category['commands'][0]): string[] {
|
|||||||
const result = [`## ${ent.encode(command.name || '')}`];
|
const result = [`## ${ent.encode(command.name || '')}`];
|
||||||
if (command.aliases?.length) {
|
if (command.aliases?.length) {
|
||||||
result.push('### Aliases');
|
result.push('### Aliases');
|
||||||
result.push(command.aliases.map((alias) => `- \`${alias}\``).join('\n'));
|
result.push(
|
||||||
|
command.aliases
|
||||||
|
.map(
|
||||||
|
(alias) =>
|
||||||
|
`- \`${alias}\`${command.deprecateAliases ? ' *(deprecated)*' : ''}`,
|
||||||
|
)
|
||||||
|
.join('\n'),
|
||||||
|
);
|
||||||
result.push(
|
result.push(
|
||||||
`\nTo use one of the aliases, replace \`${command.name}\` with the alias.`,
|
`\nTo use one of the aliases, replace \`${command.name}\` with the alias.`,
|
||||||
);
|
);
|
||||||
|
@ -19,7 +19,7 @@ import * as _ from 'lodash';
|
|||||||
|
|
||||||
import {
|
import {
|
||||||
buildOclifInstaller,
|
buildOclifInstaller,
|
||||||
buildStandaloneZip,
|
buildStandalone,
|
||||||
catchUncommitted,
|
catchUncommitted,
|
||||||
signFilesForNotarization,
|
signFilesForNotarization,
|
||||||
testShrinkwrap,
|
testShrinkwrap,
|
||||||
@ -36,7 +36,7 @@ process.env.DEBUG = ['0', 'no', 'false', '', undefined].includes(
|
|||||||
* Trivial command-line parser. Check whether the command-line argument is one
|
* Trivial command-line parser. Check whether the command-line argument is one
|
||||||
* of the following strings, then call the appropriate functions:
|
* of the following strings, then call the appropriate functions:
|
||||||
* 'build:installer' (to build a native oclif installer)
|
* 'build:installer' (to build a native oclif installer)
|
||||||
* 'build:standalone' (to build a standalone pkg package)
|
* 'build:standalone' (to build a standalone package)
|
||||||
*
|
*
|
||||||
* @param args Arguments to parse (default is process.argv.slice(2))
|
* @param args Arguments to parse (default is process.argv.slice(2))
|
||||||
*/
|
*/
|
||||||
@ -49,7 +49,7 @@ async function parse(args?: string[]) {
|
|||||||
}
|
}
|
||||||
const commands: { [cmd: string]: () => void | Promise<void> } = {
|
const commands: { [cmd: string]: () => void | Promise<void> } = {
|
||||||
'build:installer': buildOclifInstaller,
|
'build:installer': buildOclifInstaller,
|
||||||
'build:standalone': buildStandaloneZip,
|
'build:standalone': buildStandalone,
|
||||||
'sign:binaries': signFilesForNotarization,
|
'sign:binaries': signFilesForNotarization,
|
||||||
'catch-uncommitted': catchUncommitted,
|
'catch-uncommitted': catchUncommitted,
|
||||||
'test-shrinkwrap': testShrinkwrap,
|
'test-shrinkwrap': testShrinkwrap,
|
||||||
|
@ -3,7 +3,7 @@ import * as semver from 'semver';
|
|||||||
|
|
||||||
const changeTypes = ['major', 'minor', 'patch'] as const;
|
const changeTypes = ['major', 'minor', 'patch'] as const;
|
||||||
|
|
||||||
const validateChangeType = (maybeChangeType: string = 'minor') => {
|
const validateChangeType = (maybeChangeType = 'minor') => {
|
||||||
maybeChangeType = maybeChangeType.toLowerCase();
|
maybeChangeType = maybeChangeType.toLowerCase();
|
||||||
switch (maybeChangeType) {
|
switch (maybeChangeType) {
|
||||||
case 'patch':
|
case 'patch':
|
||||||
@ -136,5 +136,4 @@ async function main() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// eslint-disable-next-line @typescript-eslint/no-floating-promises
|
void main();
|
||||||
main();
|
|
||||||
|
@ -18,72 +18,10 @@
|
|||||||
import { spawn } from 'child_process';
|
import { spawn } from 'child_process';
|
||||||
import * as path from 'path';
|
import * as path from 'path';
|
||||||
import * as fs from 'fs';
|
import * as fs from 'fs';
|
||||||
import { diffTrimmedLines } from 'diff';
|
import * as whichMod from 'which';
|
||||||
|
|
||||||
export const ROOT = path.join(__dirname, '..');
|
export const ROOT = path.join(__dirname, '..');
|
||||||
|
|
||||||
/** Tap and buffer this process' stdout and stderr */
|
|
||||||
export class StdOutTap {
|
|
||||||
public stdoutBuf: string[] = [];
|
|
||||||
public stderrBuf: string[] = [];
|
|
||||||
public allBuf: string[] = []; // both stdout and stderr
|
|
||||||
|
|
||||||
protected origStdoutWrite: typeof process.stdout.write;
|
|
||||||
protected origStderrWrite: typeof process.stdout.write;
|
|
||||||
|
|
||||||
constructor(protected printDots = false) {}
|
|
||||||
|
|
||||||
tap() {
|
|
||||||
this.origStdoutWrite = process.stdout.write;
|
|
||||||
this.origStderrWrite = process.stderr.write;
|
|
||||||
|
|
||||||
process.stdout.write = (chunk: string, ...args: any[]): boolean => {
|
|
||||||
this.stdoutBuf.push(chunk);
|
|
||||||
this.allBuf.push(chunk);
|
|
||||||
const str = this.printDots ? '.' : chunk;
|
|
||||||
return this.origStdoutWrite.call(process.stdout, str, ...args);
|
|
||||||
};
|
|
||||||
|
|
||||||
process.stderr.write = (chunk: string, ...args: any[]): boolean => {
|
|
||||||
this.stderrBuf.push(chunk);
|
|
||||||
this.allBuf.push(chunk);
|
|
||||||
const str = this.printDots ? '.' : chunk;
|
|
||||||
return this.origStderrWrite.call(process.stderr, str, ...args);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
untap() {
|
|
||||||
process.stdout.write = this.origStdoutWrite;
|
|
||||||
process.stderr.write = this.origStderrWrite;
|
|
||||||
if (this.printDots) {
|
|
||||||
console.error('');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Diff strings by line, using the 'diff' npm package:
|
|
||||||
* https://www.npmjs.com/package/diff
|
|
||||||
*/
|
|
||||||
export function diffLines(str1: string, str2: string): string {
|
|
||||||
const diffObjs = diffTrimmedLines(str1, str2);
|
|
||||||
const prefix = (chunk: string, char: string) =>
|
|
||||||
chunk
|
|
||||||
.split('\n')
|
|
||||||
.map((line: string) => `${char} ${line}`)
|
|
||||||
.join('\n');
|
|
||||||
const diffStr = diffObjs
|
|
||||||
.map((part: any) => {
|
|
||||||
return part.added
|
|
||||||
? prefix(part.value, '+')
|
|
||||||
: part.removed
|
|
||||||
? prefix(part.value, '-')
|
|
||||||
: prefix(part.value, ' ');
|
|
||||||
})
|
|
||||||
.join('\n');
|
|
||||||
return diffStr;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function loadPackageJson() {
|
export function loadPackageJson() {
|
||||||
const packageJsonPath = path.join(ROOT, 'package.json');
|
const packageJsonPath = path.join(ROOT, 'package.json');
|
||||||
|
|
||||||
@ -101,7 +39,6 @@ export function loadPackageJson() {
|
|||||||
* @returns The program's full path, e.g. 'C:\WINDOWS\System32\OpenSSH\ssh.EXE'
|
* @returns The program's full path, e.g. 'C:\WINDOWS\System32\OpenSSH\ssh.EXE'
|
||||||
*/
|
*/
|
||||||
export async function which(program: string): Promise<string> {
|
export async function which(program: string): Promise<string> {
|
||||||
const whichMod = await import('which');
|
|
||||||
let programPath: string;
|
let programPath: string;
|
||||||
try {
|
try {
|
||||||
programPath = await whichMod(program);
|
programPath = await whichMod(program);
|
||||||
@ -132,7 +69,7 @@ export async function whichSpawn(
|
|||||||
.on('error', reject)
|
.on('error', reject)
|
||||||
.on('close', resolve);
|
.on('close', resolve);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
reject(err);
|
reject(err as Error);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
|
@ -57,7 +57,10 @@ require('ts-node').register({
|
|||||||
project: path.join(rootDir, 'tsconfig.json'),
|
project: path.join(rootDir, 'tsconfig.json'),
|
||||||
transpileOnly: true,
|
transpileOnly: true,
|
||||||
});
|
});
|
||||||
require('../src/app').run(undefined, { dir: __dirname, development: true });
|
void require('../src/app').run(undefined, {
|
||||||
|
dir: __dirname,
|
||||||
|
development: true,
|
||||||
|
});
|
||||||
|
|
||||||
// Modify package.json oclif paths from build/ -> src/, or vice versa
|
// Modify package.json oclif paths from build/ -> src/, or vice versa
|
||||||
function modifyOclifPaths(revert) {
|
function modifyOclifPaths(revert) {
|
||||||
|
@ -5,7 +5,7 @@
|
|||||||
process.env.UV_THREADPOOL_SIZE = '64';
|
process.env.UV_THREADPOOL_SIZE = '64';
|
||||||
|
|
||||||
// Disable oclif registering ts-node
|
// Disable oclif registering ts-node
|
||||||
process.env.OCLIF_TS_NODE = 0;
|
process.env.OCLIF_TS_NODE = '0';
|
||||||
|
|
||||||
async function run() {
|
async function run() {
|
||||||
// Use fast-boot to cache require lookups, speeding up startup
|
// Use fast-boot to cache require lookups, speeding up startup
|
||||||
@ -18,4 +18,4 @@ async function run() {
|
|||||||
await require('../build/app').run(undefined, { dir: __dirname });
|
await require('../build/app').run(undefined, { dir: __dirname });
|
||||||
}
|
}
|
||||||
|
|
||||||
run();
|
void run();
|
||||||
|
@ -8,27 +8,28 @@ _balena() {
|
|||||||
local context state line curcontext="$curcontext"
|
local context state line curcontext="$curcontext"
|
||||||
|
|
||||||
# Valid top-level completions
|
# Valid top-level completions
|
||||||
main_commands=( api-key app block build config deploy device devices env fleet internal join leave local login logout logs notes orgs os preload push release scan settings ssh ssh-key support tag tags tunnel util version whoami )
|
main_commands=( api-key app block build config deploy device device-type env fleet internal join leave local login logout organization os preload push release settings ssh-key support tag util version whoami )
|
||||||
# Sub-completions
|
# Sub-completions
|
||||||
api_key_cmds=( generate list revoke )
|
api_key_cmds=( generate list revoke )
|
||||||
app_cmds=( create )
|
app_cmds=( create )
|
||||||
block_cmds=( create )
|
block_cmds=( create )
|
||||||
config_cmds=( generate inject read reconfigure write )
|
config_cmds=( generate inject read reconfigure write )
|
||||||
device_cmds=( deactivate identify init list local-mode move os-update pin public-url purge reboot register rename restart rm shutdown start-service stop-service track-fleet )
|
device_type_cmds=( list )
|
||||||
devices_cmds=( supported )
|
device_cmds=( deactivate detect identify init list local-mode logs move note os-update pin public-url purge reboot register rename restart rm shutdown ssh start-service stop-service track-fleet tunnel )
|
||||||
env_cmds=( add list rename rm )
|
env_cmds=( list rename rm set )
|
||||||
fleet_cmds=( create list pin purge rename restart rm track-latest )
|
fleet_cmds=( create list pin purge rename restart rm track-latest )
|
||||||
internal_cmds=( osinit )
|
internal_cmds=( osinit )
|
||||||
local_cmds=( configure flash )
|
local_cmds=( configure flash )
|
||||||
|
organization_cmds=( list )
|
||||||
os_cmds=( build-config configure download initialize versions )
|
os_cmds=( build-config configure download initialize versions )
|
||||||
release_cmds=( finalize invalidate list validate )
|
release_cmds=( finalize invalidate list validate )
|
||||||
ssh_key_cmds=( add list rm )
|
ssh_key_cmds=( add list rm )
|
||||||
tag_cmds=( rm set )
|
tag_cmds=( list rm set )
|
||||||
|
|
||||||
|
|
||||||
_arguments -C \
|
_arguments -C \
|
||||||
'(- 1 *)--version[show version and exit]' \
|
'(- 1 *)--version[show version and exit]' \
|
||||||
'(- 1 *)'{-h,--help}'[show help options and exit]' \
|
'(- 1 *)--help[show help options and exit]' \
|
||||||
'1:first command:_balena_main_cmds' \
|
'1:first command:_balena_main_cmds' \
|
||||||
'2:second command:_balena_sec_cmds' \
|
'2:second command:_balena_sec_cmds' \
|
||||||
&& ret=0
|
&& ret=0
|
||||||
@ -54,12 +55,12 @@ _balena_sec_cmds() {
|
|||||||
"config")
|
"config")
|
||||||
_describe -t config_cmds 'config_cmd' config_cmds "$@" && ret=0
|
_describe -t config_cmds 'config_cmd' config_cmds "$@" && ret=0
|
||||||
;;
|
;;
|
||||||
|
"device-type")
|
||||||
|
_describe -t device_type_cmds 'device-type_cmd' device_type_cmds "$@" && ret=0
|
||||||
|
;;
|
||||||
"device")
|
"device")
|
||||||
_describe -t device_cmds 'device_cmd' device_cmds "$@" && ret=0
|
_describe -t device_cmds 'device_cmd' device_cmds "$@" && ret=0
|
||||||
;;
|
;;
|
||||||
"devices")
|
|
||||||
_describe -t devices_cmds 'devices_cmd' devices_cmds "$@" && ret=0
|
|
||||||
;;
|
|
||||||
"env")
|
"env")
|
||||||
_describe -t env_cmds 'env_cmd' env_cmds "$@" && ret=0
|
_describe -t env_cmds 'env_cmd' env_cmds "$@" && ret=0
|
||||||
;;
|
;;
|
||||||
@ -72,6 +73,9 @@ _balena_sec_cmds() {
|
|||||||
"local")
|
"local")
|
||||||
_describe -t local_cmds 'local_cmd' local_cmds "$@" && ret=0
|
_describe -t local_cmds 'local_cmd' local_cmds "$@" && ret=0
|
||||||
;;
|
;;
|
||||||
|
"organization")
|
||||||
|
_describe -t organization_cmds 'organization_cmd' organization_cmds "$@" && ret=0
|
||||||
|
;;
|
||||||
"os")
|
"os")
|
||||||
_describe -t os_cmds 'os_cmd' os_cmds "$@" && ret=0
|
_describe -t os_cmds 'os_cmd' os_cmds "$@" && ret=0
|
||||||
;;
|
;;
|
||||||
|
@ -7,22 +7,23 @@ _balena_complete()
|
|||||||
local cur prev
|
local cur prev
|
||||||
|
|
||||||
# Valid top-level completions
|
# Valid top-level completions
|
||||||
main_commands="api-key app block build config deploy device devices env fleet internal join leave local login logout logs notes orgs os preload push release scan settings ssh ssh-key support tag tags tunnel util version whoami"
|
main_commands="api-key app block build config deploy device device-type env fleet internal join leave local login logout organization os preload push release settings ssh-key support tag util version whoami"
|
||||||
# Sub-completions
|
# Sub-completions
|
||||||
api_key_cmds="generate list revoke"
|
api_key_cmds="generate list revoke"
|
||||||
app_cmds="create"
|
app_cmds="create"
|
||||||
block_cmds="create"
|
block_cmds="create"
|
||||||
config_cmds="generate inject read reconfigure write"
|
config_cmds="generate inject read reconfigure write"
|
||||||
device_cmds="deactivate identify init list local-mode move os-update pin public-url purge reboot register rename restart rm shutdown start-service stop-service track-fleet"
|
device_type_cmds="list"
|
||||||
devices_cmds="supported"
|
device_cmds="deactivate detect identify init list local-mode logs move note os-update pin public-url purge reboot register rename restart rm shutdown ssh start-service stop-service track-fleet tunnel"
|
||||||
env_cmds="add list rename rm"
|
env_cmds="list rename rm set"
|
||||||
fleet_cmds="create list pin purge rename restart rm track-latest"
|
fleet_cmds="create list pin purge rename restart rm track-latest"
|
||||||
internal_cmds="osinit"
|
internal_cmds="osinit"
|
||||||
local_cmds="configure flash"
|
local_cmds="configure flash"
|
||||||
|
organization_cmds="list"
|
||||||
os_cmds="build-config configure download initialize versions"
|
os_cmds="build-config configure download initialize versions"
|
||||||
release_cmds="finalize invalidate list validate"
|
release_cmds="finalize invalidate list validate"
|
||||||
ssh_key_cmds="add list rm"
|
ssh_key_cmds="add list rm"
|
||||||
tag_cmds="rm set"
|
tag_cmds="list rm set"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@ -48,12 +49,12 @@ _balena_complete()
|
|||||||
config)
|
config)
|
||||||
COMPREPLY=( $(compgen -W "$config_cmds" -- $cur) )
|
COMPREPLY=( $(compgen -W "$config_cmds" -- $cur) )
|
||||||
;;
|
;;
|
||||||
|
device-type)
|
||||||
|
COMPREPLY=( $(compgen -W "$device_type_cmds" -- $cur) )
|
||||||
|
;;
|
||||||
device)
|
device)
|
||||||
COMPREPLY=( $(compgen -W "$device_cmds" -- $cur) )
|
COMPREPLY=( $(compgen -W "$device_cmds" -- $cur) )
|
||||||
;;
|
;;
|
||||||
devices)
|
|
||||||
COMPREPLY=( $(compgen -W "$devices_cmds" -- $cur) )
|
|
||||||
;;
|
|
||||||
env)
|
env)
|
||||||
COMPREPLY=( $(compgen -W "$env_cmds" -- $cur) )
|
COMPREPLY=( $(compgen -W "$env_cmds" -- $cur) )
|
||||||
;;
|
;;
|
||||||
@ -66,6 +67,9 @@ _balena_complete()
|
|||||||
local)
|
local)
|
||||||
COMPREPLY=( $(compgen -W "$local_cmds" -- $cur) )
|
COMPREPLY=( $(compgen -W "$local_cmds" -- $cur) )
|
||||||
;;
|
;;
|
||||||
|
organization)
|
||||||
|
COMPREPLY=( $(compgen -W "$organization_cmds" -- $cur) )
|
||||||
|
;;
|
||||||
os)
|
os)
|
||||||
COMPREPLY=( $(compgen -W "$os_cmds" -- $cur) )
|
COMPREPLY=( $(compgen -W "$os_cmds" -- $cur) )
|
||||||
;;
|
;;
|
||||||
|
@ -14,7 +14,7 @@ $sub_cmds$
|
|||||||
|
|
||||||
_arguments -C \
|
_arguments -C \
|
||||||
'(- 1 *)--version[show version and exit]' \
|
'(- 1 *)--version[show version and exit]' \
|
||||||
'(- 1 *)'{-h,--help}'[show help options and exit]' \
|
'(- 1 *)--help[show help options and exit]' \
|
||||||
'1:first command:_balena_main_cmds' \
|
'1:first command:_balena_main_cmds' \
|
||||||
'2:second command:_balena_sec_cmds' \
|
'2:second command:_balena_sec_cmds' \
|
||||||
&& ret=0
|
&& ret=0
|
||||||
|
File diff suppressed because it is too large
Load Diff
32
eslint.config.js
Normal file
32
eslint.config.js
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
const { FlatCompat } = require('@eslint/eslintrc');
|
||||||
|
|
||||||
|
const compat = new FlatCompat({
|
||||||
|
baseDirectory: __dirname,
|
||||||
|
});
|
||||||
|
module.exports = [
|
||||||
|
...require('@balena/lint/config/eslint.config'),
|
||||||
|
...compat.config({
|
||||||
|
parserOptions: {
|
||||||
|
project: 'tsconfig.dev.json',
|
||||||
|
},
|
||||||
|
ignorePatterns: ['**/generate-completion.js', '**/bin/**/*'],
|
||||||
|
rules: {
|
||||||
|
ignoreDefinitionFiles: 0,
|
||||||
|
'@typescript-eslint/no-non-null-assertion': 'off',
|
||||||
|
'@typescript-eslint/no-shadow': 'off',
|
||||||
|
'@typescript-eslint/no-var-requires': 'off',
|
||||||
|
'@typescript-eslint/no-require-imports': 'off',
|
||||||
|
'@typescript-eslint/no-unnecessary-type-assertion': 'off',
|
||||||
|
'@typescript-eslint/prefer-nullish-coalescing': 'warn',
|
||||||
|
|
||||||
|
'no-restricted-imports': ['error', {
|
||||||
|
paths: ['resin-cli-visuals', 'chalk', 'common-tags', 'resin-cli-form'],
|
||||||
|
}],
|
||||||
|
|
||||||
|
'@typescript-eslint/no-unused-vars': ['error', {
|
||||||
|
argsIgnorePattern: '^_',
|
||||||
|
caughtErrorsIgnorePattern: '^_',
|
||||||
|
}],
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
];
|
9533
npm-shrinkwrap.json
generated
9533
npm-shrinkwrap.json
generated
File diff suppressed because it is too large
Load Diff
75
package.json
75
package.json
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "balena-cli",
|
"name": "balena-cli",
|
||||||
"version": "19.7.0",
|
"version": "22.1.1",
|
||||||
"description": "The official balena Command Line Interface",
|
"description": "The official balena Command Line Interface",
|
||||||
"main": "./build/app.js",
|
"main": "./build/app.js",
|
||||||
"homepage": "https://github.com/balena-io/balena-cli",
|
"homepage": "https://github.com/balena-io/balena-cli",
|
||||||
@ -24,26 +24,6 @@
|
|||||||
"bin": {
|
"bin": {
|
||||||
"balena": "./bin/run.js"
|
"balena": "./bin/run.js"
|
||||||
},
|
},
|
||||||
"pkg": {
|
|
||||||
"scripts": [
|
|
||||||
"build/**/*.js",
|
|
||||||
"node_modules/balena-sdk/es2018/index.js",
|
|
||||||
"node_modules/pinejs-client-request/node_modules/pinejs-client-core/es2018/index.js",
|
|
||||||
"node_modules/@balena/compose/dist/parse/schemas/*.json"
|
|
||||||
],
|
|
||||||
"assets": [
|
|
||||||
"build/auth/pages/*.ejs",
|
|
||||||
"node_modules/balena-sdk/node_modules/balena-pine/**/*",
|
|
||||||
"node_modules/balena-pine/**/*",
|
|
||||||
"node_modules/pinejs-client-core/**/*",
|
|
||||||
"node_modules/open/xdg-open",
|
|
||||||
"node_modules/windosu/*.bat",
|
|
||||||
"node_modules/windosu/*.cmd",
|
|
||||||
"node_modules/axios/**/*",
|
|
||||||
"npm-shrinkwrap.json",
|
|
||||||
"oclif.manifest.json"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"postinstall": "node patches/apply-patches.js",
|
"postinstall": "node patches/apply-patches.js",
|
||||||
"prebuild": "rimraf build/ build-bin/",
|
"prebuild": "rimraf build/ build-bin/",
|
||||||
@ -58,6 +38,7 @@
|
|||||||
"build:completion": "node completion/generate-completion.js",
|
"build:completion": "node completion/generate-completion.js",
|
||||||
"build:standalone": "ts-node --transpile-only automation/run.ts build:standalone",
|
"build:standalone": "ts-node --transpile-only automation/run.ts build:standalone",
|
||||||
"build:installer": "ts-node --transpile-only automation/run.ts build:installer",
|
"build:installer": "ts-node --transpile-only automation/run.ts build:installer",
|
||||||
|
"deduplicate-dependencies": "npm dd && git add npm-shrinkwrap.json && git commit --message \"Deduplicate dependencies\"",
|
||||||
"package": "npm run build:fast && npm run build:standalone && npm run build:installer",
|
"package": "npm run build:fast && npm run build:standalone && npm run build:installer",
|
||||||
"pretest": "npm run build",
|
"pretest": "npm run build",
|
||||||
"test": "npm run test:shrinkwrap && npm run test:core",
|
"test": "npm run test:shrinkwrap && npm run test:core",
|
||||||
@ -91,7 +72,7 @@
|
|||||||
"author": "Balena Inc. (https://balena.io/)",
|
"author": "Balena Inc. (https://balena.io/)",
|
||||||
"license": "Apache-2.0",
|
"license": "Apache-2.0",
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": "^20.6.0"
|
"node": ">=20.6.0 <23"
|
||||||
},
|
},
|
||||||
"oclif": {
|
"oclif": {
|
||||||
"bin": "balena",
|
"bin": "balena",
|
||||||
@ -111,16 +92,14 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@balena/lint": "^8.0.0",
|
"@balena/lint": "^9.1.3",
|
||||||
"@electron/notarize": "^2.0.0",
|
"@electron/notarize": "^2.0.0",
|
||||||
"@types/archiver": "^6.0.2",
|
|
||||||
"@types/bluebird": "^3.5.36",
|
"@types/bluebird": "^3.5.36",
|
||||||
"@types/body-parser": "^1.19.2",
|
"@types/body-parser": "^1.19.2",
|
||||||
"@types/chai": "^4.3.0",
|
"@types/chai": "^4.3.0",
|
||||||
"@types/chai-as-promised": "^7.1.4",
|
"@types/chai-as-promised": "^7.1.4",
|
||||||
"@types/cli-truncate": "^2.0.0",
|
"@types/cli-truncate": "^2.0.0",
|
||||||
"@types/common-tags": "^1.8.1",
|
"@types/common-tags": "^1.8.1",
|
||||||
"@types/diff": "^5.0.3",
|
|
||||||
"@types/dockerode": "3.3.23",
|
"@types/dockerode": "3.3.23",
|
||||||
"@types/ejs": "^3.1.0",
|
"@types/ejs": "^3.1.0",
|
||||||
"@types/express": "^4.17.13",
|
"@types/express": "^4.17.13",
|
||||||
@ -145,7 +124,6 @@
|
|||||||
"@types/node-cleanup": "^2.1.2",
|
"@types/node-cleanup": "^2.1.2",
|
||||||
"@types/prettyjson": "^0.0.33",
|
"@types/prettyjson": "^0.0.33",
|
||||||
"@types/progress-stream": "^2.0.2",
|
"@types/progress-stream": "^2.0.2",
|
||||||
"@types/request": "^2.48.7",
|
|
||||||
"@types/rewire": "^2.5.30",
|
"@types/rewire": "^2.5.30",
|
||||||
"@types/rimraf": "^3.0.2",
|
"@types/rimraf": "^3.0.2",
|
||||||
"@types/semver": "^7.3.9",
|
"@types/semver": "^7.3.9",
|
||||||
@ -159,16 +137,12 @@
|
|||||||
"@types/update-notifier": "^4.1.1",
|
"@types/update-notifier": "^4.1.1",
|
||||||
"@types/which": "^2.0.1",
|
"@types/which": "^2.0.1",
|
||||||
"@types/window-size": "^1.1.1",
|
"@types/window-size": "^1.1.1",
|
||||||
"@yao-pkg/pkg": "^5.11.1",
|
|
||||||
"archiver": "^7.0.1",
|
|
||||||
"catch-uncommitted": "^2.0.0",
|
"catch-uncommitted": "^2.0.0",
|
||||||
"chai": "^4.3.4",
|
"chai": "^4.3.4",
|
||||||
"chai-as-promised": "^7.1.1",
|
"chai-as-promised": "^7.1.1",
|
||||||
"cross-env": "^7.0.3",
|
"cross-env": "^7.0.3",
|
||||||
"deep-object-diff": "^1.1.0",
|
"deep-object-diff": "^1.1.0",
|
||||||
"diff": "^5.0.0",
|
|
||||||
"ent": "^2.2.0",
|
"ent": "^2.2.0",
|
||||||
"filehound": "^1.17.5",
|
|
||||||
"fs-extra": "^11.2.0",
|
"fs-extra": "^11.2.0",
|
||||||
"http-proxy": "^1.18.1",
|
"http-proxy": "^1.18.1",
|
||||||
"husky": "^9.1.5",
|
"husky": "^9.1.5",
|
||||||
@ -179,44 +153,45 @@
|
|||||||
"mocha": "^10.6.0",
|
"mocha": "^10.6.0",
|
||||||
"mock-fs": "^5.2.0",
|
"mock-fs": "^5.2.0",
|
||||||
"mock-require": "^3.0.3",
|
"mock-require": "^3.0.3",
|
||||||
"nock": "^13.2.1",
|
"nock": "^14.0.4",
|
||||||
"oclif": "^4.14.0",
|
"oclif": "^4.17.0",
|
||||||
"rewire": "^7.0.0",
|
"rewire": "^7.0.0",
|
||||||
"simple-git": "^3.14.1",
|
"simple-git": "^3.14.1",
|
||||||
"sinon": "^18.0.0",
|
"sinon": "^19.0.0",
|
||||||
"string-to-stream": "^3.0.1",
|
"string-to-stream": "^3.0.1",
|
||||||
"ts-node": "^10.4.0",
|
"ts-node": "^10.4.0",
|
||||||
"typescript": "^5.6.2"
|
"typescript": "^5.8.2"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@balena/compose": "^4.0.1",
|
"@balena/compose": "^7.0.9",
|
||||||
"@balena/dockerignore": "^1.0.2",
|
"@balena/dockerignore": "^1.0.2",
|
||||||
"@balena/env-parsing": "^1.1.8",
|
"@balena/env-parsing": "^1.1.8",
|
||||||
"@balena/es-version": "^1.0.1",
|
"@balena/es-version": "^1.0.1",
|
||||||
"@oclif/core": "^4.0.8",
|
"@oclif/core": "^4.1.0",
|
||||||
"@sentry/node": "^6.16.1",
|
"@sentry/node": "^9.0.0",
|
||||||
"balena-config-json": "^4.2.0",
|
"balena-config-json": "^4.2.7",
|
||||||
"balena-device-init": "^7.0.1",
|
"balena-device-init": "^8.1.11",
|
||||||
"balena-errors": "^4.7.3",
|
"balena-errors": "^4.7.3",
|
||||||
"balena-image-fs": "^7.0.6",
|
"balena-image-fs": "^7.5.2",
|
||||||
"balena-preload": "^15.0.6",
|
"balena-preload": "^18.0.4",
|
||||||
"balena-sdk": "^19.7.3",
|
"balena-sdk": "^21.3.0",
|
||||||
"balena-semver": "^2.3.0",
|
"balena-semver": "^2.3.0",
|
||||||
"balena-settings-client": "^5.0.2",
|
"balena-settings-client": "^5.0.2",
|
||||||
"balena-settings-storage": "^8.1.0",
|
"balena-settings-storage": "^8.1.0",
|
||||||
"body-parser": "^1.19.1",
|
"body-parser": "^1.19.1",
|
||||||
"bonjour-service": "^1.2.1",
|
"bonjour-service": "^1.2.1",
|
||||||
"chalk": "^3.0.0",
|
"chalk": "^4.0.0",
|
||||||
"chokidar": "^3.5.2",
|
"chokidar": "^3.5.2",
|
||||||
"cli-truncate": "^2.1.0",
|
"cli-truncate": "^2.1.0",
|
||||||
"color-hash": "^1.1.1",
|
"color-hash": "^1.1.1",
|
||||||
"common-tags": "^1.7.2",
|
"common-tags": "^1.7.2",
|
||||||
|
"date-fns": "^4.1.0",
|
||||||
"denymount": "^2.3.0",
|
"denymount": "^2.3.0",
|
||||||
"docker-modem": "^5.0.3",
|
"docker-modem": "^5.0.6",
|
||||||
"docker-progress": "^5.1.3",
|
"docker-progress": "^5.1.3",
|
||||||
"dockerode": "^4.0.2",
|
"dockerode": "^4.0.5",
|
||||||
"ejs": "^3.1.6",
|
"ejs": "^3.1.6",
|
||||||
"etcher-sdk": "9.1.0",
|
"etcher-sdk": "^10.0.0",
|
||||||
"express": "^4.17.2",
|
"express": "^4.17.2",
|
||||||
"fast-boot2": "^1.1.0",
|
"fast-boot2": "^1.1.0",
|
||||||
"fast-levenshtein": "^3.0.0",
|
"fast-levenshtein": "^3.0.0",
|
||||||
@ -231,6 +206,7 @@
|
|||||||
"is-root": "^2.1.0",
|
"is-root": "^2.1.0",
|
||||||
"js-yaml": "^4.1.0",
|
"js-yaml": "^4.1.0",
|
||||||
"JSONStream": "^1.0.3",
|
"JSONStream": "^1.0.3",
|
||||||
|
"jwt-decode": "^3.1.2",
|
||||||
"livepush": "^3.5.1",
|
"livepush": "^3.5.1",
|
||||||
"lodash": "^4.17.21",
|
"lodash": "^4.17.21",
|
||||||
"mime": "^2.4.6",
|
"mime": "^2.4.6",
|
||||||
@ -243,9 +219,8 @@
|
|||||||
"prettyjson": "^1.2.5",
|
"prettyjson": "^1.2.5",
|
||||||
"progress-stream": "^2.0.0",
|
"progress-stream": "^2.0.0",
|
||||||
"reconfix": "^1.0.0-v0-1-0-fork-46760acff4d165f5238bfac5e464256ef1944476",
|
"reconfix": "^1.0.0-v0-1-0-fork-46760acff4d165f5238bfac5e464256ef1944476",
|
||||||
"request": "^2.88.2",
|
"resin-cli-form": "^4.0.0",
|
||||||
"resin-cli-form": "^3.0.0",
|
"resin-cli-visuals": "^3.0.0",
|
||||||
"resin-cli-visuals": "^2.0.1",
|
|
||||||
"resin-doodles": "^0.2.0",
|
"resin-doodles": "^0.2.0",
|
||||||
"resin-stream-logger": "^0.1.2",
|
"resin-stream-logger": "^0.1.2",
|
||||||
"rimraf": "^3.0.2",
|
"rimraf": "^3.0.2",
|
||||||
@ -273,6 +248,6 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"versionist": {
|
"versionist": {
|
||||||
"publishedAt": "2024-10-18T12:35:32.440Z"
|
"publishedAt": "2025-06-19T09:32:53.877Z"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
diff --git a/node_modules/@oclif/core/lib/help/command.js b/node_modules/@oclif/core/lib/help/command.js
|
diff --git a/node_modules/@oclif/core/lib/help/command.js b/node_modules/@oclif/core/lib/help/command.js
|
||||||
index 90922c8..6b7f417 100644
|
index 33105a0..0436982 100644
|
||||||
--- a/node_modules/@oclif/core/lib/help/command.js
|
--- a/node_modules/@oclif/core/lib/help/command.js
|
||||||
+++ b/node_modules/@oclif/core/lib/help/command.js
|
+++ b/node_modules/@oclif/core/lib/help/command.js
|
||||||
@@ -58,7 +58,8 @@ class CommandHelp extends formatter_1.HelpFormatter {
|
@@ -58,7 +58,8 @@ class CommandHelp extends formatter_1.HelpFormatter {
|
||||||
@ -13,10 +13,10 @@ index 90922c8..6b7f417 100644
|
|||||||
if (a.default)
|
if (a.default)
|
||||||
description = `${(0, theme_1.colorize)(this.config?.theme?.flagDefaultValue, `[default: ${a.default}]`)} ${description}`;
|
description = `${(0, theme_1.colorize)(this.config?.theme?.flagDefaultValue, `[default: ${a.default}]`)} ${description}`;
|
||||||
diff --git a/node_modules/@oclif/core/lib/help/index.js b/node_modules/@oclif/core/lib/help/index.js
|
diff --git a/node_modules/@oclif/core/lib/help/index.js b/node_modules/@oclif/core/lib/help/index.js
|
||||||
index 4a34b89..d7eb6ac 100644
|
index 0b48c0e..ff4fed4 100644
|
||||||
--- a/node_modules/@oclif/core/lib/help/index.js
|
--- a/node_modules/@oclif/core/lib/help/index.js
|
||||||
+++ b/node_modules/@oclif/core/lib/help/index.js
|
+++ b/node_modules/@oclif/core/lib/help/index.js
|
||||||
@@ -172,11 +172,12 @@ class Help extends HelpBase {
|
@@ -173,11 +173,12 @@ class Help extends HelpBase {
|
||||||
}
|
}
|
||||||
this.log(this.formatCommand(command));
|
this.log(this.formatCommand(command));
|
||||||
this.log('');
|
this.log('');
|
@ -1,8 +1,8 @@
|
|||||||
diff --git a/node_modules/oclif/lib/commands/pack/win.js b/node_modules/oclif/lib/commands/pack/win.js
|
diff --git a/node_modules/oclif/lib/commands/pack/win.js b/node_modules/oclif/lib/commands/pack/win.js
|
||||||
index ef7f90e..8264b7c 100644
|
index bfe9205..482519e 100644
|
||||||
--- a/node_modules/oclif/lib/commands/pack/win.js
|
--- a/node_modules/oclif/lib/commands/pack/win.js
|
||||||
+++ b/node_modules/oclif/lib/commands/pack/win.js
|
+++ b/node_modules/oclif/lib/commands/pack/win.js
|
||||||
@@ -76,6 +76,12 @@ InstallDir "\$PROGRAMFILES${arch === 'x64' ? '64' : ''}\\${config.dirname}"
|
@@ -86,6 +86,12 @@ InstallDir "\$PROGRAMFILES${arch === 'x64' ? '64' : ''}\\${config.dirname}"
|
||||||
${customization}
|
${customization}
|
||||||
|
|
||||||
Section "${config.name} CLI \${VERSION}"
|
Section "${config.name} CLI \${VERSION}"
|
||||||
@ -16,20 +16,18 @@ index ef7f90e..8264b7c 100644
|
|||||||
File /r bin
|
File /r bin
|
||||||
File /r client
|
File /r client
|
||||||
diff --git a/node_modules/oclif/lib/tarballs/build.js b/node_modules/oclif/lib/tarballs/build.js
|
diff --git a/node_modules/oclif/lib/tarballs/build.js b/node_modules/oclif/lib/tarballs/build.js
|
||||||
index 14d5a6e..7b42a6f 100644
|
index f0c8d95..a72400e 100644
|
||||||
--- a/node_modules/oclif/lib/tarballs/build.js
|
--- a/node_modules/oclif/lib/tarballs/build.js
|
||||||
+++ b/node_modules/oclif/lib/tarballs/build.js
|
+++ b/node_modules/oclif/lib/tarballs/build.js
|
||||||
@@ -200,6 +200,13 @@ const extractCLI = async (tarball, c) => {
|
@@ -218,6 +218,11 @@ const extractCLI = async (tarball, c) => {
|
||||||
(0, promises_1.rm)(path.join(workspace, path.basename(tarball)), { recursive: true }),
|
(0, promises_1.rm)(path.join(workspace, path.basename(tarball)), { recursive: true }),
|
||||||
(0, fs_extra_1.remove)(path.join(workspace, 'bin', 'run.cmd')),
|
(0, fs_extra_1.remove)(path.join(workspace, 'bin', 'run.cmd')),
|
||||||
]);
|
]);
|
||||||
+
|
|
||||||
+ // The oclif installers are a production installation, while the source
|
+ // The oclif installers are a production installation, while the source
|
||||||
+ // `bin` folder may contain a `.fast-boot.json` file of a dev installation.
|
+ // `bin` folder may contain a `.fast-boot.json` file of a dev installation.
|
||||||
+ // This has previously led to issues preventing the CLI from starting, so
|
+ // This has previously led to issues preventing the CLI from starting, so
|
||||||
+ // delete `.fast-boot.json` (if any) from the destination folder.
|
+ // delete `.fast-boot.json` (if any) from the destination folder.
|
||||||
+ await (0, fs_extra_1.remove)(path.join(workspace, 'bin', '.fast-boot.json'));
|
+ await (0, fs_extra_1.remove)(path.join(workspace, 'bin', '.fast-boot.json'));
|
||||||
+
|
|
||||||
};
|
};
|
||||||
const buildTarget = async (target, c, options) => {
|
const buildTarget = async (target, c, options) => {
|
||||||
const workspace = c.workspace(target);
|
if (target.platform === 'win32' && target.arch === 'arm64' && (0, semver_1.lt)(c.nodeVersion, '20.0.0')) {
|
@ -1,16 +0,0 @@
|
|||||||
diff --git a/node_modules/open/index.js b/node_modules/open/index.js
|
|
||||||
index 13147d0..ff161dd 100644
|
|
||||||
--- a/node_modules/open/index.js
|
|
||||||
+++ b/node_modules/open/index.js
|
|
||||||
@@ -10,7 +10,10 @@ const pAccess = promisify(fs.access);
|
|
||||||
const pReadFile = promisify(fs.readFile);
|
|
||||||
|
|
||||||
// Path to included `xdg-open`.
|
|
||||||
-const localXdgOpenPath = path.join(__dirname, 'xdg-open');
|
|
||||||
+const localXdgOpenPath = process.pkg
|
|
||||||
+ ? path.join(path.dirname(process.execPath), 'xdg-open')
|
|
||||||
+ : path.join(__dirname, 'xdg-open');
|
|
||||||
+
|
|
||||||
|
|
||||||
/**
|
|
||||||
Get the mount point for fixed drives in WSL.
|
|
@ -1,14 +0,0 @@
|
|||||||
diff --git a/node_modules/node-gyp-build/node-gyp-build.js b/node_modules/node-gyp-build/node-gyp-build.js
|
|
||||||
index 61b398e..3cc3be8 100644
|
|
||||||
--- a/node_modules/node-gyp-build/node-gyp-build.js
|
|
||||||
+++ b/node_modules/node-gyp-build/node-gyp-build.js
|
|
||||||
@@ -30,6 +30,9 @@ load.resolve = load.path = function (dir) {
|
|
||||||
if (process.env[name + '_PREBUILD']) dir = process.env[name + '_PREBUILD']
|
|
||||||
} catch (err) {}
|
|
||||||
|
|
||||||
+ // pkg fix: native node modules are located externally to the pkg executable
|
|
||||||
+ dir = dir.replace(/^\/snapshot\/.+?\/node_modules\//, path.dirname(process.execPath) + path.sep)
|
|
||||||
+
|
|
||||||
if (!prebuildsOnly) {
|
|
||||||
var release = getFirst(path.join(dir, 'build/Release'), matchBuild)
|
|
||||||
if (release) return release
|
|
@ -1,38 +0,0 @@
|
|||||||
diff --git a/node_modules/windosu/lib/pipe.js b/node_modules/windosu/lib/pipe.js
|
|
||||||
index dc81fa5..a381cc7 100644
|
|
||||||
--- a/node_modules/windosu/lib/pipe.js
|
|
||||||
+++ b/node_modules/windosu/lib/pipe.js
|
|
||||||
@@ -42,7 +42,8 @@ function pipe(path, options) {
|
|
||||||
return d.promise;
|
|
||||||
}
|
|
||||||
module.exports = pipe;
|
|
||||||
-if (module === require.main) {
|
|
||||||
+
|
|
||||||
+function main() {
|
|
||||||
if (!process.argv[4]) {
|
|
||||||
console.error('Incorrect arguments!');
|
|
||||||
process.exit(-1);
|
|
||||||
@@ -52,3 +53,8 @@ if (module === require.main) {
|
|
||||||
serve: process.argv[3] == 'server'
|
|
||||||
});
|
|
||||||
}
|
|
||||||
+module.exports.main = main;
|
|
||||||
+
|
|
||||||
+if (module === require.main) {
|
|
||||||
+ main();
|
|
||||||
+}
|
|
||||||
diff --git a/node_modules/windosu/lib/windosu.js b/node_modules/windosu/lib/windosu.js
|
|
||||||
index 6502812..dd0391a 100644
|
|
||||||
--- a/node_modules/windosu/lib/windosu.js
|
|
||||||
+++ b/node_modules/windosu/lib/windosu.js
|
|
||||||
@@ -16,7 +16,9 @@ module.exports.exec = function (command, options, callback) {
|
|
||||||
temp: temp,
|
|
||||||
command: command,
|
|
||||||
cliWidth: cliWidth(),
|
|
||||||
- pipe: '"' + process.execPath + '" "' + path.join(__dirname, 'pipe.js') + '"',
|
|
||||||
+ pipe: process.pkg
|
|
||||||
+ ? '"' + process.execPath + '" pkgExec "' + path.join(__dirname, 'pipe.js') + '::main"'
|
|
||||||
+ : '"' + process.execPath + '" "' + path.join(__dirname, 'pipe.js') + '"',
|
|
||||||
input: inputName = id + '-in',
|
|
||||||
output: outputName = id + '-out',
|
|
||||||
stderr_redir: process.stdout.isTTY ? '2>&1' : '2> %ERROR%'
|
|
4
repo.yml
4
repo.yml
@ -6,6 +6,10 @@ upstream:
|
|||||||
url: 'https://github.com/balena-io/balena-sdk'
|
url: 'https://github.com/balena-io/balena-sdk'
|
||||||
- repo: 'balena-config-json'
|
- repo: 'balena-config-json'
|
||||||
url: 'https://github.com/balena-io-modules/balena-config-json'
|
url: 'https://github.com/balena-io-modules/balena-config-json'
|
||||||
|
- repo: 'balena-image-fs'
|
||||||
|
url: 'https://github.com/balena-io-modules/balena-image-fs'
|
||||||
|
- repo: 'balena-device-init'
|
||||||
|
url: 'https://github.com/balena-io-modules/balena-device-init'
|
||||||
- repo: 'balena-image-manager'
|
- repo: 'balena-image-manager'
|
||||||
url: 'https://github.com/balena-io-modules/balena-image-manager'
|
url: 'https://github.com/balena-io-modules/balena-image-manager'
|
||||||
- repo: 'balena-preload'
|
- repo: 'balena-preload'
|
||||||
|
26
src/app.ts
26
src/app.ts
@ -34,18 +34,14 @@ export const setupSentry = onceAsync(async () => {
|
|||||||
const config = await import('./config');
|
const config = await import('./config');
|
||||||
const Sentry = await import('@sentry/node');
|
const Sentry = await import('@sentry/node');
|
||||||
Sentry.init({
|
Sentry.init({
|
||||||
autoSessionTracking: false,
|
|
||||||
dsn: config.sentryDsn,
|
dsn: config.sentryDsn,
|
||||||
release: packageJSON.version,
|
release: packageJSON.version,
|
||||||
});
|
});
|
||||||
Sentry.configureScope((scope) => {
|
Sentry.getCurrentScope().setExtras({
|
||||||
scope.setExtras({
|
is_pkg: !!(process as any).pkg,
|
||||||
is_pkg: !!(process as any).pkg,
|
node_version: process.version,
|
||||||
node_version: process.version,
|
platform: process.platform,
|
||||||
platform: process.platform,
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
return Sentry.getCurrentHub();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
async function checkNodeVersion() {
|
async function checkNodeVersion() {
|
||||||
@ -101,11 +97,9 @@ async function init() {
|
|||||||
|
|
||||||
/** Execute the oclif parser and the CLI command. */
|
/** Execute the oclif parser and the CLI command. */
|
||||||
async function oclifRun(command: string[], options: AppOptions) {
|
async function oclifRun(command: string[], options: AppOptions) {
|
||||||
let deprecationPromise: Promise<void>;
|
let deprecationPromise: Promise<void> | undefined;
|
||||||
// check and enforce the CLI's deprecation policy
|
// check and enforce the CLI's deprecation policy
|
||||||
if (unsupportedFlag || process.env.BALENARC_UNSUPPORTED) {
|
if (!(unsupportedFlag || process.env.BALENARC_UNSUPPORTED)) {
|
||||||
deprecationPromise = Promise.resolve();
|
|
||||||
} else {
|
|
||||||
const { DeprecationChecker } = await import('./deprecation');
|
const { DeprecationChecker } = await import('./deprecation');
|
||||||
const deprecationChecker = new DeprecationChecker(packageJSON.version);
|
const deprecationChecker = new DeprecationChecker(packageJSON.version);
|
||||||
// warnAndAbortIfDeprecated uses previously cached data only
|
// warnAndAbortIfDeprecated uses previously cached data only
|
||||||
@ -161,18 +155,12 @@ async function oclifRun(command: string[], options: AppOptions) {
|
|||||||
/** CLI entrypoint. Called by the `bin/run.js` and `bin/dev.js` scripts. */
|
/** CLI entrypoint. Called by the `bin/run.js` and `bin/dev.js` scripts. */
|
||||||
export async function run(cliArgs = process.argv, options: AppOptions) {
|
export async function run(cliArgs = process.argv, options: AppOptions) {
|
||||||
try {
|
try {
|
||||||
const { setOfflineModeEnvVars, normalizeEnvVars, pkgExec } = await import(
|
const { setOfflineModeEnvVars, normalizeEnvVars } = await import(
|
||||||
'./utils/bootstrap'
|
'./utils/bootstrap'
|
||||||
);
|
);
|
||||||
setOfflineModeEnvVars();
|
setOfflineModeEnvVars();
|
||||||
normalizeEnvVars();
|
normalizeEnvVars();
|
||||||
|
|
||||||
// The 'pkgExec' special/internal command provides a Node.js interpreter
|
|
||||||
// for use of the standalone zip package. See pkgExec function.
|
|
||||||
if (cliArgs.length > 3 && cliArgs[2] === 'pkgExec') {
|
|
||||||
return pkgExec(cliArgs[3], cliArgs.slice(4));
|
|
||||||
}
|
|
||||||
|
|
||||||
await init();
|
await init();
|
||||||
|
|
||||||
// Look for commands that have been removed and if so, exit with a notice
|
// Look for commands that have been removed and if so, exit with a notice
|
||||||
|
@ -17,8 +17,28 @@
|
|||||||
|
|
||||||
import { Args, Command } from '@oclif/core';
|
import { Args, Command } from '@oclif/core';
|
||||||
import { ExpectedError } from '../../errors';
|
import { ExpectedError } from '../../errors';
|
||||||
import * as cf from '../../utils/common-flags';
|
import { getBalenaSdk, getCliForm, stripIndent } from '../../utils/lazy';
|
||||||
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
import {
|
||||||
|
formatDuration,
|
||||||
|
intervalToDuration,
|
||||||
|
isValid,
|
||||||
|
parseISO,
|
||||||
|
} from 'date-fns';
|
||||||
|
|
||||||
|
// In days
|
||||||
|
const durations = [1, 7, 30, 90];
|
||||||
|
|
||||||
|
async function isLoggedInWithJwt() {
|
||||||
|
const balena = getBalenaSdk();
|
||||||
|
try {
|
||||||
|
const token = await balena.auth.getToken();
|
||||||
|
const { default: jwtDecode } = await import('jwt-decode');
|
||||||
|
jwtDecode(token);
|
||||||
|
return true;
|
||||||
|
} catch {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export default class GenerateCmd extends Command {
|
export default class GenerateCmd extends Command {
|
||||||
public static description = stripIndent`
|
public static description = stripIndent`
|
||||||
@ -30,17 +50,21 @@ export default class GenerateCmd extends Command {
|
|||||||
This key can be used to log into the CLI using 'balena login --token <key>',
|
This key can be used to log into the CLI using 'balena login --token <key>',
|
||||||
or to authenticate requests to the API with an 'Authorization: Bearer <key>' header.
|
or to authenticate requests to the API with an 'Authorization: Bearer <key>' header.
|
||||||
`;
|
`;
|
||||||
public static examples = ['$ balena api-key generate "Jenkins Key"'];
|
public static examples = [
|
||||||
|
'$ balena api-key generate "Jenkins Key"',
|
||||||
|
'$ balena api-key generate "Jenkins Key" 2025-10-30',
|
||||||
|
'$ balena api-key generate "Jenkins Key" never',
|
||||||
|
];
|
||||||
|
|
||||||
public static args = {
|
public static args = {
|
||||||
name: Args.string({
|
name: Args.string({
|
||||||
description: 'the API key name',
|
description: 'the API key name',
|
||||||
required: true,
|
required: true,
|
||||||
}),
|
}),
|
||||||
};
|
expiryDate: Args.string({
|
||||||
|
description:
|
||||||
public static flags = {
|
'the expiry date of the API key as an ISO date string, or "never" for no expiry',
|
||||||
help: cf.help,
|
}),
|
||||||
};
|
};
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
@ -48,11 +72,70 @@ export default class GenerateCmd extends Command {
|
|||||||
public async run() {
|
public async run() {
|
||||||
const { args: params } = await this.parse(GenerateCmd);
|
const { args: params } = await this.parse(GenerateCmd);
|
||||||
|
|
||||||
|
let expiryDateResponse: string | number | undefined = params.expiryDate;
|
||||||
let key;
|
let key;
|
||||||
try {
|
try {
|
||||||
key = await getBalenaSdk().models.apiKey.create(params.name);
|
if (!expiryDateResponse) {
|
||||||
|
expiryDateResponse = await getCliForm().ask({
|
||||||
|
message: 'Please pick an expiry date for the API key',
|
||||||
|
type: 'list',
|
||||||
|
choices: [...durations, 'custom', 'never'].map((duration) => ({
|
||||||
|
name:
|
||||||
|
duration === 'never'
|
||||||
|
? 'No expiration'
|
||||||
|
: typeof duration === 'number'
|
||||||
|
? formatDuration(
|
||||||
|
intervalToDuration({
|
||||||
|
start: 0,
|
||||||
|
end: duration * 24 * 60 * 60 * 1000,
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
: 'Custom expiration',
|
||||||
|
value: duration,
|
||||||
|
})),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
let expiryDate: Date | null;
|
||||||
|
if (expiryDateResponse === 'never') {
|
||||||
|
expiryDate = null;
|
||||||
|
} else if (expiryDateResponse === 'custom') {
|
||||||
|
do {
|
||||||
|
expiryDate = parseISO(
|
||||||
|
await getCliForm().ask({
|
||||||
|
message:
|
||||||
|
'Please enter an expiry date for the API key as an ISO date string',
|
||||||
|
type: 'input',
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
if (!isValid(expiryDate)) {
|
||||||
|
console.error('Invalid date format');
|
||||||
|
}
|
||||||
|
} while (!isValid(expiryDate));
|
||||||
|
} else if (typeof expiryDateResponse === 'string') {
|
||||||
|
expiryDate = parseISO(expiryDateResponse);
|
||||||
|
if (!isValid(expiryDate)) {
|
||||||
|
throw new Error(
|
||||||
|
'Invalid date format, please use a valid ISO date string',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
expiryDate = new Date(
|
||||||
|
Date.now() + expiryDateResponse * 24 * 60 * 60 * 1000,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
key = await getBalenaSdk().models.apiKey.create({
|
||||||
|
name: params.name,
|
||||||
|
expiryDate: expiryDate === null ? null : expiryDate.toISOString(),
|
||||||
|
});
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
if (e.name === 'BalenaNotLoggedIn') {
|
if (e.name === 'BalenaNotLoggedIn') {
|
||||||
|
if (await isLoggedInWithJwt()) {
|
||||||
|
throw new ExpectedError(stripIndent`
|
||||||
|
This command requires you to have been recently authenticated.
|
||||||
|
Please login again with 'balena login'.
|
||||||
|
In case you are using the Web authorization method, you need to logout and re-login to the dashboard first.
|
||||||
|
`);
|
||||||
|
}
|
||||||
throw new ExpectedError(stripIndent`
|
throw new ExpectedError(stripIndent`
|
||||||
This command cannot be run when logged in with an API key.
|
This command cannot be run when logged in with an API key.
|
||||||
Please login again with 'balena login' and select an alternative method.
|
Please login again with 'balena login' and select an alternative method.
|
||||||
|
@ -21,6 +21,7 @@ import { getBalenaSdk, getVisuals, stripIndent } from '../../utils/lazy';
|
|||||||
|
|
||||||
export default class APIKeyListCmd extends Command {
|
export default class APIKeyListCmd extends Command {
|
||||||
public static aliases = ['api-keys'];
|
public static aliases = ['api-keys'];
|
||||||
|
public static deprecateAliases = true;
|
||||||
|
|
||||||
public static description = stripIndent`
|
public static description = stripIndent`
|
||||||
Print a list of balenaCloud API keys.
|
Print a list of balenaCloud API keys.
|
||||||
@ -32,7 +33,6 @@ export default class APIKeyListCmd extends Command {
|
|||||||
public static examples = ['$ balena api-key list'];
|
public static examples = ['$ balena api-key list'];
|
||||||
|
|
||||||
public static flags = {
|
public static flags = {
|
||||||
help: cf.help,
|
|
||||||
user: Flags.boolean({
|
user: Flags.boolean({
|
||||||
char: 'u',
|
char: 'u',
|
||||||
description: 'show API keys for your user',
|
description: 'show API keys for your user',
|
||||||
@ -51,7 +51,7 @@ export default class APIKeyListCmd extends Command {
|
|||||||
await getApplication(getBalenaSdk(), options.fleet, {
|
await getApplication(getBalenaSdk(), options.fleet, {
|
||||||
$select: 'actor',
|
$select: 'actor',
|
||||||
})
|
})
|
||||||
).actor
|
).actor.__id
|
||||||
: await getBalenaSdk().auth.getActorId();
|
: await getBalenaSdk().auth.getActorId();
|
||||||
const keys = await getBalenaSdk().pine.get({
|
const keys = await getBalenaSdk().pine.get({
|
||||||
resource: 'api_key',
|
resource: 'api_key',
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Args, Command } from '@oclif/core';
|
import { Args, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
||||||
|
|
||||||
export default class RevokeCmd extends Command {
|
export default class RevokeCmd extends Command {
|
||||||
@ -40,10 +39,6 @@ export default class RevokeCmd extends Command {
|
|||||||
}),
|
}),
|
||||||
};
|
};
|
||||||
|
|
||||||
public static flags = {
|
|
||||||
help: cf.help,
|
|
||||||
};
|
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
|
||||||
public async run() {
|
public async run() {
|
||||||
@ -55,9 +50,9 @@ export default class RevokeCmd extends Command {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
await Promise.all(
|
await Promise.all(
|
||||||
apiKeyIds.map(
|
apiKeyIds.map(async (id) => {
|
||||||
async (id) => await getBalenaSdk().models.apiKey.revoke(Number(id)),
|
await getBalenaSdk().models.apiKey.revoke(Number(id));
|
||||||
),
|
}),
|
||||||
);
|
);
|
||||||
console.log('Successfully revoked the given API keys');
|
console.log('Successfully revoked the given API keys');
|
||||||
}
|
}
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Flags, Args, Command } from '@oclif/core';
|
import { Flags, Args, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { stripIndent } from '../../utils/lazy';
|
import { stripIndent } from '../../utils/lazy';
|
||||||
|
|
||||||
export default class AppCreateCmd extends Command {
|
export default class AppCreateCmd extends Command {
|
||||||
@ -28,10 +27,10 @@ export default class AppCreateCmd extends Command {
|
|||||||
You can specify the organization the app should belong to using
|
You can specify the organization the app should belong to using
|
||||||
the \`--organization\` option. The organization's handle, not its name,
|
the \`--organization\` option. The organization's handle, not its name,
|
||||||
should be provided. Organization handles can be listed with the
|
should be provided. Organization handles can be listed with the
|
||||||
\`balena orgs\` command.
|
\`balena organization list\` command.
|
||||||
|
|
||||||
The app's default device type is specified with the \`--type\` option.
|
The app's default device type is specified with the \`--type\` option.
|
||||||
The \`balena devices supported\` command can be used to list the available
|
The \`balena device-type list\` command can be used to list the available
|
||||||
device types.
|
device types.
|
||||||
|
|
||||||
Interactive dropdowns will be shown for selection if no device type or
|
Interactive dropdowns will be shown for selection if no device type or
|
||||||
@ -62,9 +61,8 @@ export default class AppCreateCmd extends Command {
|
|||||||
type: Flags.string({
|
type: Flags.string({
|
||||||
char: 't',
|
char: 't',
|
||||||
description:
|
description:
|
||||||
'app device type (Check available types with `balena devices supported`)',
|
'app device type (Check available types with `balena device-type list`)',
|
||||||
}),
|
}),
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Flags, Args, Command } from '@oclif/core';
|
import { Flags, Args, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { stripIndent } from '../../utils/lazy';
|
import { stripIndent } from '../../utils/lazy';
|
||||||
|
|
||||||
export default class BlockCreateCmd extends Command {
|
export default class BlockCreateCmd extends Command {
|
||||||
@ -28,10 +27,10 @@ export default class BlockCreateCmd extends Command {
|
|||||||
You can specify the organization the block should belong to using
|
You can specify the organization the block should belong to using
|
||||||
the \`--organization\` option. The organization's handle, not its name,
|
the \`--organization\` option. The organization's handle, not its name,
|
||||||
should be provided. Organization handles can be listed with the
|
should be provided. Organization handles can be listed with the
|
||||||
\`balena orgs\` command.
|
\`balena organization list\` command.
|
||||||
|
|
||||||
The block's default device type is specified with the \`--type\` option.
|
The block's default device type is specified with the \`--type\` option.
|
||||||
The \`balena devices supported\` command can be used to list the available
|
The \`balena device-type list\` command can be used to list the available
|
||||||
device types.
|
device types.
|
||||||
|
|
||||||
Interactive dropdowns will be shown for selection if no device type or
|
Interactive dropdowns will be shown for selection if no device type or
|
||||||
@ -62,9 +61,8 @@ export default class BlockCreateCmd extends Command {
|
|||||||
type: Flags.string({
|
type: Flags.string({
|
||||||
char: 't',
|
char: 't',
|
||||||
description:
|
description:
|
||||||
'block device type (Check available types with `balena devices supported`)',
|
'block device type (Check available types with `balena device-type list`)',
|
||||||
}),
|
}),
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
@ -36,15 +36,16 @@ import { buildProject, composeCliFlags } from '../../utils/compose_ts';
|
|||||||
import type { BuildOpts, DockerCliFlags } from '../../utils/docker';
|
import type { BuildOpts, DockerCliFlags } from '../../utils/docker';
|
||||||
import { dockerCliFlags } from '../../utils/docker';
|
import { dockerCliFlags } from '../../utils/docker';
|
||||||
|
|
||||||
// TODO: For this special one we can't use Interfaces.InferredFlags/InferredArgs
|
type ComposeGenerateOptsParam = Parameters<typeof compose.generateOpts>[0];
|
||||||
// because of the 'registry-secrets' type which is defined in the actual code
|
|
||||||
// as a path (string | undefined) but then the cli turns it into an object
|
interface PrepareBuildOpts
|
||||||
interface FlagsDef extends ComposeCliFlags, DockerCliFlags {
|
extends ComposeCliFlags,
|
||||||
|
DockerCliFlags,
|
||||||
|
ComposeGenerateOptsParam {
|
||||||
arch?: string;
|
arch?: string;
|
||||||
deviceType?: string;
|
deviceType?: string;
|
||||||
fleet?: string;
|
fleet?: string;
|
||||||
source?: string; // Not part of command profile - source param copied here.
|
source?: string;
|
||||||
help: void;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export default class BuildCmd extends Command {
|
export default class BuildCmd extends Command {
|
||||||
@ -95,9 +96,6 @@ ${dockerignoreHelp}
|
|||||||
fleet: cf.fleet,
|
fleet: cf.fleet,
|
||||||
...composeCliFlags,
|
...composeCliFlags,
|
||||||
...dockerCliFlags,
|
...dockerCliFlags,
|
||||||
// NOTE: Not supporting -h for help, because of clash with -h in DockerCliFlags
|
|
||||||
// Revisit this in future release.
|
|
||||||
help: Flags.help({}),
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static primary = true;
|
public static primary = true;
|
||||||
@ -117,29 +115,31 @@ ${dockerignoreHelp}
|
|||||||
const logger = Logger.getLogger();
|
const logger = Logger.getLogger();
|
||||||
logger.logDebug('Parsing input...');
|
logger.logDebug('Parsing input...');
|
||||||
|
|
||||||
// `build` accepts `source` as a parameter, but compose expects it as an option
|
const prepareBuildOpts = {
|
||||||
options.source = params.source;
|
...options,
|
||||||
delete params.source;
|
source: params.source,
|
||||||
|
};
|
||||||
|
|
||||||
await this.resolveArchFromDeviceType(sdk, options);
|
await this.resolveArchFromDeviceType(sdk, prepareBuildOpts);
|
||||||
|
|
||||||
await this.validateOptions(options, sdk);
|
await this.validateOptions(prepareBuildOpts, sdk);
|
||||||
|
|
||||||
// Build args are under consideration for removal - warn user
|
// Build args are under consideration for removal - warn user
|
||||||
if (options.buildArg) {
|
if (prepareBuildOpts.buildArg) {
|
||||||
console.log(buildArgDeprecation);
|
console.log(buildArgDeprecation);
|
||||||
}
|
}
|
||||||
|
|
||||||
const app = await this.getAppAndResolveArch(options);
|
const app = await this.getAppAndResolveArch(prepareBuildOpts);
|
||||||
|
|
||||||
const { docker, buildOpts, composeOpts } = await this.prepareBuild(options);
|
const { docker, buildOpts, composeOpts } =
|
||||||
|
await this.prepareBuild(prepareBuildOpts);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
await this.buildProject(docker, logger, composeOpts, {
|
await this.buildProject(docker, logger, composeOpts, {
|
||||||
appType: app?.application_type?.[0],
|
appType: app?.application_type?.[0],
|
||||||
arch: options.arch!,
|
arch: prepareBuildOpts.arch!,
|
||||||
deviceType: options.deviceType!,
|
deviceType: prepareBuildOpts.deviceType!,
|
||||||
buildEmulated: options.emulated,
|
buildEmulated: prepareBuildOpts.emulated,
|
||||||
buildOpts,
|
buildOpts,
|
||||||
});
|
});
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
@ -151,7 +151,7 @@ ${dockerignoreHelp}
|
|||||||
logger.logSuccess('Build succeeded!');
|
logger.logSuccess('Build succeeded!');
|
||||||
}
|
}
|
||||||
|
|
||||||
protected async validateOptions(opts: FlagsDef, sdk: BalenaSDK) {
|
protected async validateOptions(opts: PrepareBuildOpts, sdk: BalenaSDK) {
|
||||||
// Validate option combinations
|
// Validate option combinations
|
||||||
if (
|
if (
|
||||||
(opts.fleet == null && (opts.arch == null || opts.deviceType == null)) ||
|
(opts.fleet == null && (opts.arch == null || opts.deviceType == null)) ||
|
||||||
@ -179,7 +179,10 @@ ${dockerignoreHelp}
|
|||||||
opts['registry-secrets'] = registrySecrets;
|
opts['registry-secrets'] = registrySecrets;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected async resolveArchFromDeviceType(sdk: BalenaSDK, opts: FlagsDef) {
|
protected async resolveArchFromDeviceType(
|
||||||
|
sdk: BalenaSDK,
|
||||||
|
opts: PrepareBuildOpts,
|
||||||
|
) {
|
||||||
if (opts.deviceType != null && opts.arch == null) {
|
if (opts.deviceType != null && opts.arch == null) {
|
||||||
try {
|
try {
|
||||||
const deviceTypeOpts = {
|
const deviceTypeOpts = {
|
||||||
@ -212,7 +215,7 @@ ${dockerignoreHelp}
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected async getAppAndResolveArch(opts: FlagsDef) {
|
protected async getAppAndResolveArch(opts: PrepareBuildOpts) {
|
||||||
if (opts.fleet) {
|
if (opts.fleet) {
|
||||||
const { getAppWithArch } = await import('../../utils/helpers');
|
const { getAppWithArch } = await import('../../utils/helpers');
|
||||||
const app = await getAppWithArch(opts.fleet);
|
const app = await getAppWithArch(opts.fleet);
|
||||||
@ -222,7 +225,7 @@ ${dockerignoreHelp}
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected async prepareBuild(options: FlagsDef) {
|
protected async prepareBuild(options: PrepareBuildOpts) {
|
||||||
const { getDocker, generateBuildOpts } = await import('../../utils/docker');
|
const { getDocker, generateBuildOpts } = await import('../../utils/docker');
|
||||||
const [docker, buildOpts, composeOpts] = await Promise.all([
|
const [docker, buildOpts, composeOpts] = await Promise.all([
|
||||||
getDocker(options),
|
getDocker(options),
|
||||||
|
@ -82,7 +82,7 @@ export default class ConfigGenerateCmd extends Command {
|
|||||||
}),
|
}),
|
||||||
deviceType: Flags.string({
|
deviceType: Flags.string({
|
||||||
description:
|
description:
|
||||||
"device type slug (run 'balena devices supported' for possible values)",
|
"device type slug (run 'balena device-type list' for possible values)",
|
||||||
}),
|
}),
|
||||||
'generate-device-api-key': Flags.boolean({
|
'generate-device-api-key': Flags.boolean({
|
||||||
description: 'generate a fresh device key for the device',
|
description: 'generate a fresh device key for the device',
|
||||||
@ -117,7 +117,6 @@ export default class ConfigGenerateCmd extends Command {
|
|||||||
'expiry date assigned to generated provisioning api key (format: YYYY-MM-DD)',
|
'expiry date assigned to generated provisioning api key (format: YYYY-MM-DD)',
|
||||||
exclusive: ['device'],
|
exclusive: ['device'],
|
||||||
}),
|
}),
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
@ -44,7 +44,6 @@ export default class ConfigInjectCmd extends Command {
|
|||||||
|
|
||||||
public static flags = {
|
public static flags = {
|
||||||
drive: cf.driveOrImg,
|
drive: cf.driveOrImg,
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static root = true;
|
public static root = true;
|
||||||
@ -65,7 +64,12 @@ export default class ConfigInjectCmd extends Command {
|
|||||||
);
|
);
|
||||||
|
|
||||||
const config = await import('balena-config-json');
|
const config = await import('balena-config-json');
|
||||||
await config.write(drive, '', configJSON);
|
await config.write(
|
||||||
|
drive,
|
||||||
|
// Will be removed in the next major of balena-config-json
|
||||||
|
undefined,
|
||||||
|
configJSON,
|
||||||
|
);
|
||||||
|
|
||||||
console.info('Done');
|
console.info('Done');
|
||||||
}
|
}
|
||||||
|
@ -38,7 +38,6 @@ export default class ConfigReadCmd extends Command {
|
|||||||
|
|
||||||
public static flags = {
|
public static flags = {
|
||||||
drive: cf.driveOrImg,
|
drive: cf.driveOrImg,
|
||||||
help: cf.help,
|
|
||||||
json: cf.json,
|
json: cf.json,
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -55,7 +54,7 @@ export default class ConfigReadCmd extends Command {
|
|||||||
await safeUmount(drive);
|
await safeUmount(drive);
|
||||||
|
|
||||||
const config = await import('balena-config-json');
|
const config = await import('balena-config-json');
|
||||||
const configJSON = await config.read(drive, '');
|
const configJSON = await config.read(drive);
|
||||||
|
|
||||||
if (options.json) {
|
if (options.json) {
|
||||||
console.log(JSON.stringify(configJSON, null, 4));
|
console.log(JSON.stringify(configJSON, null, 4));
|
||||||
|
@ -44,7 +44,6 @@ export default class ConfigReconfigureCmd extends Command {
|
|||||||
description: 'show advanced commands',
|
description: 'show advanced commands',
|
||||||
char: 'v',
|
char: 'v',
|
||||||
}),
|
}),
|
||||||
help: cf.help,
|
|
||||||
version: Flags.string({
|
version: Flags.string({
|
||||||
description: 'balenaOS version, for example "2.32.0" or "2.44.0+rev1"',
|
description: 'balenaOS version, for example "2.32.0" or "2.44.0+rev1"',
|
||||||
}),
|
}),
|
||||||
@ -63,7 +62,7 @@ export default class ConfigReconfigureCmd extends Command {
|
|||||||
await safeUmount(drive);
|
await safeUmount(drive);
|
||||||
|
|
||||||
const config = await import('balena-config-json');
|
const config = await import('balena-config-json');
|
||||||
const { uuid } = await config.read(drive, '');
|
const { uuid } = await config.read(drive);
|
||||||
await safeUmount(drive);
|
await safeUmount(drive);
|
||||||
|
|
||||||
if (!uuid) {
|
if (!uuid) {
|
||||||
|
@ -49,7 +49,6 @@ export default class ConfigWriteCmd extends Command {
|
|||||||
|
|
||||||
public static flags = {
|
public static flags = {
|
||||||
drive: cf.driveOrImg,
|
drive: cf.driveOrImg,
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static root = true;
|
public static root = true;
|
||||||
@ -65,14 +64,19 @@ export default class ConfigWriteCmd extends Command {
|
|||||||
await safeUmount(drive);
|
await safeUmount(drive);
|
||||||
|
|
||||||
const config = await import('balena-config-json');
|
const config = await import('balena-config-json');
|
||||||
const configJSON = await config.read(drive, '');
|
const configJSON = await config.read(drive);
|
||||||
|
|
||||||
console.info(`Setting ${params.key} to ${params.value}`);
|
console.info(`Setting ${params.key} to ${params.value}`);
|
||||||
ConfigWriteCmd.updateConfigJson(configJSON, params.key, params.value);
|
ConfigWriteCmd.updateConfigJson(configJSON, params.key, params.value);
|
||||||
|
|
||||||
await denyMount(drive, async () => {
|
await denyMount(drive, async () => {
|
||||||
await safeUmount(drive);
|
await safeUmount(drive);
|
||||||
await config.write(drive, '', configJSON);
|
await config.write(
|
||||||
|
drive,
|
||||||
|
// Will be removed in the next major of balena-config-json
|
||||||
|
undefined,
|
||||||
|
configJSON,
|
||||||
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
console.info('Done');
|
console.info('Done');
|
||||||
|
@ -60,7 +60,6 @@ interface FlagsDef extends ComposeCliFlags, DockerCliFlags {
|
|||||||
'release-tag'?: string[];
|
'release-tag'?: string[];
|
||||||
draft: boolean;
|
draft: boolean;
|
||||||
note?: string;
|
note?: string;
|
||||||
help: void;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export default class DeployCmd extends Command {
|
export default class DeployCmd extends Command {
|
||||||
@ -139,9 +138,6 @@ ${dockerignoreHelp}
|
|||||||
note: Flags.string({ description: 'The notes for this release' }),
|
note: Flags.string({ description: 'The notes for this release' }),
|
||||||
...composeCliFlags,
|
...composeCliFlags,
|
||||||
...dockerCliFlags,
|
...dockerCliFlags,
|
||||||
// NOTE: Not supporting -h for help, because of clash with -h in DockerCliFlags
|
|
||||||
// Revisit this in future release.
|
|
||||||
help: Flags.help({}),
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
@ -372,6 +368,7 @@ ${dockerignoreHelp}
|
|||||||
!opts.shouldUploadLogs,
|
!opts.shouldUploadLogs,
|
||||||
composeOpts.projectPath,
|
composeOpts.projectPath,
|
||||||
opts.createAsDraft,
|
opts.createAsDraft,
|
||||||
|
project.descriptors,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -17,14 +17,20 @@
|
|||||||
import { Flags, Command } from '@oclif/core';
|
import { Flags, Command } from '@oclif/core';
|
||||||
import type * as BalenaSdk from 'balena-sdk';
|
import type * as BalenaSdk from 'balena-sdk';
|
||||||
import * as _ from 'lodash';
|
import * as _ from 'lodash';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { getBalenaSdk, getVisuals, stripIndent } from '../../utils/lazy';
|
import { getBalenaSdk, getVisuals, stripIndent } from '../../utils/lazy';
|
||||||
|
|
||||||
export default class DevicesSupportedCmd extends Command {
|
export default class DeviceTypeListCmd extends Command {
|
||||||
public static description = stripIndent`
|
public static aliases = ['devices supported'];
|
||||||
List the supported device types (like 'raspberrypi3' or 'intel-nuc').
|
public static deprecateAliases = true;
|
||||||
|
|
||||||
List the supported device types (like 'raspberrypi3' or 'intel-nuc').
|
public static description = stripIndent`
|
||||||
|
List the device types supported by balena (like 'raspberrypi3' or 'intel-nuc').
|
||||||
|
|
||||||
|
List the device types supported by balena (like 'raspberrypi3' or 'intel-nuc').
|
||||||
|
|
||||||
|
By default, only actively supported device types are listed.
|
||||||
|
The --all option can be used to list all device types, including those that are
|
||||||
|
no longer supported by balena.
|
||||||
|
|
||||||
The --json option is recommended when scripting the output of this command,
|
The --json option is recommended when scripting the output of this command,
|
||||||
because the JSON format is less likely to change and it better represents data
|
because the JSON format is less likely to change and it better represents data
|
||||||
@ -33,20 +39,24 @@ export default class DevicesSupportedCmd extends Command {
|
|||||||
(https://stedolan.github.io/jq/manual/).
|
(https://stedolan.github.io/jq/manual/).
|
||||||
`;
|
`;
|
||||||
public static examples = [
|
public static examples = [
|
||||||
'$ balena devices supported',
|
'$ balena device-type list',
|
||||||
'$ balena devices supported --json',
|
'$ balena device-type list --all',
|
||||||
|
'$ balena device-type list --json',
|
||||||
];
|
];
|
||||||
|
|
||||||
public static flags = {
|
public static flags = {
|
||||||
help: cf.help,
|
|
||||||
json: Flags.boolean({
|
json: Flags.boolean({
|
||||||
char: 'j',
|
char: 'j',
|
||||||
description: 'produce JSON output instead of tabular output',
|
description: 'produce JSON output instead of tabular output',
|
||||||
}),
|
}),
|
||||||
|
all: Flags.boolean({
|
||||||
|
description: 'include device types no longer supported by balena',
|
||||||
|
default: false,
|
||||||
|
}),
|
||||||
};
|
};
|
||||||
|
|
||||||
public async run() {
|
public async run() {
|
||||||
const { flags: options } = await this.parse(DevicesSupportedCmd);
|
const { flags: options } = await this.parse(DeviceTypeListCmd);
|
||||||
const pineOptions = {
|
const pineOptions = {
|
||||||
$select: ['slug', 'name'],
|
$select: ['slug', 'name'],
|
||||||
$expand: {
|
$expand: {
|
||||||
@ -57,9 +67,11 @@ export default class DevicesSupportedCmd extends Command {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
} satisfies BalenaSdk.PineOptions<BalenaSdk.DeviceType>;
|
} satisfies BalenaSdk.PineOptions<BalenaSdk.DeviceType>;
|
||||||
const dts = (await getBalenaSdk().models.deviceType.getAllSupported(
|
const dts = (
|
||||||
pineOptions,
|
options.all
|
||||||
)) as Array<
|
? await getBalenaSdk().models.deviceType.getAll(pineOptions)
|
||||||
|
: await getBalenaSdk().models.deviceType.getAllSupported(pineOptions)
|
||||||
|
) as Array<
|
||||||
BalenaSdk.PineTypedResult<BalenaSdk.DeviceType, typeof pineOptions>
|
BalenaSdk.PineTypedResult<BalenaSdk.DeviceType, typeof pineOptions>
|
||||||
>;
|
>;
|
||||||
interface DT {
|
interface DT {
|
@ -42,7 +42,6 @@ export default class DeviceDeactivateCmd extends Command {
|
|||||||
|
|
||||||
public static flags = {
|
public static flags = {
|
||||||
yes: cf.yes,
|
yes: cf.yes,
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
@ -16,10 +16,12 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Flags, Command } from '@oclif/core';
|
import { Flags, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { getCliUx, stripIndent } from '../../utils/lazy';
|
import { getCliUx, stripIndent } from '../../utils/lazy';
|
||||||
|
|
||||||
export default class ScanCmd extends Command {
|
export default class DeviceDetectCmd extends Command {
|
||||||
|
public static aliases = ['scan'];
|
||||||
|
public static deprecateAliases = true;
|
||||||
|
|
||||||
public static description = stripIndent`
|
public static description = stripIndent`
|
||||||
Scan for balenaOS devices on your local network.
|
Scan for balenaOS devices on your local network.
|
||||||
|
|
||||||
@ -32,9 +34,9 @@ export default class ScanCmd extends Command {
|
|||||||
`;
|
`;
|
||||||
|
|
||||||
public static examples = [
|
public static examples = [
|
||||||
'$ balena scan',
|
'$ balena device detect',
|
||||||
'$ balena scan --timeout 120',
|
'$ balena device detect --timeout 120',
|
||||||
'$ balena scan --verbose',
|
'$ balena device detect --verbose',
|
||||||
];
|
];
|
||||||
|
|
||||||
public static flags = {
|
public static flags = {
|
||||||
@ -47,7 +49,6 @@ export default class ScanCmd extends Command {
|
|||||||
char: 't',
|
char: 't',
|
||||||
description: 'scan timeout in seconds',
|
description: 'scan timeout in seconds',
|
||||||
}),
|
}),
|
||||||
help: cf.help,
|
|
||||||
json: Flags.boolean({
|
json: Flags.boolean({
|
||||||
default: false,
|
default: false,
|
||||||
char: 'j',
|
char: 'j',
|
||||||
@ -70,7 +71,7 @@ export default class ScanCmd extends Command {
|
|||||||
const dockerPort = 2375;
|
const dockerPort = 2375;
|
||||||
const dockerTimeout = 2000;
|
const dockerTimeout = 2000;
|
||||||
|
|
||||||
const { flags: options } = await this.parse(ScanCmd);
|
const { flags: options } = await this.parse(DeviceDetectCmd);
|
||||||
|
|
||||||
const discoverTimeout =
|
const discoverTimeout =
|
||||||
options.timeout != null ? options.timeout * 1000 : undefined;
|
options.timeout != null ? options.timeout * 1000 : undefined;
|
||||||
@ -90,7 +91,7 @@ export default class ScanCmd extends Command {
|
|||||||
try {
|
try {
|
||||||
await docker.ping();
|
await docker.ping();
|
||||||
return true;
|
return true;
|
||||||
} catch (err) {
|
} catch {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}),
|
}),
|
||||||
@ -144,10 +145,10 @@ export default class ScanCmd extends Command {
|
|||||||
if (!options.verbose) {
|
if (!options.verbose) {
|
||||||
devicesInfo.forEach((d: any) => {
|
devicesInfo.forEach((d: any) => {
|
||||||
d.dockerInfo = _.isObject(d.dockerInfo)
|
d.dockerInfo = _.isObject(d.dockerInfo)
|
||||||
? _.pick(d.dockerInfo, ScanCmd.dockerInfoProperties)
|
? _.pick(d.dockerInfo, DeviceDetectCmd.dockerInfoProperties)
|
||||||
: d.dockerInfo;
|
: d.dockerInfo;
|
||||||
d.dockerVersion = _.isObject(d.dockerVersion)
|
d.dockerVersion = _.isObject(d.dockerVersion)
|
||||||
? _.pick(d.dockerVersion, ScanCmd.dockerVersionProperties)
|
? _.pick(d.dockerVersion, DeviceDetectCmd.dockerVersionProperties)
|
||||||
: d.dockerVersion;
|
: d.dockerVersion;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -164,8 +165,9 @@ export default class ScanCmd extends Command {
|
|||||||
if (!options.json && cmdOutput.length === 0) {
|
if (!options.json && cmdOutput.length === 0) {
|
||||||
console.error(
|
console.error(
|
||||||
process.platform === 'win32'
|
process.platform === 'win32'
|
||||||
? ScanCmd.noDevicesFoundMessage + ScanCmd.windowsTipMessage
|
? DeviceDetectCmd.noDevicesFoundMessage +
|
||||||
: ScanCmd.noDevicesFoundMessage,
|
DeviceDetectCmd.windowsTipMessage
|
||||||
|
: DeviceDetectCmd.noDevicesFoundMessage,
|
||||||
);
|
);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -197,11 +199,11 @@ export default class ScanCmd extends Command {
|
|||||||
protected static windowsTipMessage = `
|
protected static windowsTipMessage = `
|
||||||
|
|
||||||
Note for Windows users:
|
Note for Windows users:
|
||||||
The 'scan' command relies on the Bonjour service. Check whether Bonjour is
|
The 'device detect' command relies on the Bonjour service. Check whether Bonjour is
|
||||||
installed (Control Panel > Programs and Features). If not, you can download
|
installed (Control Panel > Programs and Features). If not, you can download
|
||||||
Bonjour for Windows (included with Bonjour Print Services) from here:
|
Bonjour for Windows (included with Bonjour Print Services) from here:
|
||||||
https://support.apple.com/kb/DL999
|
https://support.apple.com/kb/DL999
|
||||||
|
|
||||||
After installing Bonjour, restart your PC and run the 'balena scan' command
|
After installing Bonjour, restart your PC and run the 'balena device detect' command
|
||||||
again.`;
|
again.`;
|
||||||
}
|
}
|
@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Args, Command } from '@oclif/core';
|
import { Args, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
||||||
import { ExpectedError } from '../../errors';
|
import { ExpectedError } from '../../errors';
|
||||||
|
|
||||||
@ -35,10 +34,6 @@ export default class DeviceIdentifyCmd extends Command {
|
|||||||
}),
|
}),
|
||||||
};
|
};
|
||||||
|
|
||||||
public static flags = {
|
|
||||||
help: cf.help,
|
|
||||||
};
|
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
|
||||||
public async run() {
|
public async run() {
|
||||||
|
@ -63,7 +63,6 @@ export default class DeviceCmd extends Command {
|
|||||||
|
|
||||||
public static flags = {
|
public static flags = {
|
||||||
json: cf.json,
|
json: cf.json,
|
||||||
help: cf.help,
|
|
||||||
view: Flags.boolean({
|
view: Flags.boolean({
|
||||||
default: false,
|
default: false,
|
||||||
description: 'open device dashboard page',
|
description: 'open device dashboard page',
|
||||||
@ -78,45 +77,59 @@ export default class DeviceCmd extends Command {
|
|||||||
|
|
||||||
const balena = getBalenaSdk();
|
const balena = getBalenaSdk();
|
||||||
|
|
||||||
const device = (await balena.models.device.get(
|
let device: ExtendedDevice;
|
||||||
params.uuid,
|
if (options.json) {
|
||||||
options.json
|
const [deviceBase, deviceComputed] = await Promise.all([
|
||||||
? {
|
balena.models.device.get(params.uuid, {
|
||||||
$expand: {
|
$expand: {
|
||||||
device_tag: {
|
device_tag: {
|
||||||
$select: ['tag_key', 'value'],
|
$select: ['tag_key', 'value'],
|
||||||
},
|
|
||||||
...expandForAppName.$expand,
|
|
||||||
},
|
},
|
||||||
}
|
...expandForAppName.$expand,
|
||||||
: {
|
|
||||||
$select: [
|
|
||||||
'device_name',
|
|
||||||
'id',
|
|
||||||
'overall_status',
|
|
||||||
'is_online',
|
|
||||||
'ip_address',
|
|
||||||
'mac_address',
|
|
||||||
'last_connectivity_event',
|
|
||||||
'uuid',
|
|
||||||
'supervisor_version',
|
|
||||||
'is_web_accessible',
|
|
||||||
'note',
|
|
||||||
'os_version',
|
|
||||||
'memory_usage',
|
|
||||||
'memory_total',
|
|
||||||
'public_address',
|
|
||||||
'storage_block_device',
|
|
||||||
'storage_usage',
|
|
||||||
'storage_total',
|
|
||||||
'cpu_usage',
|
|
||||||
'cpu_temp',
|
|
||||||
'cpu_id',
|
|
||||||
'is_undervolted',
|
|
||||||
],
|
|
||||||
...expandForAppName,
|
|
||||||
},
|
},
|
||||||
)) as ExtendedDevice;
|
}),
|
||||||
|
balena.models.device.get(params.uuid, {
|
||||||
|
$select: [
|
||||||
|
'overall_status',
|
||||||
|
'overall_progress',
|
||||||
|
'should_be_running__release',
|
||||||
|
],
|
||||||
|
}),
|
||||||
|
]);
|
||||||
|
|
||||||
|
device = {
|
||||||
|
...deviceBase,
|
||||||
|
...deviceComputed,
|
||||||
|
} as ExtendedDevice;
|
||||||
|
} else {
|
||||||
|
device = (await balena.models.device.get(params.uuid, {
|
||||||
|
$select: [
|
||||||
|
'device_name',
|
||||||
|
'id',
|
||||||
|
'overall_status',
|
||||||
|
'is_online',
|
||||||
|
'ip_address',
|
||||||
|
'mac_address',
|
||||||
|
'last_connectivity_event',
|
||||||
|
'uuid',
|
||||||
|
'supervisor_version',
|
||||||
|
'is_web_accessible',
|
||||||
|
'note',
|
||||||
|
'os_version',
|
||||||
|
'memory_usage',
|
||||||
|
'memory_total',
|
||||||
|
'public_address',
|
||||||
|
'storage_block_device',
|
||||||
|
'storage_usage',
|
||||||
|
'storage_total',
|
||||||
|
'cpu_usage',
|
||||||
|
'cpu_temp',
|
||||||
|
'cpu_id',
|
||||||
|
'is_undervolted',
|
||||||
|
],
|
||||||
|
...expandForAppName,
|
||||||
|
})) as ExtendedDevice;
|
||||||
|
}
|
||||||
|
|
||||||
if (options.view) {
|
if (options.view) {
|
||||||
const open = await import('open');
|
const open = await import('open');
|
||||||
|
@ -28,7 +28,6 @@ interface FlagsDef {
|
|||||||
'os-version'?: string;
|
'os-version'?: string;
|
||||||
drive?: string;
|
drive?: string;
|
||||||
config?: string;
|
config?: string;
|
||||||
help: void;
|
|
||||||
'provisioning-key-name'?: string;
|
'provisioning-key-name'?: string;
|
||||||
'provisioning-key-expiry-date'?: string;
|
'provisioning-key-expiry-date'?: string;
|
||||||
}
|
}
|
||||||
@ -100,7 +99,6 @@ export default class DeviceInitCmd extends Command {
|
|||||||
description:
|
description:
|
||||||
'expiry date assigned to generated provisioning api key (format: YYYY-MM-DD)',
|
'expiry date assigned to generated provisioning api key (format: YYYY-MM-DD)',
|
||||||
}),
|
}),
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
@ -157,7 +155,7 @@ export default class DeviceInitCmd extends Command {
|
|||||||
try {
|
try {
|
||||||
logger.logDebug(`Process failed, removing device ${device.uuid}`);
|
logger.logDebug(`Process failed, removing device ${device.uuid}`);
|
||||||
await balena.models.device.remove(device.uuid);
|
await balena.models.device.remove(device.uuid);
|
||||||
} catch (e) {
|
} catch {
|
||||||
// Ignore removal failures, and throw original error
|
// Ignore removal failures, and throw original error
|
||||||
}
|
}
|
||||||
throw e;
|
throw e;
|
||||||
|
@ -37,6 +37,7 @@ const devicesSelectFields = {
|
|||||||
|
|
||||||
export default class DeviceListCmd extends Command {
|
export default class DeviceListCmd extends Command {
|
||||||
public static aliases = ['devices'];
|
public static aliases = ['devices'];
|
||||||
|
public static deprecateAliases = true;
|
||||||
|
|
||||||
public static description = stripIndent`
|
public static description = stripIndent`
|
||||||
List all devices.
|
List all devices.
|
||||||
@ -58,7 +59,6 @@ export default class DeviceListCmd extends Command {
|
|||||||
public static flags = {
|
public static flags = {
|
||||||
fleet: cf.fleet,
|
fleet: cf.fleet,
|
||||||
json: cf.json,
|
json: cf.json,
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static primary = true;
|
public static primary = true;
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Flags, Args, Command } from '@oclif/core';
|
import { Flags, Args, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
||||||
|
|
||||||
export default class DeviceLocalModeCmd extends Command {
|
export default class DeviceLocalModeCmd extends Command {
|
||||||
@ -54,7 +53,6 @@ export default class DeviceLocalModeCmd extends Command {
|
|||||||
description: 'output boolean indicating local mode status',
|
description: 'output boolean indicating local mode status',
|
||||||
exclusive: ['enable', 'disable'],
|
exclusive: ['enable', 'disable'],
|
||||||
}),
|
}),
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
@ -16,13 +16,15 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Flags, Args, Command } from '@oclif/core';
|
import { Flags, Args, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
||||||
import type { LogMessage } from 'balena-sdk';
|
import type { LogMessage } from 'balena-sdk';
|
||||||
|
|
||||||
const MAX_RETRY = 1000;
|
const MAX_RETRY = 1000;
|
||||||
|
|
||||||
export default class LogsCmd extends Command {
|
export default class DeviceLogsCmd extends Command {
|
||||||
|
public static aliases = ['logs'];
|
||||||
|
public static deprecateAliases = true;
|
||||||
|
|
||||||
public static description = stripIndent`
|
public static description = stripIndent`
|
||||||
Show device logs.
|
Show device logs.
|
||||||
|
|
||||||
@ -42,15 +44,15 @@ export default class LogsCmd extends Command {
|
|||||||
Note: --service and --system flags must come after the device parameter, as per examples.
|
Note: --service and --system flags must come after the device parameter, as per examples.
|
||||||
`;
|
`;
|
||||||
public static examples = [
|
public static examples = [
|
||||||
'$ balena logs 23c73a1',
|
'$ balena device logs 23c73a1',
|
||||||
'$ balena logs 23c73a1 --tail',
|
'$ balena device logs 23c73a1 --tail',
|
||||||
'',
|
'',
|
||||||
'$ balena logs 192.168.0.31',
|
'$ balena device logs 192.168.0.31',
|
||||||
'$ balena logs 192.168.0.31 --service my-service',
|
'$ balena device logs 192.168.0.31 --service my-service',
|
||||||
'$ balena logs 192.168.0.31 --service my-service-1 --service my-service-2',
|
'$ balena device logs 192.168.0.31 --service my-service-1 --service my-service-2',
|
||||||
'',
|
'',
|
||||||
'$ balena logs 23c73a1.local --system',
|
'$ balena device logs 23c73a1.local --system',
|
||||||
'$ balena logs 23c73a1.local --system --service my-service',
|
'$ balena device logs 23c73a1.local --system --service my-service',
|
||||||
];
|
];
|
||||||
|
|
||||||
public static args = {
|
public static args = {
|
||||||
@ -84,13 +86,12 @@ export default class LogsCmd extends Command {
|
|||||||
'Only show system logs. This can be used in combination with --service.',
|
'Only show system logs. This can be used in combination with --service.',
|
||||||
char: 'S',
|
char: 'S',
|
||||||
}),
|
}),
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static primary = true;
|
public static primary = true;
|
||||||
|
|
||||||
public async run() {
|
public async run() {
|
||||||
const { args: params, flags: options } = await this.parse(LogsCmd);
|
const { args: params, flags: options } = await this.parse(DeviceLogsCmd);
|
||||||
|
|
||||||
const balena = getBalenaSdk();
|
const balena = getBalenaSdk();
|
||||||
const { serviceIdToName } = await import('../../utils/cloud');
|
const { serviceIdToName } = await import('../../utils/cloud');
|
||||||
@ -134,7 +135,7 @@ export default class LogsCmd extends Command {
|
|||||||
logger.logDebug('Checking we can access device');
|
logger.logDebug('Checking we can access device');
|
||||||
try {
|
try {
|
||||||
await deviceApi.ping();
|
await deviceApi.ping();
|
||||||
} catch (e) {
|
} catch {
|
||||||
const { ExpectedError } = await import('../../errors');
|
const { ExpectedError } = await import('../../errors');
|
||||||
throw new ExpectedError(
|
throw new ExpectedError(
|
||||||
`Cannot access device at address ${params.device}. Device may not be in local mode.`,
|
`Cannot access device at address ${params.device}. Device may not be in local mode.`,
|
@ -55,7 +55,6 @@ export default class DeviceMoveCmd extends Command {
|
|||||||
|
|
||||||
public static flags = {
|
public static flags = {
|
||||||
fleet: cf.fleet,
|
fleet: cf.fleet,
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
@ -20,7 +20,10 @@ import { ExpectedError } from '../../errors';
|
|||||||
import * as cf from '../../utils/common-flags';
|
import * as cf from '../../utils/common-flags';
|
||||||
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
||||||
|
|
||||||
export default class NoteCmd extends Command {
|
export default class DeviceNoteCmd extends Command {
|
||||||
|
public static aliases = ['notes'];
|
||||||
|
public static deprecateAliases = true;
|
||||||
|
|
||||||
public static description = stripIndent`
|
public static description = stripIndent`
|
||||||
Set a device note.
|
Set a device note.
|
||||||
|
|
||||||
@ -31,8 +34,8 @@ export default class NoteCmd extends Command {
|
|||||||
`;
|
`;
|
||||||
|
|
||||||
public static examples = [
|
public static examples = [
|
||||||
'$ balena note "My useful note" --device 7cf02a6',
|
'$ balena device note "My useful note" --device 7cf02a6',
|
||||||
'$ cat note.txt | balena note --device 7cf02a6',
|
'$ cat note.txt | balena device note --device 7cf02a6',
|
||||||
];
|
];
|
||||||
|
|
||||||
public static args = {
|
public static args = {
|
||||||
@ -47,13 +50,12 @@ export default class NoteCmd extends Command {
|
|||||||
exclusive: ['device'],
|
exclusive: ['device'],
|
||||||
hidden: true,
|
hidden: true,
|
||||||
}),
|
}),
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
|
||||||
public async run() {
|
public async run() {
|
||||||
const { args: params, flags: options } = await this.parse(NoteCmd);
|
const { args: params, flags: options } = await this.parse(DeviceNoteCmd);
|
||||||
|
|
||||||
if (params.note?.length === 0) {
|
if (params.note?.length === 0) {
|
||||||
throw new ExpectedError('Missing note content');
|
throw new ExpectedError('Missing note content');
|
@ -20,6 +20,7 @@ import * as cf from '../../utils/common-flags';
|
|||||||
import { getBalenaSdk, stripIndent, getCliForm } from '../../utils/lazy';
|
import { getBalenaSdk, stripIndent, getCliForm } from '../../utils/lazy';
|
||||||
import type { Device } from 'balena-sdk';
|
import type { Device } from 'balena-sdk';
|
||||||
import { ExpectedError } from '../../errors';
|
import { ExpectedError } from '../../errors';
|
||||||
|
import { getExpandedProp } from '../../utils/pine';
|
||||||
|
|
||||||
export default class DeviceOsUpdateCmd extends Command {
|
export default class DeviceOsUpdateCmd extends Command {
|
||||||
public static description = stripIndent`
|
public static description = stripIndent`
|
||||||
@ -57,7 +58,6 @@ export default class DeviceOsUpdateCmd extends Command {
|
|||||||
exclusive: ['version'],
|
exclusive: ['version'],
|
||||||
}),
|
}),
|
||||||
yes: cf.yes,
|
yes: cf.yes,
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
@ -127,27 +127,64 @@ export default class DeviceOsUpdateCmd extends Command {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
const choices = await Promise.all(
|
||||||
|
hupVersionInfo.versions.map(async (version) => {
|
||||||
|
const takeoverRequired =
|
||||||
|
(await sdk.models.os.getOsUpdateType(
|
||||||
|
getExpandedProp(is_of__device_type, 'slug')!,
|
||||||
|
currentOsVersion,
|
||||||
|
version,
|
||||||
|
)) === 'takeover';
|
||||||
|
|
||||||
|
return {
|
||||||
|
name: `${version}${hupVersionInfo.recommended === version ? ' (recommended)' : ''}${takeoverRequired ? ' ADVANCED UPDATE: Requires disk re-partitioning with no rollback option' : ''}`,
|
||||||
|
value: version,
|
||||||
|
};
|
||||||
|
}),
|
||||||
|
);
|
||||||
targetOsVersion = await getCliForm().ask({
|
targetOsVersion = await getCliForm().ask({
|
||||||
message: 'Target OS version',
|
message: 'Target OS version',
|
||||||
type: 'list',
|
type: 'list',
|
||||||
choices: hupVersionInfo.versions.map((version) => ({
|
choices,
|
||||||
name:
|
|
||||||
hupVersionInfo.recommended === version
|
|
||||||
? `${version} (recommended)`
|
|
||||||
: version,
|
|
||||||
value: version,
|
|
||||||
})),
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const takeoverRequired =
|
||||||
|
(await sdk.models.os.getOsUpdateType(
|
||||||
|
getExpandedProp(is_of__device_type, 'slug')!,
|
||||||
|
currentOsVersion,
|
||||||
|
targetOsVersion,
|
||||||
|
)) === 'takeover';
|
||||||
const patterns = await import('../../utils/patterns');
|
const patterns = await import('../../utils/patterns');
|
||||||
|
// Warn the user if the update requires a takeover
|
||||||
|
if (takeoverRequired) {
|
||||||
|
await patterns.confirm(
|
||||||
|
options.yes || false,
|
||||||
|
stripIndent`Before you proceed, note that this update process is different from a regular HostOS Update:
|
||||||
|
DATA LOSS: This update requires disk re-partitioning, which will erase all data stored on the device.
|
||||||
|
NO ROLLBACK: Unlike our HostOS update mechanism, this process does not allow reverting to a previous version in case of failure.
|
||||||
|
Make sure to back up all important data before continuing. For more details, check our documentation: https://docs.balena.io/reference/OS/updates/update-process/
|
||||||
|
`,
|
||||||
|
);
|
||||||
|
}
|
||||||
// Confirm and start update
|
// Confirm and start update
|
||||||
await patterns.confirm(
|
await patterns.confirm(
|
||||||
options.yes || false,
|
options.yes || false,
|
||||||
'Host OS updates require a device restart when they complete. Are you sure you want to proceed?',
|
'Host OS updates require a device restart when they complete. Are you sure you want to proceed?',
|
||||||
);
|
);
|
||||||
|
|
||||||
await sdk.models.device.startOsUpdate(uuid, targetOsVersion);
|
await sdk.models.device
|
||||||
await patterns.awaitDeviceOsUpdate(uuid, targetOsVersion);
|
.startOsUpdate(uuid, targetOsVersion, {
|
||||||
|
runDetached: true,
|
||||||
|
})
|
||||||
|
.then(() => {
|
||||||
|
console.log(
|
||||||
|
`The balena OS update has started. You can keep track of the progress via the dashboard.\n` +
|
||||||
|
`To open the dashboard page related to a device via the CLI, you can use \`balena device UUID --view\``,
|
||||||
|
);
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
console.error(`Failed to start OS update for device ${uuid}:`, error);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Args, Command } from '@oclif/core';
|
import { Args, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
||||||
import { getExpandedProp } from '../../utils/pine';
|
import { getExpandedProp } from '../../utils/pine';
|
||||||
|
|
||||||
@ -43,10 +42,6 @@ export default class DevicePinCmd extends Command {
|
|||||||
}),
|
}),
|
||||||
};
|
};
|
||||||
|
|
||||||
public static flags = {
|
|
||||||
help: cf.help,
|
|
||||||
};
|
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
|
||||||
public async run() {
|
public async run() {
|
||||||
@ -56,7 +51,7 @@ export default class DevicePinCmd extends Command {
|
|||||||
|
|
||||||
const device = await balena.models.device.get(params.uuid, {
|
const device = await balena.models.device.get(params.uuid, {
|
||||||
$expand: {
|
$expand: {
|
||||||
should_be_running__release: {
|
is_pinned_on__release: {
|
||||||
$select: 'commit',
|
$select: 'commit',
|
||||||
},
|
},
|
||||||
belongs_to__application: {
|
belongs_to__application: {
|
||||||
@ -66,7 +61,7 @@ export default class DevicePinCmd extends Command {
|
|||||||
});
|
});
|
||||||
|
|
||||||
const pinnedRelease = getExpandedProp(
|
const pinnedRelease = getExpandedProp(
|
||||||
device.should_be_running__release,
|
device.is_pinned_on__release,
|
||||||
'commit',
|
'commit',
|
||||||
);
|
);
|
||||||
const appSlug = getExpandedProp(device.belongs_to__application, 'slug');
|
const appSlug = getExpandedProp(device.belongs_to__application, 'slug');
|
||||||
|
@ -17,7 +17,6 @@
|
|||||||
|
|
||||||
import { Flags, Args, Command } from '@oclif/core';
|
import { Flags, Args, Command } from '@oclif/core';
|
||||||
import { ExpectedError } from '../../errors';
|
import { ExpectedError } from '../../errors';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
||||||
|
|
||||||
export default class DevicePublicUrlCmd extends Command {
|
export default class DevicePublicUrlCmd extends Command {
|
||||||
@ -56,7 +55,6 @@ export default class DevicePublicUrlCmd extends Command {
|
|||||||
description: 'determine if public URL is enabled',
|
description: 'determine if public URL is enabled',
|
||||||
exclusive: ['enable', 'disable'],
|
exclusive: ['enable', 'disable'],
|
||||||
}),
|
}),
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Args, Command } from '@oclif/core';
|
import { Args, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { getBalenaSdk, getCliUx, stripIndent } from '../../utils/lazy';
|
import { getBalenaSdk, getCliUx, stripIndent } from '../../utils/lazy';
|
||||||
|
|
||||||
export default class DevicePurgeCmd extends Command {
|
export default class DevicePurgeCmd extends Command {
|
||||||
@ -41,10 +40,6 @@ export default class DevicePurgeCmd extends Command {
|
|||||||
}),
|
}),
|
||||||
};
|
};
|
||||||
|
|
||||||
public static flags = {
|
|
||||||
help: cf.help,
|
|
||||||
};
|
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
|
||||||
public async run() {
|
public async run() {
|
||||||
|
@ -36,7 +36,6 @@ export default class DeviceRebootCmd extends Command {
|
|||||||
|
|
||||||
public static flags = {
|
public static flags = {
|
||||||
force: cf.force,
|
force: cf.force,
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Flags, Command } from '@oclif/core';
|
import { Flags, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import * as ca from '../../utils/common-args';
|
import * as ca from '../../utils/common-args';
|
||||||
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
||||||
import { applicationIdInfo } from '../../utils/messages';
|
import { applicationIdInfo } from '../../utils/messages';
|
||||||
@ -50,9 +49,8 @@ export default class DeviceRegisterCmd extends Command {
|
|||||||
}),
|
}),
|
||||||
deviceType: Flags.string({
|
deviceType: Flags.string({
|
||||||
description:
|
description:
|
||||||
"device type slug (run 'balena devices supported' for possible values)",
|
"device type slug (run 'balena device-type list' for possible values)",
|
||||||
}),
|
}),
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
@ -78,6 +76,6 @@ export default class DeviceRegisterCmd extends Command {
|
|||||||
options.deviceType,
|
options.deviceType,
|
||||||
);
|
);
|
||||||
|
|
||||||
return result && result.uuid;
|
return result.uuid;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Args, Command } from '@oclif/core';
|
import { Args, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { getBalenaSdk, stripIndent, getCliForm } from '../../utils/lazy';
|
import { getBalenaSdk, stripIndent, getCliForm } from '../../utils/lazy';
|
||||||
|
|
||||||
export default class DeviceRenameCmd extends Command {
|
export default class DeviceRenameCmd extends Command {
|
||||||
@ -42,10 +41,6 @@ export default class DeviceRenameCmd extends Command {
|
|||||||
}),
|
}),
|
||||||
};
|
};
|
||||||
|
|
||||||
public static flags = {
|
|
||||||
help: cf.help,
|
|
||||||
};
|
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
|
||||||
public async run() {
|
public async run() {
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Flags, Args, Command } from '@oclif/core';
|
import { Flags, Args, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { getBalenaSdk, getCliUx, stripIndent } from '../../utils/lazy';
|
import { getBalenaSdk, getCliUx, stripIndent } from '../../utils/lazy';
|
||||||
import type {
|
import type {
|
||||||
BalenaSDK,
|
BalenaSDK,
|
||||||
@ -58,7 +57,6 @@ export default class DeviceRestartCmd extends Command {
|
|||||||
'comma-separated list (no blank spaces) of service names to restart',
|
'comma-separated list (no blank spaces) of service names to restart',
|
||||||
char: 's',
|
char: 's',
|
||||||
}),
|
}),
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
@ -156,7 +154,7 @@ export default class DeviceRestartCmd extends Command {
|
|||||||
|
|
||||||
async restartAllServices(balena: BalenaSDK, deviceUuid: string) {
|
async restartAllServices(balena: BalenaSDK, deviceUuid: string) {
|
||||||
// Note: device.restartApplication throws `BalenaDeviceNotFound: Device not found` if device not online.
|
// Note: device.restartApplication throws `BalenaDeviceNotFound: Device not found` if device not online.
|
||||||
// Need to use device.get first to distinguish between non-existant and offline devices.
|
// Need to use device.get first to distinguish between non-existant and disconnected devices.
|
||||||
// Remove this workaround when SDK issue resolved: https://github.com/balena-io/balena-sdk/issues/649
|
// Remove this workaround when SDK issue resolved: https://github.com/balena-io/balena-sdk/issues/649
|
||||||
const { instanceOf, ExpectedError } = await import('../../errors');
|
const { instanceOf, ExpectedError } = await import('../../errors');
|
||||||
try {
|
try {
|
||||||
|
@ -44,7 +44,6 @@ export default class DeviceRmCmd extends Command {
|
|||||||
|
|
||||||
public static flags = {
|
public static flags = {
|
||||||
yes: cf.yes,
|
yes: cf.yes,
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
@ -37,7 +37,6 @@ export default class DeviceShutdownCmd extends Command {
|
|||||||
|
|
||||||
public static flags = {
|
public static flags = {
|
||||||
force: cf.force,
|
force: cf.force,
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
@ -16,14 +16,16 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Flags, Args, Command } from '@oclif/core';
|
import { Flags, Args, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
||||||
import {
|
import {
|
||||||
parseAsInteger,
|
parseAsInteger,
|
||||||
validateLocalHostnameOrIp,
|
validateLocalHostnameOrIp,
|
||||||
} from '../../utils/validation';
|
} from '../../utils/validation';
|
||||||
|
|
||||||
export default class SshCmd extends Command {
|
export default class DeviceSSHCmd extends Command {
|
||||||
|
public static aliases = ['ssh'];
|
||||||
|
public static deprecateAliases = true;
|
||||||
|
|
||||||
public static description = stripIndent`
|
public static description = stripIndent`
|
||||||
Open a SSH prompt on a device's host OS or service container.
|
Open a SSH prompt on a device's host OS or service container.
|
||||||
|
|
||||||
@ -52,14 +54,14 @@ export default class SshCmd extends Command {
|
|||||||
`;
|
`;
|
||||||
|
|
||||||
public static examples = [
|
public static examples = [
|
||||||
'$ balena ssh MyFleet',
|
'$ balena device ssh MyFleet',
|
||||||
'$ balena ssh f49cefd',
|
'$ balena device ssh f49cefd',
|
||||||
'$ balena ssh f49cefd my-service',
|
'$ balena device ssh f49cefd my-service',
|
||||||
'$ balena ssh f49cefd --port <port>',
|
'$ balena device ssh f49cefd --port <port>',
|
||||||
'$ balena ssh 192.168.0.1 --verbose',
|
'$ balena device ssh 192.168.0.1 --verbose',
|
||||||
'$ balena ssh f49cefd.local my-service',
|
'$ balena device ssh f49cefd.local my-service',
|
||||||
'$ echo "uptime; exit;" | balena ssh f49cefd',
|
'$ echo "uptime; exit;" | balena device ssh f49cefd',
|
||||||
'$ echo "uptime; exit;" | balena ssh 192.168.0.1 myService',
|
'$ echo "uptime; exit;" | balena device ssh 192.168.0.1 myService',
|
||||||
];
|
];
|
||||||
|
|
||||||
public static args = {
|
public static args = {
|
||||||
@ -80,7 +82,7 @@ export default class SshCmd extends Command {
|
|||||||
SSH server port number (default 22222) if the target is an IP address or .local
|
SSH server port number (default 22222) if the target is an IP address or .local
|
||||||
hostname. Otherwise, port number for the balenaCloud gateway (default 22).`,
|
hostname. Otherwise, port number for the balenaCloud gateway (default 22).`,
|
||||||
char: 'p',
|
char: 'p',
|
||||||
parse: async (p) => parseAsInteger(p, 'port'),
|
parse: (p) => parseAsInteger(p, 'port'),
|
||||||
}),
|
}),
|
||||||
tty: Flags.boolean({
|
tty: Flags.boolean({
|
||||||
default: false,
|
default: false,
|
||||||
@ -97,25 +99,25 @@ export default class SshCmd extends Command {
|
|||||||
default: false,
|
default: false,
|
||||||
description: 'bypass global proxy configuration for the ssh connection',
|
description: 'bypass global proxy configuration for the ssh connection',
|
||||||
}),
|
}),
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static primary = true;
|
public static primary = true;
|
||||||
public static offlineCompatible = true;
|
public static offlineCompatible = true;
|
||||||
|
|
||||||
public async run() {
|
public async run() {
|
||||||
const { args: params, flags: options } = await this.parse(SshCmd);
|
const { args: params, flags: options } = await this.parse(DeviceSSHCmd);
|
||||||
|
|
||||||
// Local connection
|
// Local connection
|
||||||
if (validateLocalHostnameOrIp(params.fleetOrDevice)) {
|
if (validateLocalHostnameOrIp(params.fleetOrDevice)) {
|
||||||
const { performLocalDeviceSSH } = await import('../../utils/device/ssh');
|
const { performLocalDeviceSSH } = await import('../../utils/device/ssh');
|
||||||
return await performLocalDeviceSSH({
|
await performLocalDeviceSSH({
|
||||||
hostname: params.fleetOrDevice,
|
hostname: params.fleetOrDevice,
|
||||||
port: options.port || 'local',
|
port: options.port || 'local',
|
||||||
forceTTY: options.tty,
|
forceTTY: options.tty,
|
||||||
verbose: options.verbose,
|
verbose: options.verbose,
|
||||||
service: params.service,
|
service: params.service,
|
||||||
});
|
});
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Remote connection
|
// Remote connection
|
||||||
@ -131,7 +133,7 @@ export default class SshCmd extends Command {
|
|||||||
const useProxy = !!proxyConfig && !options.noproxy;
|
const useProxy = !!proxyConfig && !options.noproxy;
|
||||||
|
|
||||||
// this will be a tunnelled SSH connection...
|
// this will be a tunnelled SSH connection...
|
||||||
await checkNotUsingOfflineMode();
|
checkNotUsingOfflineMode();
|
||||||
await checkLoggedIn();
|
await checkLoggedIn();
|
||||||
const deviceUuid = await getOnlineTargetDeviceUuid(
|
const deviceUuid = await getOnlineTargetDeviceUuid(
|
||||||
sdk,
|
sdk,
|
@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Args, Command } from '@oclif/core';
|
import { Args, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { getBalenaSdk, getCliUx, stripIndent } from '../../utils/lazy';
|
import { getBalenaSdk, getCliUx, stripIndent } from '../../utils/lazy';
|
||||||
import type { BalenaSDK } from 'balena-sdk';
|
import type { BalenaSDK } from 'balena-sdk';
|
||||||
|
|
||||||
@ -45,10 +44,6 @@ export default class DeviceStartServiceCmd extends Command {
|
|||||||
}),
|
}),
|
||||||
};
|
};
|
||||||
|
|
||||||
public static flags = {
|
|
||||||
help: cf.help,
|
|
||||||
};
|
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
|
||||||
public async run() {
|
public async run() {
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Args, Command } from '@oclif/core';
|
import { Args, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { getBalenaSdk, getCliUx, stripIndent } from '../../utils/lazy';
|
import { getBalenaSdk, getCliUx, stripIndent } from '../../utils/lazy';
|
||||||
import type { BalenaSDK } from 'balena-sdk';
|
import type { BalenaSDK } from 'balena-sdk';
|
||||||
|
|
||||||
@ -45,10 +44,6 @@ export default class DeviceStopServiceCmd extends Command {
|
|||||||
}),
|
}),
|
||||||
};
|
};
|
||||||
|
|
||||||
public static flags = {
|
|
||||||
help: cf.help,
|
|
||||||
};
|
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
|
||||||
public async run() {
|
public async run() {
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Args, Command } from '@oclif/core';
|
import { Args, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
||||||
|
|
||||||
export default class DeviceTrackFleetCmd extends Command {
|
export default class DeviceTrackFleetCmd extends Command {
|
||||||
@ -34,10 +33,6 @@ export default class DeviceTrackFleetCmd extends Command {
|
|||||||
}),
|
}),
|
||||||
};
|
};
|
||||||
|
|
||||||
public static flags = {
|
|
||||||
help: cf.help,
|
|
||||||
};
|
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
|
||||||
public async run() {
|
public async run() {
|
||||||
|
@ -21,13 +21,15 @@ import {
|
|||||||
InvalidPortMappingError,
|
InvalidPortMappingError,
|
||||||
ExpectedError,
|
ExpectedError,
|
||||||
} from '../../errors';
|
} from '../../errors';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
||||||
import { lowercaseIfSlug } from '../../utils/normalization';
|
import { lowercaseIfSlug } from '../../utils/normalization';
|
||||||
|
|
||||||
import type { Server, Socket } from 'net';
|
import type { Server, Socket } from 'net';
|
||||||
|
|
||||||
export default class TunnelCmd extends Command {
|
export default class DeviceTunnelCmd extends Command {
|
||||||
|
public static aliases = ['tunnel'];
|
||||||
|
public static deprecateAliases = true;
|
||||||
|
|
||||||
public static description = stripIndent`
|
public static description = stripIndent`
|
||||||
Tunnel local ports to your balenaOS device.
|
Tunnel local ports to your balenaOS device.
|
||||||
|
|
||||||
@ -54,19 +56,19 @@ export default class TunnelCmd extends Command {
|
|||||||
|
|
||||||
public static examples = [
|
public static examples = [
|
||||||
'# map remote port 22222 to localhost:22222',
|
'# map remote port 22222 to localhost:22222',
|
||||||
'$ balena tunnel myFleet -p 22222',
|
'$ balena device tunnel myFleet -p 22222',
|
||||||
'',
|
'',
|
||||||
'# map remote port 22222 to localhost:222',
|
'# map remote port 22222 to localhost:222',
|
||||||
'$ balena tunnel 2ead211 -p 22222:222',
|
'$ balena device tunnel 2ead211 -p 22222:222',
|
||||||
'',
|
'',
|
||||||
'# map remote port 22222 to any address on your host machine, port 22222',
|
'# map remote port 22222 to any address on your host machine, port 22222',
|
||||||
'$ balena tunnel 1546690 -p 22222:0.0.0.0',
|
'$ balena device tunnel 1546690 -p 22222:0.0.0.0',
|
||||||
'',
|
'',
|
||||||
'# map remote port 22222 to any address on your host machine, port 222',
|
'# map remote port 22222 to any address on your host machine, port 222',
|
||||||
'$ balena tunnel myFleet -p 22222:0.0.0.0:222',
|
'$ balena device tunnel myFleet -p 22222:0.0.0.0:222',
|
||||||
'',
|
'',
|
||||||
'# multiple port tunnels can be specified at any one time',
|
'# multiple port tunnels can be specified at any one time',
|
||||||
'$ balena tunnel myFleet -p 8080:3000 -p 8081:9000',
|
'$ balena device tunnel myFleet -p 8080:3000 -p 8081:9000',
|
||||||
];
|
];
|
||||||
|
|
||||||
public static args = {
|
public static args = {
|
||||||
@ -84,14 +86,13 @@ export default class TunnelCmd extends Command {
|
|||||||
char: 'p',
|
char: 'p',
|
||||||
multiple: true,
|
multiple: true,
|
||||||
}),
|
}),
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static primary = true;
|
public static primary = true;
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
|
||||||
public async run() {
|
public async run() {
|
||||||
const { args: params, flags: options } = await this.parse(TunnelCmd);
|
const { args: params, flags: options } = await this.parse(DeviceTunnelCmd);
|
||||||
|
|
||||||
const Logger = await import('../../utils/logger');
|
const Logger = await import('../../utils/logger');
|
||||||
|
|
2
src/commands/env/list.ts
vendored
2
src/commands/env/list.ts
vendored
@ -47,6 +47,7 @@ interface ServiceEnvironmentVariableInfo
|
|||||||
|
|
||||||
export default class EnvListCmd extends Command {
|
export default class EnvListCmd extends Command {
|
||||||
public static aliases = ['envs'];
|
public static aliases = ['envs'];
|
||||||
|
public static deprecateAliases = true;
|
||||||
|
|
||||||
public static description = stripIndent`
|
public static description = stripIndent`
|
||||||
List the environment or config variables of a fleet, device or service.
|
List the environment or config variables of a fleet, device or service.
|
||||||
@ -104,7 +105,6 @@ export default class EnvListCmd extends Command {
|
|||||||
exclusive: ['service'],
|
exclusive: ['service'],
|
||||||
}),
|
}),
|
||||||
device: { ...cf.device, exclusive: ['fleet'] },
|
device: { ...cf.device, exclusive: ['fleet'] },
|
||||||
help: cf.help,
|
|
||||||
json: cf.json,
|
json: cf.json,
|
||||||
service: { ...cf.service, exclusive: ['config'] },
|
service: { ...cf.service, exclusive: ['config'] },
|
||||||
};
|
};
|
||||||
|
4
src/commands/env/rename.ts
vendored
4
src/commands/env/rename.ts
vendored
@ -15,7 +15,6 @@
|
|||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
import { Args, Command } from '@oclif/core';
|
import { Args, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import * as ec from '../../utils/env-common';
|
import * as ec from '../../utils/env-common';
|
||||||
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
||||||
import { parseAsInteger } from '../../utils/validation';
|
import { parseAsInteger } from '../../utils/validation';
|
||||||
@ -42,7 +41,7 @@ export default class EnvRenameCmd extends Command {
|
|||||||
id: Args.integer({
|
id: Args.integer({
|
||||||
required: true,
|
required: true,
|
||||||
description: "variable's numeric database ID",
|
description: "variable's numeric database ID",
|
||||||
parse: async (input) => parseAsInteger(input, 'id'),
|
parse: (input) => parseAsInteger(input, 'id'),
|
||||||
}),
|
}),
|
||||||
value: Args.string({
|
value: Args.string({
|
||||||
required: true,
|
required: true,
|
||||||
@ -55,7 +54,6 @@ export default class EnvRenameCmd extends Command {
|
|||||||
config: ec.booleanConfig,
|
config: ec.booleanConfig,
|
||||||
device: ec.booleanDevice,
|
device: ec.booleanDevice,
|
||||||
service: ec.booleanService,
|
service: ec.booleanService,
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public async run() {
|
public async run() {
|
||||||
|
2
src/commands/env/rm.ts
vendored
2
src/commands/env/rm.ts
vendored
@ -46,7 +46,7 @@ export default class EnvRmCmd extends Command {
|
|||||||
id: Args.integer({
|
id: Args.integer({
|
||||||
required: true,
|
required: true,
|
||||||
description: "variable's numeric database ID",
|
description: "variable's numeric database ID",
|
||||||
parse: async (input) => parseAsInteger(input, 'id'),
|
parse: (input) => parseAsInteger(input, 'id'),
|
||||||
}),
|
}),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -25,7 +25,6 @@ import { applicationIdInfo } from '../../utils/messages';
|
|||||||
interface FlagsDef {
|
interface FlagsDef {
|
||||||
fleet?: string;
|
fleet?: string;
|
||||||
device?: string; // device UUID
|
device?: string; // device UUID
|
||||||
help: void;
|
|
||||||
quiet: boolean;
|
quiet: boolean;
|
||||||
service?: string; // service name
|
service?: string; // service name
|
||||||
}
|
}
|
||||||
@ -35,11 +34,14 @@ interface ArgsDef {
|
|||||||
value?: string;
|
value?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default class EnvAddCmd extends Command {
|
export default class EnvSetCmd extends Command {
|
||||||
public static description = stripIndent`
|
public static aliases = ['env add'];
|
||||||
Add env or config variable to fleets, devices or services.
|
public static deprecateAliases = true;
|
||||||
|
|
||||||
Add an environment or config variable to one or more fleets, devices or
|
public static description = stripIndent`
|
||||||
|
Add or update env or config variable to fleets, devices or services.
|
||||||
|
|
||||||
|
Add or update an environment or config variable to one or more fleets, devices or
|
||||||
services, as selected by the respective command-line options. Either the
|
services, as selected by the respective command-line options. Either the
|
||||||
--fleet or the --device option must be provided, and either may be be
|
--fleet or the --device option must be provided, and either may be be
|
||||||
used alongside the --service option to define a service-specific variable.
|
used alongside the --service option to define a service-specific variable.
|
||||||
@ -66,15 +68,15 @@ export default class EnvAddCmd extends Command {
|
|||||||
`;
|
`;
|
||||||
|
|
||||||
public static examples = [
|
public static examples = [
|
||||||
'$ balena env add TERM --fleet MyFleet',
|
'$ balena env set TERM --fleet MyFleet',
|
||||||
'$ balena env add EDITOR vim -f myorg/myfleet',
|
'$ balena env set EDITOR vim -f myorg/myfleet',
|
||||||
'$ balena env add EDITOR vim --fleet MyFleet,MyFleet2',
|
'$ balena env set EDITOR vim --fleet MyFleet,MyFleet2',
|
||||||
'$ balena env add EDITOR vim --fleet MyFleet --service MyService',
|
'$ balena env set EDITOR vim --fleet MyFleet --service MyService',
|
||||||
'$ balena env add EDITOR vim --fleet MyFleet,MyFleet2 --service MyService,MyService2',
|
'$ balena env set EDITOR vim --fleet MyFleet,MyFleet2 --service MyService,MyService2',
|
||||||
'$ balena env add EDITOR vim --device 7cf02a6',
|
'$ balena env set EDITOR vim --device 7cf02a6',
|
||||||
'$ balena env add EDITOR vim --device 7cf02a6,d6f1433',
|
'$ balena env set EDITOR vim --device 7cf02a6,d6f1433',
|
||||||
'$ balena env add EDITOR vim --device 7cf02a6 --service MyService',
|
'$ balena env set EDITOR vim --device 7cf02a6 --service MyService',
|
||||||
'$ balena env add EDITOR vim --device 7cf02a6,d6f1433 --service MyService,MyService2',
|
'$ balena env set EDITOR vim --device 7cf02a6,d6f1433 --service MyService,MyService2',
|
||||||
];
|
];
|
||||||
|
|
||||||
public static args = {
|
public static args = {
|
||||||
@ -95,13 +97,12 @@ export default class EnvAddCmd extends Command {
|
|||||||
public static flags = {
|
public static flags = {
|
||||||
fleet: { ...cf.fleet, exclusive: ['device'] },
|
fleet: { ...cf.fleet, exclusive: ['device'] },
|
||||||
device: { ...cf.device, exclusive: ['fleet'] },
|
device: { ...cf.device, exclusive: ['fleet'] },
|
||||||
help: cf.help,
|
|
||||||
quiet: cf.quiet,
|
quiet: cf.quiet,
|
||||||
service: cf.service,
|
service: cf.service,
|
||||||
};
|
};
|
||||||
|
|
||||||
public async run() {
|
public async run() {
|
||||||
const { args: params, flags: options } = await this.parse(EnvAddCmd);
|
const { args: params, flags: options } = await this.parse(EnvSetCmd);
|
||||||
const cmd = this;
|
const cmd = this;
|
||||||
|
|
||||||
if (!options.fleet && !options.device) {
|
if (!options.fleet && !options.device) {
|
@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Flags, Args, Command } from '@oclif/core';
|
import { Flags, Args, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { stripIndent } from '../../utils/lazy';
|
import { stripIndent } from '../../utils/lazy';
|
||||||
|
|
||||||
export default class FleetCreateCmd extends Command {
|
export default class FleetCreateCmd extends Command {
|
||||||
@ -28,10 +27,10 @@ export default class FleetCreateCmd extends Command {
|
|||||||
You can specify the organization the fleet should belong to using
|
You can specify the organization the fleet should belong to using
|
||||||
the \`--organization\` option. The organization's handle, not its name,
|
the \`--organization\` option. The organization's handle, not its name,
|
||||||
should be provided. Organization handles can be listed with the
|
should be provided. Organization handles can be listed with the
|
||||||
\`balena orgs\` command.
|
\`balena organization list\` command.
|
||||||
|
|
||||||
The fleet's default device type is specified with the \`--type\` option.
|
The fleet's default device type is specified with the \`--type\` option.
|
||||||
The \`balena devices supported\` command can be used to list the available
|
The \`balena device-type list\` command can be used to list the available
|
||||||
device types.
|
device types.
|
||||||
|
|
||||||
Interactive dropdowns will be shown for selection if no device type or
|
Interactive dropdowns will be shown for selection if no device type or
|
||||||
@ -62,9 +61,8 @@ export default class FleetCreateCmd extends Command {
|
|||||||
type: Flags.string({
|
type: Flags.string({
|
||||||
char: 't',
|
char: 't',
|
||||||
description:
|
description:
|
||||||
'fleet device type (Check available types with `balena devices supported`)',
|
'fleet device type (Check available types with `balena device-type list`)',
|
||||||
}),
|
}),
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
@ -40,7 +40,6 @@ export default class FleetCmd extends Command {
|
|||||||
};
|
};
|
||||||
|
|
||||||
public static flags = {
|
public static flags = {
|
||||||
help: cf.help,
|
|
||||||
view: Flags.boolean({
|
view: Flags.boolean({
|
||||||
default: false,
|
default: false,
|
||||||
description: 'open fleet dashboard page',
|
description: 'open fleet dashboard page',
|
||||||
|
@ -28,6 +28,7 @@ interface ExtendedApplication extends ApplicationWithDeviceTypeSlug {
|
|||||||
|
|
||||||
export default class FleetListCmd extends Command {
|
export default class FleetListCmd extends Command {
|
||||||
public static aliases = ['fleets'];
|
public static aliases = ['fleets'];
|
||||||
|
public static deprecateAliases = true;
|
||||||
|
|
||||||
public static description = stripIndent`
|
public static description = stripIndent`
|
||||||
List all fleets.
|
List all fleets.
|
||||||
@ -41,7 +42,6 @@ export default class FleetListCmd extends Command {
|
|||||||
public static examples = ['$ balena fleet list'];
|
public static examples = ['$ balena fleet list'];
|
||||||
|
|
||||||
public static flags = {
|
public static flags = {
|
||||||
help: cf.help,
|
|
||||||
json: cf.json,
|
json: cf.json,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Args, Command } from '@oclif/core';
|
import { Args, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
||||||
import { getExpandedProp } from '../../utils/pine';
|
import { getExpandedProp } from '../../utils/pine';
|
||||||
|
|
||||||
@ -43,10 +42,6 @@ export default class FleetPinCmd extends Command {
|
|||||||
}),
|
}),
|
||||||
};
|
};
|
||||||
|
|
||||||
public static flags = {
|
|
||||||
help: cf.help,
|
|
||||||
};
|
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
|
||||||
public async run() {
|
public async run() {
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Command } from '@oclif/core';
|
import { Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import * as ca from '../../utils/common-args';
|
import * as ca from '../../utils/common-args';
|
||||||
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
||||||
import { applicationIdInfo } from '../../utils/messages';
|
import { applicationIdInfo } from '../../utils/messages';
|
||||||
@ -40,10 +39,6 @@ export default class FleetPurgeCmd extends Command {
|
|||||||
fleet: ca.fleetRequired,
|
fleet: ca.fleetRequired,
|
||||||
};
|
};
|
||||||
|
|
||||||
public static flags = {
|
|
||||||
help: cf.help,
|
|
||||||
};
|
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
|
||||||
public async run() {
|
public async run() {
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Args, Command } from '@oclif/core';
|
import { Args, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import * as ca from '../../utils/common-args';
|
import * as ca from '../../utils/common-args';
|
||||||
import { getBalenaSdk, stripIndent, getCliForm } from '../../utils/lazy';
|
import { getBalenaSdk, stripIndent, getCliForm } from '../../utils/lazy';
|
||||||
import { applicationIdInfo } from '../../utils/messages';
|
import { applicationIdInfo } from '../../utils/messages';
|
||||||
@ -46,10 +45,6 @@ export default class FleetRenameCmd extends Command {
|
|||||||
}),
|
}),
|
||||||
};
|
};
|
||||||
|
|
||||||
public static flags = {
|
|
||||||
help: cf.help,
|
|
||||||
};
|
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
|
||||||
public async run() {
|
public async run() {
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Command } from '@oclif/core';
|
import { Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import * as ca from '../../utils/common-args';
|
import * as ca from '../../utils/common-args';
|
||||||
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
||||||
import { applicationIdInfo } from '../../utils/messages';
|
import { applicationIdInfo } from '../../utils/messages';
|
||||||
@ -39,10 +38,6 @@ export default class FleetRestartCmd extends Command {
|
|||||||
fleet: ca.fleetRequired,
|
fleet: ca.fleetRequired,
|
||||||
};
|
};
|
||||||
|
|
||||||
public static flags = {
|
|
||||||
help: cf.help,
|
|
||||||
};
|
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
|
||||||
public async run() {
|
public async run() {
|
||||||
|
@ -44,7 +44,6 @@ export default class FleetRmCmd extends Command {
|
|||||||
|
|
||||||
public static flags = {
|
public static flags = {
|
||||||
yes: cf.yes,
|
yes: cf.yes,
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Args, Command } from '@oclif/core';
|
import { Args, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
import { getBalenaSdk, stripIndent } from '../../utils/lazy';
|
||||||
|
|
||||||
export default class FleetTrackLatestCmd extends Command {
|
export default class FleetTrackLatestCmd extends Command {
|
||||||
@ -37,10 +36,6 @@ export default class FleetTrackLatestCmd extends Command {
|
|||||||
}),
|
}),
|
||||||
};
|
};
|
||||||
|
|
||||||
public static flags = {
|
|
||||||
help: cf.help,
|
|
||||||
};
|
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
|
||||||
public async run() {
|
public async run() {
|
||||||
|
@ -65,7 +65,6 @@ export default class JoinCmd extends Command {
|
|||||||
description: 'the interval in minutes to check for updates',
|
description: 'the interval in minutes to check for updates',
|
||||||
char: 'i',
|
char: 'i',
|
||||||
}),
|
}),
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Args, Command } from '@oclif/core';
|
import { Args, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { stripIndent } from '../../utils/lazy';
|
import { stripIndent } from '../../utils/lazy';
|
||||||
import { parseAsLocalHostnameOrIp } from '../../utils/validation';
|
import { parseAsLocalHostnameOrIp } from '../../utils/validation';
|
||||||
|
|
||||||
@ -49,10 +48,6 @@ export default class LeaveCmd extends Command {
|
|||||||
}),
|
}),
|
||||||
};
|
};
|
||||||
|
|
||||||
public static flags = {
|
|
||||||
help: cf.help,
|
|
||||||
};
|
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
public static primary = true;
|
public static primary = true;
|
||||||
|
|
||||||
|
@ -16,8 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Args, Command } from '@oclif/core';
|
import { Args, Command } from '@oclif/core';
|
||||||
import { promisify } from 'util';
|
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { stripIndent } from '../../utils/lazy';
|
import { stripIndent } from '../../utils/lazy';
|
||||||
|
|
||||||
export default class LocalConfigureCmd extends Command {
|
export default class LocalConfigureCmd extends Command {
|
||||||
@ -39,10 +37,6 @@ export default class LocalConfigureCmd extends Command {
|
|||||||
}),
|
}),
|
||||||
};
|
};
|
||||||
|
|
||||||
public static flags = {
|
|
||||||
help: cf.help,
|
|
||||||
};
|
|
||||||
|
|
||||||
public static root = true;
|
public static root = true;
|
||||||
public static offlineCompatible = true;
|
public static offlineCompatible = true;
|
||||||
|
|
||||||
@ -242,7 +236,7 @@ export default class LocalConfigureCmd extends Command {
|
|||||||
const bootPartition = await getBootPartition(target);
|
const bootPartition = await getBootPartition(target);
|
||||||
|
|
||||||
const files = await imagefs.interact(target, bootPartition, async (_fs) => {
|
const files = await imagefs.interact(target, bootPartition, async (_fs) => {
|
||||||
return await promisify(_fs.readdir)(this.CONNECTIONS_FOLDER);
|
return await _fs.promises.readdir(this.CONNECTIONS_FOLDER);
|
||||||
});
|
});
|
||||||
|
|
||||||
let connectionFileName;
|
let connectionFileName;
|
||||||
@ -251,13 +245,11 @@ export default class LocalConfigureCmd extends Command {
|
|||||||
} else if (_.includes(files, 'resin-sample.ignore')) {
|
} else if (_.includes(files, 'resin-sample.ignore')) {
|
||||||
// Fresh image, new mode, accoding to https://github.com/balena-os/meta-balena/pull/770/files
|
// Fresh image, new mode, accoding to https://github.com/balena-os/meta-balena/pull/770/files
|
||||||
await imagefs.interact(target, bootPartition, async (_fs) => {
|
await imagefs.interact(target, bootPartition, async (_fs) => {
|
||||||
const readFileAsync = promisify(_fs.readFile);
|
const contents = await _fs.promises.readFile(
|
||||||
const writeFileAsync = promisify(_fs.writeFile);
|
|
||||||
const contents = await readFileAsync(
|
|
||||||
`${this.CONNECTIONS_FOLDER}/resin-sample.ignore`,
|
`${this.CONNECTIONS_FOLDER}/resin-sample.ignore`,
|
||||||
{ encoding: 'utf8' },
|
{ encoding: 'utf8' },
|
||||||
);
|
);
|
||||||
return await writeFileAsync(
|
await _fs.promises.writeFile(
|
||||||
`${this.CONNECTIONS_FOLDER}/resin-wifi`,
|
`${this.CONNECTIONS_FOLDER}/resin-wifi`,
|
||||||
contents,
|
contents,
|
||||||
);
|
);
|
||||||
@ -274,13 +266,13 @@ export default class LocalConfigureCmd extends Command {
|
|||||||
} else {
|
} else {
|
||||||
// In case there's no file at all (shouldn't happen normally, but the file might have been removed)
|
// In case there's no file at all (shouldn't happen normally, but the file might have been removed)
|
||||||
await imagefs.interact(target, bootPartition, async (_fs) => {
|
await imagefs.interact(target, bootPartition, async (_fs) => {
|
||||||
return await promisify(_fs.writeFile)(
|
await _fs.promises.writeFile(
|
||||||
`${this.CONNECTIONS_FOLDER}/resin-wifi`,
|
`${this.CONNECTIONS_FOLDER}/resin-wifi`,
|
||||||
this.CONNECTION_FILE,
|
this.CONNECTION_FILE,
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
return await this.getConfigurationSchema(bootPartition, connectionFileName);
|
return this.getConfigurationSchema(bootPartition, connectionFileName);
|
||||||
}
|
}
|
||||||
|
|
||||||
async removeHostname(schema: any) {
|
async removeHostname(schema: any) {
|
||||||
|
@ -48,7 +48,6 @@ export default class LocalFlashCmd extends Command {
|
|||||||
public static flags = {
|
public static flags = {
|
||||||
drive: cf.drive,
|
drive: cf.drive,
|
||||||
yes: cf.yes,
|
yes: cf.yes,
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static offlineCompatible = true;
|
public static offlineCompatible = true;
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Flags, Args, Command } from '@oclif/core';
|
import { Flags, Args, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { getBalenaSdk, stripIndent, getCliForm } from '../../utils/lazy';
|
import { getBalenaSdk, stripIndent, getCliForm } from '../../utils/lazy';
|
||||||
import { ExpectedError } from '../../errors';
|
import { ExpectedError } from '../../errors';
|
||||||
import type { WhoamiResult } from 'balena-sdk';
|
import type { WhoamiResult } from 'balena-sdk';
|
||||||
@ -29,7 +28,6 @@ interface FlagsDef {
|
|||||||
user?: string;
|
user?: string;
|
||||||
password?: string;
|
password?: string;
|
||||||
port?: number;
|
port?: number;
|
||||||
help: void;
|
|
||||||
hideExperimentalWarning: boolean;
|
hideExperimentalWarning: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -111,7 +109,6 @@ export default class LoginCmd extends Command {
|
|||||||
default: false,
|
default: false,
|
||||||
description: 'Hides warning for experimental features',
|
description: 'Hides warning for experimental features',
|
||||||
}),
|
}),
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static primary = true;
|
public static primary = true;
|
||||||
@ -135,7 +132,7 @@ export default class LoginCmd extends Command {
|
|||||||
// We can safely assume this won't be undefined as doLogin will throw if this call fails
|
// We can safely assume this won't be undefined as doLogin will throw if this call fails
|
||||||
// We also don't need to worry too much about the amount of calls to whoami
|
// We also don't need to worry too much about the amount of calls to whoami
|
||||||
// as these are cached by the SDK
|
// as these are cached by the SDK
|
||||||
const whoamiResult = (await balena.auth.whoami()) as WhoamiResult;
|
const whoamiResult = (await balena.auth.whoami())!;
|
||||||
|
|
||||||
if (whoamiResult.actorType !== 'user' && !options.hideExperimentalWarning) {
|
if (whoamiResult.actorType !== 'user' && !options.hideExperimentalWarning) {
|
||||||
console.info(stripIndent`
|
console.info(stripIndent`
|
||||||
@ -171,7 +168,7 @@ ${messages.reachingOut}`);
|
|||||||
|
|
||||||
async doLogin(
|
async doLogin(
|
||||||
loginOptions: FlagsDef,
|
loginOptions: FlagsDef,
|
||||||
balenaUrl: string = 'balena-cloud.com',
|
balenaUrl = 'balena-cloud.com',
|
||||||
token?: string,
|
token?: string,
|
||||||
): Promise<void> {
|
): Promise<void> {
|
||||||
// Token
|
// Token
|
||||||
|
@ -16,25 +16,23 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Command } from '@oclif/core';
|
import { Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { getBalenaSdk, getVisuals, stripIndent } from '../../utils/lazy';
|
import { getBalenaSdk, getVisuals, stripIndent } from '../../utils/lazy';
|
||||||
|
|
||||||
export default class OrgsCmd extends Command {
|
export default class OrganizationListCmd extends Command {
|
||||||
|
public static aliases = ['orgs'];
|
||||||
|
public static deprecateAliases = true;
|
||||||
|
|
||||||
public static description = stripIndent`
|
public static description = stripIndent`
|
||||||
List all organizations.
|
List all organizations.
|
||||||
|
|
||||||
list all the organizations that you are a member of.
|
list all the organizations that you are a member of.
|
||||||
`;
|
`;
|
||||||
public static examples = ['$ balena orgs'];
|
public static examples = ['$ balena organization list'];
|
||||||
|
|
||||||
public static flags = {
|
|
||||||
help: cf.help,
|
|
||||||
};
|
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
|
||||||
public async run() {
|
public async run() {
|
||||||
await this.parse(OrgsCmd);
|
await this.parse(OrganizationListCmd);
|
||||||
|
|
||||||
const { getOwnOrganizations } = await import('../../utils/sdk');
|
const { getOwnOrganizations } = await import('../../utils/sdk');
|
||||||
|
|
@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Flags, Args, Command } from '@oclif/core';
|
import { Flags, Args, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { getCliForm, stripIndent } from '../../utils/lazy';
|
import { getCliForm, stripIndent } from '../../utils/lazy';
|
||||||
import * as _ from 'lodash';
|
import * as _ from 'lodash';
|
||||||
import type { DeviceTypeJson } from 'balena-sdk';
|
import type { DeviceTypeJson } from 'balena-sdk';
|
||||||
@ -55,7 +54,6 @@ export default class OsBuildConfigCmd extends Command {
|
|||||||
char: 'o',
|
char: 'o',
|
||||||
required: true,
|
required: true,
|
||||||
}),
|
}),
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
@ -18,7 +18,6 @@
|
|||||||
import { Flags, Args, Command } from '@oclif/core';
|
import { Flags, Args, Command } from '@oclif/core';
|
||||||
import type { Interfaces } from '@oclif/core';
|
import type { Interfaces } from '@oclif/core';
|
||||||
import type * as BalenaSdk from 'balena-sdk';
|
import type * as BalenaSdk from 'balena-sdk';
|
||||||
import { promisify } from 'util';
|
|
||||||
import * as _ from 'lodash';
|
import * as _ from 'lodash';
|
||||||
import { ExpectedError } from '../../errors';
|
import { ExpectedError } from '../../errors';
|
||||||
import * as cf from '../../utils/common-flags';
|
import * as cf from '../../utils/common-flags';
|
||||||
@ -154,7 +153,6 @@ export default class OsConfigureCmd extends Command {
|
|||||||
'expiry date assigned to generated provisioning api key (format: YYYY-MM-DD)',
|
'expiry date assigned to generated provisioning api key (format: YYYY-MM-DD)',
|
||||||
exclusive: ['config', 'device'],
|
exclusive: ['config', 'device'],
|
||||||
}),
|
}),
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
@ -293,7 +291,7 @@ export default class OsConfigureCmd extends Command {
|
|||||||
|
|
||||||
for (const { name, content } of files) {
|
for (const { name, content } of files) {
|
||||||
await imagefs.interact(image, bootPartition, async (_fs) => {
|
await imagefs.interact(image, bootPartition, async (_fs) => {
|
||||||
return await promisify(_fs.writeFile)(
|
await _fs.promises.writeFile(
|
||||||
path.join(CONNECTIONS_FOLDER, name),
|
path.join(CONNECTIONS_FOLDER, name),
|
||||||
content,
|
content,
|
||||||
);
|
);
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { Flags, Args, Command } from '@oclif/core';
|
import { Flags, Args, Command } from '@oclif/core';
|
||||||
import * as cf from '../../utils/common-flags';
|
|
||||||
import { stripIndent } from '../../utils/lazy';
|
import { stripIndent } from '../../utils/lazy';
|
||||||
|
|
||||||
export default class OsDownloadCmd extends Command {
|
export default class OsDownloadCmd extends Command {
|
||||||
@ -24,7 +23,7 @@ export default class OsDownloadCmd extends Command {
|
|||||||
Download an unconfigured OS image.
|
Download an unconfigured OS image.
|
||||||
|
|
||||||
Download an unconfigured OS image for the specified device type.
|
Download an unconfigured OS image for the specified device type.
|
||||||
Check available device types with 'balena devices supported'.
|
Check available device types with 'balena device-type list'.
|
||||||
|
|
||||||
Note: Currently this command only works with balenaCloud, not openBalena.
|
Note: Currently this command only works with balenaCloud, not openBalena.
|
||||||
If using openBalena, please download the OS from: https://www.balena.io/os/
|
If using openBalena, please download the OS from: https://www.balena.io/os/
|
||||||
@ -78,7 +77,6 @@ export default class OsDownloadCmd extends Command {
|
|||||||
or 'menu-esr' (interactive menu, ESR versions)
|
or 'menu-esr' (interactive menu, ESR versions)
|
||||||
`,
|
`,
|
||||||
}),
|
}),
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public async run() {
|
public async run() {
|
||||||
|
@ -51,7 +51,6 @@ export default class OsInitializeCmd extends Command {
|
|||||||
type: cf.deviceType,
|
type: cf.deviceType,
|
||||||
drive: cf.drive,
|
drive: cf.drive,
|
||||||
yes: cf.yes,
|
yes: cf.yes,
|
||||||
help: cf.help,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public static authenticated = true;
|
public static authenticated = true;
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user