mirror of
https://github.com/harvester/harvester-ui-extension.git
synced 2026-03-21 20:51:45 +00:00
Compare commits
81 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
97e93dba0b | ||
|
|
9a8a709e56 | ||
|
|
d1949641a7 | ||
|
|
9c9f59c939 | ||
|
|
ccc14c7fb9 | ||
|
|
2ba471907e | ||
|
|
5aea476f64 | ||
|
|
519c7d9f1f | ||
|
|
a9c392c13f | ||
|
|
888ec7a50f | ||
|
|
a2486a7d38 | ||
|
|
df3d249923 | ||
|
|
23344e0c07 | ||
|
|
62b80b3cec | ||
|
|
1cf94ee550 | ||
|
|
dfaa9fbe33 | ||
|
|
94b1c24479 | ||
|
|
e0b2b9ec57 | ||
|
|
06b38a0f99 | ||
|
|
4be3634c82 | ||
|
|
de103ff91e | ||
|
|
37a91601c9 | ||
|
|
9794671ccd | ||
|
|
337f9186f1 | ||
|
|
7c187e894d | ||
|
|
c2df13ad73 | ||
|
|
93f027a57c | ||
|
|
da83f04b6c | ||
|
|
a97cb08e3f | ||
|
|
3fdc9f03a3 | ||
|
|
c8a613874a | ||
|
|
2db7ee7397 | ||
|
|
8b9b5b41b7 | ||
|
|
77599900b5 | ||
|
|
473c1ba355 | ||
|
|
708a95b67b | ||
|
|
fecb3de0cf | ||
|
|
0781bde188 | ||
|
|
0647600e88 | ||
|
|
99dbba7958 | ||
|
|
3dcc50980b | ||
|
|
ee1c3de188 | ||
|
|
915559962a | ||
|
|
b1b1a31c04 | ||
|
|
7f52562d22 | ||
|
|
b140c05697 | ||
|
|
ad9fef63c0 | ||
|
|
786e271ac6 | ||
|
|
c169853e5a | ||
|
|
1352246e1e | ||
|
|
49374bb18a | ||
|
|
fe3a12e28c | ||
|
|
a86302c9d5 | ||
|
|
5fe7e13fcd | ||
|
|
c079984047 | ||
|
|
5769588633 | ||
|
|
b29950f99c | ||
|
|
6c27a46274 | ||
|
|
b03fffbc30 | ||
|
|
5b668a176c | ||
|
|
b4019a2c86 | ||
|
|
416098ffd8 | ||
|
|
3d7b96d86d | ||
|
|
0b37467f76 | ||
|
|
fab7fbec5e | ||
|
|
d94003f8c2 | ||
|
|
dbb5b01cc3 | ||
|
|
467933bda0 | ||
|
|
1b183febdc | ||
|
|
70d3b656f7 | ||
|
|
10d19cd329 | ||
|
|
87e44cb658 | ||
|
|
1715ae754c | ||
|
|
30de2b1a18 | ||
|
|
6fedcc353c | ||
|
|
f9bff21e84 | ||
|
|
6735826e15 | ||
|
|
9e17e239cf | ||
|
|
a1cf41bda9 | ||
|
|
db58024351 | ||
|
|
81bf19419c |
3
.github/renovate.json
vendored
3
.github/renovate.json
vendored
@ -7,7 +7,8 @@
|
|||||||
],
|
],
|
||||||
"baseBranches": [
|
"baseBranches": [
|
||||||
"main",
|
"main",
|
||||||
"/^release-harvester-v\\d+\\.\\d+$/"
|
"release-harvester-v1.7",
|
||||||
|
"release-harvester-v1.8"
|
||||||
],
|
],
|
||||||
"automergeMajor": false,
|
"automergeMajor": false,
|
||||||
"semanticCommits": "enabled",
|
"semanticCommits": "enabled",
|
||||||
|
|||||||
34
.github/workflows/fossa.yml
vendored
Normal file
34
.github/workflows/fossa.yml
vendored
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
name: FOSSA Scanning
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches: ["main", "release-harvester-v*"]
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
id-token: write
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
fossa-scanning:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
timeout-minutes: 30
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6
|
||||||
|
|
||||||
|
# The FOSSA token is shared between all repos in Harvester's GH org. It can
|
||||||
|
# be used directly and there is no need to request specific access to EIO.
|
||||||
|
- name: Read FOSSA token
|
||||||
|
uses: rancher-eio/read-vault-secrets@main
|
||||||
|
with:
|
||||||
|
secrets: |
|
||||||
|
secret/data/github/org/harvester/fossa/credentials token | FOSSA_API_KEY_PUSH_ONLY
|
||||||
|
|
||||||
|
- name: FOSSA scan
|
||||||
|
uses: fossas/fossa-action@main
|
||||||
|
with:
|
||||||
|
api-key: ${{ env.FOSSA_API_KEY_PUSH_ONLY }}
|
||||||
|
# Only runs the scan and do not provide/returns any results back to the
|
||||||
|
# pipeline.
|
||||||
|
run-tests: false
|
||||||
21
.github/workflows/run-lint.yaml
vendored
21
.github/workflows/run-lint.yaml
vendored
@ -1,7 +1,7 @@
|
|||||||
name: Tests
|
name: Tests
|
||||||
|
|
||||||
on:
|
on:
|
||||||
workflow_call: # This tells GH that the workflow is reusable
|
workflow_call: # This tells GH that the workflow is reusable
|
||||||
push:
|
push:
|
||||||
branches:
|
branches:
|
||||||
- main
|
- main
|
||||||
@ -41,8 +41,14 @@ jobs:
|
|||||||
FROM="$GITHUB_BASE_SHA"
|
FROM="$GITHUB_BASE_SHA"
|
||||||
TO="$GITHUB_HEAD_SHA"
|
TO="$GITHUB_HEAD_SHA"
|
||||||
elif [ -n "$GITHUB_BEFORE" ] && [ -n "$GITHUB_AFTER" ]; then
|
elif [ -n "$GITHUB_BEFORE" ] && [ -n "$GITHUB_AFTER" ]; then
|
||||||
FROM="$GITHUB_BEFORE"
|
if [ "$GITHUB_BEFORE" = "0000000000000000000000000000000000000000" ]; then
|
||||||
TO="$GITHUB_AFTER"
|
# first push to HEAD
|
||||||
|
FROM=""
|
||||||
|
TO="$GITHUB_AFTER"
|
||||||
|
else
|
||||||
|
FROM="$GITHUB_BEFORE"
|
||||||
|
TO="$GITHUB_AFTER"
|
||||||
|
fi
|
||||||
else
|
else
|
||||||
echo "No valid commit range found, skipping commitlint."
|
echo "No valid commit range found, skipping commitlint."
|
||||||
exit 0
|
exit 0
|
||||||
@ -51,7 +57,14 @@ jobs:
|
|||||||
echo "FROM=$FROM"
|
echo "FROM=$FROM"
|
||||||
echo "TO=$TO"
|
echo "TO=$TO"
|
||||||
|
|
||||||
npx commitlint --from "$FROM" --to "$TO" --verbose
|
if [ -z "$FROM" ]; then
|
||||||
|
echo "Linting last commit $TO"
|
||||||
|
npx commitlint --last --verbose
|
||||||
|
|
||||||
|
else
|
||||||
|
echo "Linting commits from $FROM to $TO"
|
||||||
|
npx commitlint --from "$FROM" --to "$TO" --verbose
|
||||||
|
fi
|
||||||
env:
|
env:
|
||||||
GITHUB_EVENT_NAME: ${{ github.event_name }}
|
GITHUB_EVENT_NAME: ${{ github.event_name }}
|
||||||
GITHUB_BASE_SHA: ${{ github.event.pull_request.base.sha }}
|
GITHUB_BASE_SHA: ${{ github.event.pull_request.base.sha }}
|
||||||
|
|||||||
@ -7,7 +7,7 @@ The Harvester UI Extension is a Rancher extension that provides the user interfa
|
|||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
For detailed installation instructions, please refer to the [official Harvester documentation](https://docs.harvesterhci.io/v1.5/rancher/harvester-ui-extension#installation-on-rancher-210).
|
For Harvester UI extension installation instructions, please refer to the page **Rancher Integration** -> **Harvester UI Extension** in [official Harvester documentation](https://docs.harvesterhci.io).
|
||||||
|
|
||||||
|
|
||||||
## Development Setup
|
## Development Setup
|
||||||
@ -163,7 +163,7 @@ If you want to contribute, start by reading this document, then visit our [Getti
|
|||||||
|
|
||||||
## License
|
## License
|
||||||
|
|
||||||
Copyright (c) 2014-2025 [SUSE, LLC.](https://www.suse.com/)
|
Copyright (c) 2014-2026 [SUSE, LLC.](https://www.suse.com/)
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
you may not use this file except in compliance with the License.
|
you may not use this file except in compliance with the License.
|
||||||
|
|||||||
22
package.json
22
package.json
@ -1,17 +1,22 @@
|
|||||||
{
|
{
|
||||||
"name": "harvester-ui-extension",
|
"name": "harvester-ui-extension",
|
||||||
"version": "1.7.0-dev",
|
"version": "1.8.0-rc2",
|
||||||
"private": false,
|
"private": false,
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=20.0.0"
|
"node": ">=20.0.0"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@babel/plugin-transform-class-static-block": "7.28.3",
|
"@babel/plugin-transform-class-static-block": "7.28.6",
|
||||||
"@rancher/shell": "3.0.5-rc.8",
|
"@rancher/shell": "3.0.9-rc.6",
|
||||||
|
"@vue-flow/background": "^1.3.0",
|
||||||
|
"@vue-flow/controls": "^1.1.1",
|
||||||
|
"@vue-flow/core": "^1.33.5",
|
||||||
|
"@vue-flow/minimap": "^1.4.0",
|
||||||
"cache-loader": "^4.1.0",
|
"cache-loader": "^4.1.0",
|
||||||
"color": "4.2.3",
|
"color": "4.2.3",
|
||||||
"ip": "2.0.1",
|
"ip": "2.0.1",
|
||||||
"node-polyfill-webpack-plugin": "^3.0.0",
|
"node-polyfill-webpack-plugin": "^3.0.0",
|
||||||
|
"elkjs": "^0.11.0",
|
||||||
"vue-draggable-next": "^2.2.1",
|
"vue-draggable-next": "^2.2.1",
|
||||||
"yaml": "^2.5.1"
|
"yaml": "^2.5.1"
|
||||||
},
|
},
|
||||||
@ -24,13 +29,13 @@
|
|||||||
"glob": "7.2.3",
|
"glob": "7.2.3",
|
||||||
"glob-parent": "6.0.2",
|
"glob-parent": "6.0.2",
|
||||||
"json5": "2.2.3",
|
"json5": "2.2.3",
|
||||||
"@types/lodash": "4.17.20",
|
"@types/lodash": "4.17.24",
|
||||||
"merge": "2.1.1",
|
"merge": "2.1.1",
|
||||||
"node-forge": "1.3.1",
|
"node-forge": "1.3.3",
|
||||||
"nth-check": "2.1.1",
|
"nth-check": "2.1.1",
|
||||||
"qs": "6.14.0",
|
"qs": "6.15.0",
|
||||||
"roarr": "7.21.2",
|
"roarr": "7.21.4",
|
||||||
"semver": "7.7.3",
|
"semver": "7.7.4",
|
||||||
"@vue/cli-service/html-webpack-plugin": "^5.0.0"
|
"@vue/cli-service/html-webpack-plugin": "^5.0.0"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
@ -38,6 +43,7 @@
|
|||||||
"build": "./node_modules/.bin/vue-cli-service build",
|
"build": "./node_modules/.bin/vue-cli-service build",
|
||||||
"clean": "./node_modules/@rancher/shell/scripts/clean",
|
"clean": "./node_modules/@rancher/shell/scripts/clean",
|
||||||
"lint": "./node_modules/.bin/eslint --max-warnings 0 --ext .js,.ts,.vue .",
|
"lint": "./node_modules/.bin/eslint --max-warnings 0 --ext .js,.ts,.vue .",
|
||||||
|
"lint:fix": "./node_modules/.bin/eslint --fix --max-warnings 0 --ext .js,.ts,.vue .",
|
||||||
"build-pkg": "./node_modules/@rancher/shell/scripts/build-pkg.sh",
|
"build-pkg": "./node_modules/@rancher/shell/scripts/build-pkg.sh",
|
||||||
"serve-pkgs": "./node_modules/@rancher/shell/scripts/serve-pkgs",
|
"serve-pkgs": "./node_modules/@rancher/shell/scripts/serve-pkgs",
|
||||||
"publish-pkgs": "./node_modules/@rancher/shell/scripts/extension/publish",
|
"publish-pkgs": "./node_modules/@rancher/shell/scripts/extension/publish",
|
||||||
|
|||||||
177
pkg/harvester/README.md
Normal file
177
pkg/harvester/README.md
Normal file
@ -0,0 +1,177 @@
|
|||||||
|
# harvester-ui-extension
|
||||||
|
|
||||||
|
The Harvester UI Extension is a Rancher extension that provides the user interface for [Harvester](https://harvesterhci.io) within the [Rancher Dashboard](https://github.com/rancher/dashboard).
|
||||||
|
|
||||||
|
> **Note:**
|
||||||
|
> This extension is available starting from **Rancher 2.10.0**. Ensure your Rancher version is **2.10.0 or later** to access Harvester integration.
|
||||||
|
|
||||||
|
## Installation
|
||||||
|
|
||||||
|
For Harvester UI extension installation instructions, please refer to the page **Rancher Integration** -> **Harvester UI Extension** in [official Harvester documentation](https://docs.harvesterhci.io).
|
||||||
|
|
||||||
|
## Development Setup
|
||||||
|
|
||||||
|
Ensure **Node.js v20 or later** is installed for development and debugging.
|
||||||
|
|
||||||
|
### Standalone Mode
|
||||||
|
|
||||||
|
Run the extension standalone with hot reload at `https://localhost:8005`.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Install dependencies
|
||||||
|
yarn install
|
||||||
|
|
||||||
|
# Start the development server
|
||||||
|
RANCHER_ENV=harvester API=https://your-harvester-ip yarn dev
|
||||||
|
|
||||||
|
# Example with specific server version
|
||||||
|
RANCHER_ENV=harvester VUE_APP_SERVER_VERSION=v1.5.0 API=https://192.168.1.123 yarn dev
|
||||||
|
```
|
||||||
|
|
||||||
|
You may also define environment variables in a `.env` file:
|
||||||
|
|
||||||
|
```env
|
||||||
|
RANCHER_ENV=harvester
|
||||||
|
VUE_APP_SERVER_VERSION=v1.5.0
|
||||||
|
API=https://192.168.1.123
|
||||||
|
```
|
||||||
|
|
||||||
|
### Rancher Integration Mode
|
||||||
|
|
||||||
|
To run as a Rancher extension, follow the [Rancher UI Extension Guide](https://extensions.rancher.io/extensions/next/extensions-getting-started#running-the-app).
|
||||||
|
|
||||||
|
```bash
|
||||||
|
API=https://your-rancher-ip yarn dev
|
||||||
|
```
|
||||||
|
|
||||||
|
## Commit Message Guidelines
|
||||||
|
|
||||||
|
This project uses [commit-lint](https://commitlint.js.org/) with [Conventional Commits](https://www.conventionalcommits.org/) to ensure consistent and meaningful commit messages.
|
||||||
|
|
||||||
|
### Commit Message Format
|
||||||
|
|
||||||
|
All commit messages must follow the conventional commit format:
|
||||||
|
|
||||||
|
```
|
||||||
|
<type>[optional scope]: <description>
|
||||||
|
|
||||||
|
[optional body]
|
||||||
|
|
||||||
|
[optional footer(s)]
|
||||||
|
```
|
||||||
|
|
||||||
|
### Supported Types
|
||||||
|
|
||||||
|
- **feat**: New features
|
||||||
|
- **fix**: Bug fixes
|
||||||
|
- **docs**: Documentation changes
|
||||||
|
- **style**: Code style changes (formatting, missing semicolons, etc.)
|
||||||
|
- **refactor**: Code refactoring
|
||||||
|
- **perf**: Performance improvements
|
||||||
|
- **test**: Adding or updating tests
|
||||||
|
- **build**: Build system or external dependencies
|
||||||
|
- **ci**: CI/CD changes
|
||||||
|
- **chore**: Other changes that don't modify src or test files
|
||||||
|
- **revert**: Reverts a previous commit
|
||||||
|
- **wip**: Work in progress
|
||||||
|
- **deps**: Dependency updates
|
||||||
|
- **security**: Security fixes
|
||||||
|
|
||||||
|
### Examples
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Feature
|
||||||
|
git commit -m "feat: add new virtual machine creation wizard"
|
||||||
|
|
||||||
|
# Bug fix
|
||||||
|
git commit -m "fix: resolve memory leak in VM console"
|
||||||
|
|
||||||
|
# Documentation
|
||||||
|
git commit -m "docs: update installation instructions"
|
||||||
|
|
||||||
|
# Breaking change
|
||||||
|
git commit -m "feat!: change API endpoint structure
|
||||||
|
|
||||||
|
BREAKING CHANGE: The /api/v1/vms endpoint has been replaced with /api/v2/vms"
|
||||||
|
```
|
||||||
|
|
||||||
|
### Git Hooks
|
||||||
|
|
||||||
|
The project uses [Husky](https://typicode.github.io/husky/) to automatically validate commit messages and run linting before commits:
|
||||||
|
|
||||||
|
- **pre-commit**: Runs ESLint to ensure code quality
|
||||||
|
- **commit-msg**: Validates commit message format using commit-lint
|
||||||
|
|
||||||
|
These hooks are automatically installed when you run `yarn install`.
|
||||||
|
|
||||||
|
### Manual Validation
|
||||||
|
|
||||||
|
You can manually validate commit messages:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Validate the last commit
|
||||||
|
yarn commitlint
|
||||||
|
|
||||||
|
# Validate a specific commit
|
||||||
|
npx commitlint --from <commit-hash>
|
||||||
|
|
||||||
|
# Validate a range of commits
|
||||||
|
npx commitlint --from <start-hash> --to <end-hash>
|
||||||
|
```
|
||||||
|
|
||||||
|
## Branch Structure
|
||||||
|
|
||||||
|
- **`main`** – Main development branch
|
||||||
|
- **`release-harvester-vX.Y`** – Stable release branches per version series
|
||||||
|
- **`vX.Y-head`** – Testing branches for ongoing changes to extension builds in each release series
|
||||||
|
|
||||||
|
> **Note:**
|
||||||
|
> The `vX.Y-head` branches are auto-generated and kept in sync with release branches. Use these for testing the latest changes in each version series.
|
||||||
|
|
||||||
|
## Testing Guidelines
|
||||||
|
|
||||||
|
### UI Extension Testing
|
||||||
|
|
||||||
|
To validate changes in a release series, switch to the appropriate `vX.Y-head` branch. For main branch testing, use `main-head`.
|
||||||
|
|
||||||
|
- Examples:
|
||||||
|
- Test `1.0.x` series → `v1.0-head`
|
||||||
|
- Test `1.5.x` series → `v1.5-head`
|
||||||
|
|
||||||
|
**Steps:**
|
||||||
|
1. Navigate to **Rancher UI** → **Local** → **App** → **Repositories**
|
||||||
|
2. Refresh the Harvester repository using the target `vX.Y-head` branch
|
||||||
|
3. Go to the **Extensions** page and install the desired version
|
||||||
|
|
||||||
|
### Standalone Mode Testing
|
||||||
|
|
||||||
|
To test the standalone UI, configure Harvester to load the UI from an external source.
|
||||||
|
|
||||||
|
- Examples of `ui-index`:
|
||||||
|
- Main branch → `https://releases.rancher.com/harvester-ui/dashboard/latest/index.html`
|
||||||
|
- Release series `1.5.x` → `https://releases.rancher.com/harvester-ui/dashboard/release-harvester-v1.5/index.html`
|
||||||
|
|
||||||
|
**Steps:**
|
||||||
|
1. Go to **Harvester UI** → **Advanced** → **Settings** → **UI**
|
||||||
|
2. Set **ui-source** to `External`
|
||||||
|
3. Set **ui-index** to the desired URL
|
||||||
|
|
||||||
|
## Contributing
|
||||||
|
|
||||||
|
If you want to contribute, start by reading this document, then visit our [Getting Started guide](https://extensions.rancher.io/extensions/next/extensions-getting-started) to learn how to develop and submit changes.
|
||||||
|
|
||||||
|
## License
|
||||||
|
|
||||||
|
Copyright (c) 2014-2026 [SUSE, LLC.](https://www.suse.com/)
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
[http://www.apache.org/licenses/LICENSE-2.0](http://www.apache.org/licenses/LICENSE-2.0)
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
@ -1,6 +1,8 @@
|
|||||||
<script>
|
<script>
|
||||||
import Collapse from '@shell/components/Collapse';
|
import Collapse from '@shell/components/Collapse';
|
||||||
import PercentageBar from '@shell/components/PercentageBar';
|
import PercentageBar from '@shell/components/PercentageBar';
|
||||||
|
import { HCI } from '../types';
|
||||||
|
import { HCI as HCI_ANNOTATIONS } from '@pkg/harvester/config/labels-annotations';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: 'HarvesterUpgradeProgressList',
|
name: 'HarvesterUpgradeProgressList',
|
||||||
@ -25,13 +27,45 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
async fetch() {
|
||||||
|
await this.$store.dispatch('harvester/findAll', { type: HCI.UPGRADE });
|
||||||
|
},
|
||||||
|
|
||||||
data() {
|
data() {
|
||||||
return { open: true };
|
return { open: true };
|
||||||
},
|
},
|
||||||
|
|
||||||
|
computed: {
|
||||||
|
showResumeButton() {
|
||||||
|
return this.title === 'Upgrading Node';
|
||||||
|
},
|
||||||
|
latestUpgradeCR() {
|
||||||
|
return this.$store.getters['harvester/all'](HCI.UPGRADE).find( (U) => U.isLatestUpgrade);
|
||||||
|
},
|
||||||
|
resumeUpgradePausedNodeEnabled() {
|
||||||
|
return this.$store.getters['harvester-common/getFeatureEnabled']('resumeUpgradePausedNode');
|
||||||
|
},
|
||||||
|
},
|
||||||
methods: {
|
methods: {
|
||||||
handleSwitch() {
|
handleSwitch() {
|
||||||
this.open = !this.open;
|
this.open = !this.open;
|
||||||
|
},
|
||||||
|
async resumeNodeUpgrade(nodeName) {
|
||||||
|
if (!this.latestUpgradeCR || !nodeName) return;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const upgradePauseMapString = this.latestUpgradeCR.metadata.annotations[HCI_ANNOTATIONS.NODE_UPGRADE_PAUSE_MAP] || '{}';
|
||||||
|
const upgradePauseMap = JSON.parse(upgradePauseMapString);
|
||||||
|
|
||||||
|
// update the upgrade CR annotation harvesterhci.io/node-upgrade-pause-map to unpause the node upgrade process
|
||||||
|
upgradePauseMap[`${ nodeName }`] = 'unpause';
|
||||||
|
this.latestUpgradeCR.setAnnotation(HCI_ANNOTATIONS.NODE_UPGRADE_PAUSE_MAP, JSON.stringify(upgradePauseMap));
|
||||||
|
await this.latestUpgradeCR.save();
|
||||||
|
} catch (e) {
|
||||||
|
console.error(`unable to update harvester upgrade CR annotations: ${ this.latestUpgradeCR.id }.`, e); // eslint-disable-line no-console
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -63,12 +97,28 @@ export default {
|
|||||||
v-for="(item, i) in list"
|
v-for="(item, i) in list"
|
||||||
:key="i"
|
:key="i"
|
||||||
>
|
>
|
||||||
<p>
|
<div class="upgrade-node-header">
|
||||||
{{ item.name }} <span
|
<div class="upgrade-node-title">
|
||||||
class="status"
|
<p>
|
||||||
:class="{ [item.state]: true }"
|
{{ item.name }}
|
||||||
>{{ item.state }}</span>
|
</p>
|
||||||
</p>
|
<span
|
||||||
|
class="status"
|
||||||
|
:class="{ [item.state]: true }"
|
||||||
|
>
|
||||||
|
{{ item.state }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
v-if="showResumeButton && resumeUpgradePausedNodeEnabled && item.state === 'Node-upgrade paused'"
|
||||||
|
type="button"
|
||||||
|
class="btn bg-info btn-sm"
|
||||||
|
data-testid="add-item"
|
||||||
|
@click="resumeNodeUpgrade(item.name)"
|
||||||
|
>
|
||||||
|
{{ t('action.resume') }}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
<PercentageBar
|
<PercentageBar
|
||||||
:model-value="item.percent"
|
:model-value="item.percent"
|
||||||
preferred-direction="MORE"
|
preferred-direction="MORE"
|
||||||
@ -102,10 +152,21 @@ export default {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
.custom-content {
|
.custom-content {
|
||||||
margin-bottom: 14px;
|
.upgrade-node-title {
|
||||||
p {
|
flex: 1 0 80%;
|
||||||
|
margin-right: 10px;
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
}
|
||||||
|
.upgrade-node-header {
|
||||||
|
display:flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
margin-bottom: 4px;
|
margin-bottom: 4px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
margin-bottom: 14px;
|
||||||
|
|
||||||
.status {
|
.status {
|
||||||
float: right;
|
float: right;
|
||||||
}
|
}
|
||||||
@ -117,6 +178,8 @@ export default {
|
|||||||
}
|
}
|
||||||
.warning {
|
.warning {
|
||||||
color: var(--error);
|
color: var(--error);
|
||||||
|
margin-bottom: 8px;
|
||||||
|
margin-top: 4px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
123
pkg/harvester/components/settings/cluster-registration-url.vue
Normal file
123
pkg/harvester/components/settings/cluster-registration-url.vue
Normal file
@ -0,0 +1,123 @@
|
|||||||
|
<script>
|
||||||
|
import MessageLink from '@shell/components/MessageLink';
|
||||||
|
import CreateEditView from '@shell/mixins/create-edit-view';
|
||||||
|
import { LabeledInput } from '@components/Form/LabeledInput';
|
||||||
|
import { HCI_SETTING } from '../../config/settings';
|
||||||
|
import { Checkbox } from '@components/Form/Checkbox';
|
||||||
|
import { Banner } from '@components/Banner';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'HarvesterEditClusterRegistrationURL',
|
||||||
|
|
||||||
|
components: {
|
||||||
|
LabeledInput, MessageLink, Checkbox, Banner
|
||||||
|
},
|
||||||
|
|
||||||
|
mixins: [CreateEditView],
|
||||||
|
|
||||||
|
data() {
|
||||||
|
let parseDefaultValue = {};
|
||||||
|
|
||||||
|
try {
|
||||||
|
parseDefaultValue = JSON.parse(this.value.value);
|
||||||
|
} catch (error) {
|
||||||
|
parseDefaultValue.url = this.value.value;
|
||||||
|
parseDefaultValue.insecureSkipTLSVerify = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
parseDefaultValue,
|
||||||
|
errors: []
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
computed: {
|
||||||
|
toCA() {
|
||||||
|
return `${ HCI_SETTING.ADDITIONAL_CA }?mode=edit`;
|
||||||
|
},
|
||||||
|
clusterRegistrationTLSVerifyEnabled() {
|
||||||
|
return this.$store.getters['harvester-common/getFeatureEnabled']('clusterRegistrationTLSVerify');
|
||||||
|
},
|
||||||
|
registrationURL: {
|
||||||
|
get() {
|
||||||
|
return this.clusterRegistrationTLSVerifyEnabled ? this.parseDefaultValue.url : this.parseDefaultValue;
|
||||||
|
},
|
||||||
|
|
||||||
|
set(value) {
|
||||||
|
if (this.clusterRegistrationTLSVerifyEnabled) {
|
||||||
|
this.parseDefaultValue.url = value;
|
||||||
|
} else {
|
||||||
|
this.parseDefaultValue = value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
methods: {
|
||||||
|
getDefaultValue() {
|
||||||
|
if (this.clusterRegistrationTLSVerifyEnabled) {
|
||||||
|
return { url: '', insecureSkipTLSVerify: false };
|
||||||
|
} else {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
updateUrl() {
|
||||||
|
this.update();
|
||||||
|
},
|
||||||
|
|
||||||
|
update() {
|
||||||
|
if (this.clusterRegistrationTLSVerifyEnabled) {
|
||||||
|
this.value['value'] = JSON.stringify(this.parseDefaultValue);
|
||||||
|
} else {
|
||||||
|
this.value['value'] = this.parseDefaultValue;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
useDefault() {
|
||||||
|
this.parseDefaultValue = this.getDefaultValue();
|
||||||
|
},
|
||||||
|
|
||||||
|
updateInsecureSkipTLSVerify(newValue) {
|
||||||
|
const { url = '' } = this.parseDefaultValue;
|
||||||
|
|
||||||
|
this.parseDefaultValue = { url, insecureSkipTLSVerify: newValue };
|
||||||
|
this.update();
|
||||||
|
},
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div
|
||||||
|
class="row"
|
||||||
|
>
|
||||||
|
<div class="col span-12">
|
||||||
|
<Banner color="info">
|
||||||
|
<MessageLink
|
||||||
|
:to="toCA"
|
||||||
|
target="_blank"
|
||||||
|
prefix-label="harvester.setting.clusterRegistrationUrl.tip.prefix"
|
||||||
|
middle-label="harvester.setting.clusterRegistrationUrl.tip.middle"
|
||||||
|
suffix-label="harvester.setting.clusterRegistrationUrl.tip.suffix"
|
||||||
|
/>
|
||||||
|
</Banner>
|
||||||
|
<LabeledInput
|
||||||
|
v-model:value="registrationURL"
|
||||||
|
class="mb-20"
|
||||||
|
:mode="mode"
|
||||||
|
:label="t('harvester.setting.clusterRegistrationUrl.url')"
|
||||||
|
@update:value="updateUrl"
|
||||||
|
/>
|
||||||
|
<div v-if="clusterRegistrationTLSVerifyEnabled">
|
||||||
|
<Checkbox
|
||||||
|
v-model:value="parseDefaultValue.insecureSkipTLSVerify"
|
||||||
|
class="check mb-5"
|
||||||
|
type="checkbox"
|
||||||
|
:label="t('harvester.setting.clusterRegistrationUrl.insecureSkipTLSVerify')"
|
||||||
|
@update:value="updateInsecureSkipTLSVerify"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
104
pkg/harvester/components/settings/instance-manager-resources.vue
Normal file
104
pkg/harvester/components/settings/instance-manager-resources.vue
Normal file
@ -0,0 +1,104 @@
|
|||||||
|
<script>
|
||||||
|
import UnitInput from '@shell/components/form/UnitInput';
|
||||||
|
import { Banner } from '@components/Banner';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'HarvesterInstanceManagerResources',
|
||||||
|
|
||||||
|
components: {
|
||||||
|
UnitInput,
|
||||||
|
Banner,
|
||||||
|
},
|
||||||
|
|
||||||
|
props: {
|
||||||
|
value: {
|
||||||
|
type: Object,
|
||||||
|
default: () => ({
|
||||||
|
value: '',
|
||||||
|
default: '{}'
|
||||||
|
})
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
data() {
|
||||||
|
const resources = this.parseJSON(this.value?.value) || this.parseJSON(this.value?.default) || {};
|
||||||
|
|
||||||
|
return {
|
||||||
|
resources,
|
||||||
|
parseError: null,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
methods: {
|
||||||
|
parseJSON(string) {
|
||||||
|
try {
|
||||||
|
return JSON.parse(string);
|
||||||
|
} catch (e) {
|
||||||
|
this.parseError = this.t('harvester.setting.instanceManagerResources.parseError', { error: e.message });
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
update() {
|
||||||
|
if (!this.value) return;
|
||||||
|
|
||||||
|
const cpu = { ...this.resources?.cpu };
|
||||||
|
|
||||||
|
if (cpu.v1 !== null) cpu.v1 = String(cpu.v1);
|
||||||
|
if (cpu.v2 !== null) cpu.v2 = String(cpu.v2);
|
||||||
|
|
||||||
|
this.value.value = JSON.stringify({ ...this.resources, cpu });
|
||||||
|
},
|
||||||
|
|
||||||
|
useDefault() {
|
||||||
|
if (this.value?.default) {
|
||||||
|
this.resources = this.parseJSON(this.value.default) || {};
|
||||||
|
this.update();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<Banner
|
||||||
|
v-if="parseError"
|
||||||
|
color="error"
|
||||||
|
>
|
||||||
|
{{ parseError }}
|
||||||
|
</Banner>
|
||||||
|
|
||||||
|
<div class="row">
|
||||||
|
<div class="col span-12">
|
||||||
|
<UnitInput
|
||||||
|
v-model:value="resources.cpu.v1"
|
||||||
|
:label="t('harvester.setting.instanceManagerResources.v1')"
|
||||||
|
suffix="%"
|
||||||
|
:delay="0"
|
||||||
|
type="number"
|
||||||
|
min="0"
|
||||||
|
max="100"
|
||||||
|
required
|
||||||
|
:mode="mode"
|
||||||
|
class="mb-20"
|
||||||
|
@update:value="update"
|
||||||
|
/>
|
||||||
|
<UnitInput
|
||||||
|
v-model:value="resources.cpu.v2"
|
||||||
|
:label="t('harvester.setting.instanceManagerResources.v2')"
|
||||||
|
suffix="%"
|
||||||
|
:delay="0"
|
||||||
|
type="number"
|
||||||
|
min="0"
|
||||||
|
max="100"
|
||||||
|
required
|
||||||
|
:mode="mode"
|
||||||
|
class="mb-20"
|
||||||
|
@update:value="update"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
371
pkg/harvester/components/settings/rwx-network.vue
Normal file
371
pkg/harvester/components/settings/rwx-network.vue
Normal file
@ -0,0 +1,371 @@
|
|||||||
|
<script>
|
||||||
|
import { LabeledInput } from '@components/Form/LabeledInput';
|
||||||
|
import LabeledSelect from '@shell/components/form/LabeledSelect.vue';
|
||||||
|
import { RadioGroup } from '@components/Form/Radio';
|
||||||
|
import ArrayList from '@shell/components/form/ArrayList';
|
||||||
|
import { isValidCIDR } from '@shell/utils/validators/cidr';
|
||||||
|
import { _EDIT } from '@shell/config/query-params';
|
||||||
|
import { Banner } from '@components/Banner';
|
||||||
|
import { allHash } from '@shell/utils/promise';
|
||||||
|
import { HCI } from '../../types';
|
||||||
|
import { NETWORK_TYPE } from '../../config/types';
|
||||||
|
|
||||||
|
const { L2VLAN, UNTAGGED } = NETWORK_TYPE;
|
||||||
|
const SHARE_STORAGE_NETWORK = 'share-storage-network';
|
||||||
|
const NETWORK = 'network';
|
||||||
|
|
||||||
|
const DEFAULT_DEDICATED_NETWORK = {
|
||||||
|
vlan: '',
|
||||||
|
clusterNetwork: '',
|
||||||
|
range: '',
|
||||||
|
exclude: [],
|
||||||
|
};
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'RwxNetworkSetting',
|
||||||
|
|
||||||
|
components: {
|
||||||
|
RadioGroup,
|
||||||
|
Banner,
|
||||||
|
ArrayList,
|
||||||
|
LabeledInput,
|
||||||
|
LabeledSelect,
|
||||||
|
},
|
||||||
|
|
||||||
|
props: {
|
||||||
|
registerBeforeHook: {
|
||||||
|
type: Function,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
|
||||||
|
mode: {
|
||||||
|
type: String,
|
||||||
|
default: _EDIT,
|
||||||
|
},
|
||||||
|
|
||||||
|
value: {
|
||||||
|
type: Object,
|
||||||
|
default: () => {
|
||||||
|
return {};
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
async fetch() {
|
||||||
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
|
|
||||||
|
await allHash({
|
||||||
|
clusterNetworks: this.$store.dispatch(`${ inStore }/findAll`, { type: HCI.CLUSTER_NETWORK }),
|
||||||
|
vlanStatus: this.$store.dispatch(`${ inStore }/findAll`, { type: HCI.VLAN_STATUS }),
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
data() {
|
||||||
|
let enabled = false; // enabled / disabled options
|
||||||
|
let shareStorageNetwork = false; // shareStorageNetwork / dedicatedRwxNetwork options
|
||||||
|
let dedicatedNetwork = { ...DEFAULT_DEDICATED_NETWORK };
|
||||||
|
let networkType = L2VLAN;
|
||||||
|
let exclude = [];
|
||||||
|
|
||||||
|
try {
|
||||||
|
const parsedValue = JSON.parse(this.value.value || this.value.default || '{}');
|
||||||
|
const parsedNetwork = parsedValue?.[NETWORK] || parsedValue || {};
|
||||||
|
|
||||||
|
if (parsedValue && typeof parsedValue === 'object') {
|
||||||
|
shareStorageNetwork = !!parsedValue[SHARE_STORAGE_NETWORK];
|
||||||
|
networkType = 'vlan' in parsedNetwork ? L2VLAN : UNTAGGED;
|
||||||
|
dedicatedNetwork = {
|
||||||
|
vlan: parsedNetwork.vlan || '',
|
||||||
|
clusterNetwork: parsedNetwork.clusterNetwork || '',
|
||||||
|
range: parsedNetwork.range || '',
|
||||||
|
};
|
||||||
|
exclude = parsedNetwork?.exclude?.toString().split(',') || [];
|
||||||
|
enabled = shareStorageNetwork || !!(parsedNetwork.vlan || parsedNetwork.clusterNetwork || parsedNetwork.range);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
enabled = false;
|
||||||
|
shareStorageNetwork = false;
|
||||||
|
dedicatedNetwork = { ...DEFAULT_DEDICATED_NETWORK };
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
enabled,
|
||||||
|
shareStorageNetwork,
|
||||||
|
dedicatedNetwork,
|
||||||
|
networkType,
|
||||||
|
exclude,
|
||||||
|
defaultAddValue: '',
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
created() {
|
||||||
|
if (this.registerBeforeHook) {
|
||||||
|
this.registerBeforeHook(this.willSave, 'willSave');
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
computed: {
|
||||||
|
showDedicatedNetworkConfig() {
|
||||||
|
return this.enabled && !this.shareStorageNetwork;
|
||||||
|
},
|
||||||
|
|
||||||
|
showVlan() {
|
||||||
|
return this.networkType === L2VLAN;
|
||||||
|
},
|
||||||
|
|
||||||
|
networkTypes() {
|
||||||
|
return [L2VLAN, UNTAGGED];
|
||||||
|
},
|
||||||
|
|
||||||
|
clusterNetworkOptions() {
|
||||||
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
|
const clusterNetworks = this.$store.getters[`${ inStore }/all`](HCI.CLUSTER_NETWORK) || [];
|
||||||
|
const clusterNetworksOptions = this.networkType === UNTAGGED ? clusterNetworks.filter((network) => network.id !== 'mgmt') : clusterNetworks;
|
||||||
|
|
||||||
|
return clusterNetworksOptions.map((network) => {
|
||||||
|
const disabled = !network.isReadyForStorageNetwork;
|
||||||
|
|
||||||
|
return {
|
||||||
|
label: disabled ? `${ network.id } (${ this.t('generic.notReady') })` : network.id,
|
||||||
|
value: network.id,
|
||||||
|
disabled,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
methods: {
|
||||||
|
onUpdateEnabled() {
|
||||||
|
if (!this.enabled) {
|
||||||
|
this.shareStorageNetwork = false;
|
||||||
|
this.dedicatedNetwork = { ...DEFAULT_DEDICATED_NETWORK };
|
||||||
|
}
|
||||||
|
|
||||||
|
this.update();
|
||||||
|
},
|
||||||
|
|
||||||
|
onUpdateNetworkType() {
|
||||||
|
if (this.shareStorageNetwork) {
|
||||||
|
this.dedicatedNetwork = { ...DEFAULT_DEDICATED_NETWORK };
|
||||||
|
}
|
||||||
|
|
||||||
|
this.update();
|
||||||
|
},
|
||||||
|
|
||||||
|
onUpdateDedicatedType(neu) {
|
||||||
|
this.dedicatedNetwork.clusterNetwork = '';
|
||||||
|
|
||||||
|
if (neu === L2VLAN) {
|
||||||
|
this.dedicatedNetwork.vlan = '';
|
||||||
|
} else {
|
||||||
|
delete this.dedicatedNetwork.vlan;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.update();
|
||||||
|
},
|
||||||
|
|
||||||
|
inputVlan(neu) {
|
||||||
|
if (neu === '') {
|
||||||
|
this.dedicatedNetwork.vlan = '';
|
||||||
|
this.update();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const newValue = Number(neu);
|
||||||
|
|
||||||
|
if (newValue > 4094) {
|
||||||
|
this.dedicatedNetwork.vlan = 4094;
|
||||||
|
} else if (newValue < 1) {
|
||||||
|
this.dedicatedNetwork.vlan = 1;
|
||||||
|
} else {
|
||||||
|
this.dedicatedNetwork.vlan = newValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.update();
|
||||||
|
},
|
||||||
|
|
||||||
|
useDefault() {
|
||||||
|
this.enabled = false;
|
||||||
|
this.shareStorageNetwork = false;
|
||||||
|
this.dedicatedNetwork = { ...DEFAULT_DEDICATED_NETWORK };
|
||||||
|
this.update();
|
||||||
|
},
|
||||||
|
|
||||||
|
update() {
|
||||||
|
const value = { [SHARE_STORAGE_NETWORK]: false };
|
||||||
|
|
||||||
|
if (this.enabled && this.shareStorageNetwork) {
|
||||||
|
value[SHARE_STORAGE_NETWORK] = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.showDedicatedNetworkConfig) {
|
||||||
|
value[NETWORK] = {};
|
||||||
|
|
||||||
|
if (this.showVlan) {
|
||||||
|
value[NETWORK].vlan = this.dedicatedNetwork.vlan;
|
||||||
|
}
|
||||||
|
|
||||||
|
value[NETWORK].clusterNetwork = this.dedicatedNetwork.clusterNetwork;
|
||||||
|
value[NETWORK].range = this.dedicatedNetwork.range;
|
||||||
|
|
||||||
|
const excludeList = this.exclude.filter((ip) => ip);
|
||||||
|
|
||||||
|
if (Array.isArray(excludeList) && excludeList.length > 0) {
|
||||||
|
value[NETWORK].exclude = excludeList;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this.value.value = JSON.stringify(value);
|
||||||
|
},
|
||||||
|
|
||||||
|
willSave() {
|
||||||
|
this.update();
|
||||||
|
|
||||||
|
if (!this.showDedicatedNetworkConfig) {
|
||||||
|
return Promise.resolve();
|
||||||
|
}
|
||||||
|
|
||||||
|
const errors = [];
|
||||||
|
|
||||||
|
if (this.showVlan && !this.dedicatedNetwork.vlan) {
|
||||||
|
errors.push(this.t('validation.required', { key: this.t('harvester.setting.storageNetwork.vlan') }, true));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!this.dedicatedNetwork.clusterNetwork) {
|
||||||
|
errors.push(this.t('validation.required', { key: this.t('harvester.setting.storageNetwork.clusterNetwork') }, true));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!this.dedicatedNetwork.range) {
|
||||||
|
errors.push(this.t('validation.required', { key: this.t('harvester.setting.storageNetwork.range.label') }, true));
|
||||||
|
} else if (!isValidCIDR(this.dedicatedNetwork.range)) {
|
||||||
|
errors.push(this.t('harvester.setting.storageNetwork.range.invalid', null, true));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.exclude) {
|
||||||
|
const hasInvalidCIDR = this.exclude.find((cidr) => {
|
||||||
|
return cidr && !isValidCIDR(cidr);
|
||||||
|
});
|
||||||
|
|
||||||
|
if (hasInvalidCIDR) {
|
||||||
|
errors.push(this.t('harvester.setting.storageNetwork.exclude.invalid', null, true));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (errors.length > 0) {
|
||||||
|
return Promise.reject(errors);
|
||||||
|
}
|
||||||
|
|
||||||
|
return Promise.resolve();
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div :class="mode">
|
||||||
|
<Banner color="warning">
|
||||||
|
<t
|
||||||
|
k="harvester.setting.rwxNetwork.warning"
|
||||||
|
:raw="true"
|
||||||
|
/>
|
||||||
|
</Banner>
|
||||||
|
<RadioGroup
|
||||||
|
v-model:value="enabled"
|
||||||
|
class="mb-20"
|
||||||
|
name="rwx-network-enable"
|
||||||
|
:options="[true,false]"
|
||||||
|
:labels="[t('generic.enabled'), t('generic.disabled')]"
|
||||||
|
@update:value="onUpdateEnabled"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<RadioGroup
|
||||||
|
v-if="enabled"
|
||||||
|
v-model:value="shareStorageNetwork"
|
||||||
|
class="mb-20"
|
||||||
|
name="rwx-network-type"
|
||||||
|
:options="[true,false]"
|
||||||
|
:labels="[t('harvester.setting.rwxNetwork.shareStorageNetwork'), t('harvester.setting.rwxNetwork.dedicatedRwxNetwork')]"
|
||||||
|
@update:value="onUpdateNetworkType"
|
||||||
|
/>
|
||||||
|
<Banner
|
||||||
|
v-if="shareStorageNetwork"
|
||||||
|
class="mb-20"
|
||||||
|
color="warning"
|
||||||
|
>
|
||||||
|
<t
|
||||||
|
k="harvester.setting.rwxNetwork.shareStorageNetworkWarning"
|
||||||
|
:raw="true"
|
||||||
|
/>
|
||||||
|
</Banner>
|
||||||
|
<template v-if="showDedicatedNetworkConfig">
|
||||||
|
<LabeledSelect
|
||||||
|
v-model:value="networkType"
|
||||||
|
class="mb-20"
|
||||||
|
:options="networkTypes"
|
||||||
|
:mode="mode"
|
||||||
|
:label="t('harvester.fields.type')"
|
||||||
|
required
|
||||||
|
@update:value="onUpdateDedicatedType"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<LabeledInput
|
||||||
|
v-if="showVlan"
|
||||||
|
v-model:value.number="dedicatedNetwork.vlan"
|
||||||
|
type="number"
|
||||||
|
class="mb-20"
|
||||||
|
:mode="mode"
|
||||||
|
required
|
||||||
|
placeholder="e.g. 1 - 4094"
|
||||||
|
label-key="harvester.setting.storageNetwork.vlan"
|
||||||
|
@update:value="inputVlan"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<LabeledSelect
|
||||||
|
v-model:value="dedicatedNetwork.clusterNetwork"
|
||||||
|
label-key="harvester.setting.storageNetwork.clusterNetwork"
|
||||||
|
class="mb-20"
|
||||||
|
required
|
||||||
|
:options="clusterNetworkOptions"
|
||||||
|
@update:value="update"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<LabeledInput
|
||||||
|
v-model:value="dedicatedNetwork.range"
|
||||||
|
class="mb-5"
|
||||||
|
:mode="mode"
|
||||||
|
required
|
||||||
|
:placeholder="t('harvester.setting.storageNetwork.range.placeholder')"
|
||||||
|
label-key="harvester.setting.storageNetwork.range.label"
|
||||||
|
@update:value="update"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<ArrayList
|
||||||
|
v-model:value="exclude"
|
||||||
|
:show-header="true"
|
||||||
|
:default-add-value="defaultAddValue"
|
||||||
|
:mode="mode"
|
||||||
|
:add-label="t('harvester.setting.storageNetwork.exclude.addIp')"
|
||||||
|
class="mt-20"
|
||||||
|
@update:value="update"
|
||||||
|
>
|
||||||
|
<template #column-headers>
|
||||||
|
<div class="box mb-10">
|
||||||
|
<div class="key">
|
||||||
|
{{ t('harvester.setting.storageNetwork.exclude.label') }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
<template #columns="scope">
|
||||||
|
<div class="key">
|
||||||
|
<input
|
||||||
|
v-model="scope.row.value"
|
||||||
|
:placeholder="t('harvester.setting.storageNetwork.exclude.placeholder')"
|
||||||
|
@update:value="update"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</ArrayList>
|
||||||
|
</template>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
@ -18,16 +18,20 @@ export default {
|
|||||||
|
|
||||||
await this.$store.dispatch('harvester/findAll', { type: NAMESPACE });
|
await this.$store.dispatch('harvester/findAll', { type: NAMESPACE });
|
||||||
|
|
||||||
try {
|
if (this.customSupportBundleFeatureEnabled) {
|
||||||
const url = this.$store.getters['harvester-common/getHarvesterClusterUrl']('v1/harvester/namespaces?link=supportbundle');
|
try {
|
||||||
const response = await this.$store.dispatch('harvester/request', { url });
|
const url = this.$store.getters['harvester-common/getHarvesterClusterUrl']('v1/harvester/namespaces?link=supportbundle');
|
||||||
|
const response = await this.$store.dispatch('harvester/request', { url });
|
||||||
|
|
||||||
this.defaultNamespaces = response.data || [];
|
this.defaultNamespaces = response.data || [];
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
|
this.defaultNamespaces = [];
|
||||||
|
}
|
||||||
|
} else {
|
||||||
this.defaultNamespaces = [];
|
this.defaultNamespaces = [];
|
||||||
} finally {
|
|
||||||
this.loading = false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this.loading = false;
|
||||||
},
|
},
|
||||||
|
|
||||||
data() {
|
data() {
|
||||||
@ -42,24 +46,38 @@ export default {
|
|||||||
},
|
},
|
||||||
|
|
||||||
computed: {
|
computed: {
|
||||||
|
customSupportBundleFeatureEnabled() {
|
||||||
|
return this.$store.getters['harvester-common/getFeatureEnabled']('customSupportBundle');
|
||||||
|
},
|
||||||
|
|
||||||
allNamespaces() {
|
allNamespaces() {
|
||||||
return this.$store.getters['harvester/all'](NAMESPACE).map((ns) => ns.id);
|
return this.$store.getters['harvester/all'](NAMESPACE).map((ns) => ns.id);
|
||||||
},
|
},
|
||||||
|
|
||||||
filteredNamespaces() {
|
filteredNamespaces() {
|
||||||
|
if (!this.customSupportBundleFeatureEnabled) {
|
||||||
|
return this.allNamespaces;
|
||||||
|
}
|
||||||
|
|
||||||
const defaultIds = this.defaultNamespaces.map((ns) => ns.id);
|
const defaultIds = this.defaultNamespaces.map((ns) => ns.id);
|
||||||
|
|
||||||
return this.allNamespaces.filter((ns) => !defaultIds.includes(ns));
|
return this.allNamespaces.filter((ns) => !defaultIds.includes(ns));
|
||||||
},
|
},
|
||||||
|
|
||||||
namespaceOptions() {
|
namespaceOptions() {
|
||||||
|
const mappedNamespaces = this.filteredNamespaces.map((ns) => ({ label: ns, value: ns }));
|
||||||
|
|
||||||
|
if (!this.customSupportBundleFeatureEnabled) {
|
||||||
|
return mappedNamespaces;
|
||||||
|
}
|
||||||
|
|
||||||
const allSelected =
|
const allSelected =
|
||||||
this.namespaces.length === this.filteredNamespaces.length &&
|
this.namespaces.length === this.filteredNamespaces.length &&
|
||||||
this.filteredNamespaces.every((ns) => this.namespaces.includes(ns));
|
this.filteredNamespaces.every((ns) => this.namespaces.includes(ns));
|
||||||
|
|
||||||
const controlOption = allSelected ? { label: this.t('harvester.modal.bundle.namespaces.unselectAll'), value: UNSELECT_ALL } : { label: this.t('harvester.modal.bundle.namespaces.selectAll'), value: SELECT_ALL };
|
const controlOption = allSelected ? { label: this.t('harvester.modal.bundle.namespaces.unselectAll'), value: UNSELECT_ALL } : { label: this.t('harvester.modal.bundle.namespaces.selectAll'), value: SELECT_ALL };
|
||||||
|
|
||||||
return [controlOption, ...this.filteredNamespaces];
|
return [controlOption, ...mappedNamespaces];
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|||||||
@ -4,6 +4,8 @@ import { LabeledInput } from '@components/Form/LabeledInput';
|
|||||||
import LabeledSelect from '@shell/components/form/LabeledSelect';
|
import LabeledSelect from '@shell/components/form/LabeledSelect';
|
||||||
import { RadioGroup } from '@components/Form/Radio';
|
import { RadioGroup } from '@components/Form/Radio';
|
||||||
import { mapGetters } from 'vuex';
|
import { mapGetters } from 'vuex';
|
||||||
|
import { allHash } from '@shell/utils/promise';
|
||||||
|
import { NODE } from '@shell/config/types';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: 'HarvesterUpgradeConfig',
|
name: 'HarvesterUpgradeConfig',
|
||||||
@ -15,6 +17,13 @@ export default {
|
|||||||
},
|
},
|
||||||
mixins: [CreateEditView],
|
mixins: [CreateEditView],
|
||||||
|
|
||||||
|
async fetch() {
|
||||||
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
|
const hash = { nodes: this.$store.dispatch(`${ inStore }/findAll`, { type: NODE }) };
|
||||||
|
|
||||||
|
await allHash(hash);
|
||||||
|
},
|
||||||
|
|
||||||
data() {
|
data() {
|
||||||
let parseDefaultValue = {};
|
let parseDefaultValue = {};
|
||||||
|
|
||||||
@ -39,7 +48,25 @@ export default {
|
|||||||
{ value: 'skip', label: 'skip' },
|
{ value: 'skip', label: 'skip' },
|
||||||
{ value: 'parallel', label: 'parallel' }
|
{ value: 'parallel', label: 'parallel' }
|
||||||
];
|
];
|
||||||
}
|
},
|
||||||
|
nodeUpgradeOptions() {
|
||||||
|
return [
|
||||||
|
{ value: 'auto', label: 'auto' },
|
||||||
|
{ value: 'manual', label: 'manual' }
|
||||||
|
];
|
||||||
|
},
|
||||||
|
nodesOptions() {
|
||||||
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
|
const nodes = this.$store.getters[`${ inStore }/all`](NODE);
|
||||||
|
|
||||||
|
return nodes.map((node) => ({ value: node.id, label: node.name }));
|
||||||
|
},
|
||||||
|
showPauseNodes() {
|
||||||
|
return this.parseDefaultValue.nodeUpgradeOption?.strategy?.mode === 'manual';
|
||||||
|
},
|
||||||
|
resumeUpgradePausedNodeEnabled() {
|
||||||
|
return this.$store.getters['harvester-common/getFeatureEnabled']('resumeUpgradePausedNode');
|
||||||
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
created() {
|
created() {
|
||||||
@ -48,6 +75,18 @@ export default {
|
|||||||
|
|
||||||
methods: {
|
methods: {
|
||||||
normalizeValue(obj) {
|
normalizeValue(obj) {
|
||||||
|
// handle nodeUpgradeOption.strategy
|
||||||
|
if (obj?.nodeUpgradeOption?.strategy?.mode === 'auto') {
|
||||||
|
delete obj.nodeUpgradeOption.strategy.pauseNodes;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (obj?.nodeUpgradeOption?.strategy?.mode === 'manual') {
|
||||||
|
if (!Array.isArray(obj.nodeUpgradeOption.strategy.pauseNodes)) {
|
||||||
|
obj.nodeUpgradeOption.strategy.pauseNodes = this.nodesOptions.map((node) => node.value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// handle imagePreloadOption.strategy
|
||||||
if (!obj.imagePreloadOption) {
|
if (!obj.imagePreloadOption) {
|
||||||
obj.imagePreloadOption = { strategy: { type: 'sequential' } };
|
obj.imagePreloadOption = { strategy: { type: 'sequential' } };
|
||||||
}
|
}
|
||||||
@ -105,8 +144,8 @@ export default {
|
|||||||
this.update();
|
this.update();
|
||||||
},
|
},
|
||||||
deep: true
|
deep: true
|
||||||
}
|
},
|
||||||
}
|
},
|
||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
@ -144,6 +183,28 @@ export default {
|
|||||||
:labels="[t('generic.enabled'), t('generic.disabled')]"
|
:labels="[t('generic.enabled'), t('generic.disabled')]"
|
||||||
@update:value="update"
|
@update:value="update"
|
||||||
/>
|
/>
|
||||||
|
<div v-if="resumeUpgradePausedNodeEnabled">
|
||||||
|
<label class="mb-5"><b>{{ t('harvester.setting.upgrade.nodeUpgradeOption') }}</b></label>
|
||||||
|
<LabeledSelect
|
||||||
|
v-model:value="parseDefaultValue.nodeUpgradeOption.strategy.mode"
|
||||||
|
class="mb-20 label-select"
|
||||||
|
:mode="mode"
|
||||||
|
:label="t('harvester.setting.upgrade.strategy')"
|
||||||
|
:options="nodeUpgradeOptions"
|
||||||
|
@update:value="update"
|
||||||
|
/>
|
||||||
|
<LabeledSelect
|
||||||
|
v-if="showPauseNodes"
|
||||||
|
v-model:value="parseDefaultValue.nodeUpgradeOption.strategy.pauseNodes"
|
||||||
|
class="mb-20 label-select"
|
||||||
|
:clearable="true"
|
||||||
|
:multiple="true"
|
||||||
|
:mode="mode"
|
||||||
|
:label="t('harvester.setting.upgrade.pauseNodes')"
|
||||||
|
:options="nodesOptions"
|
||||||
|
@update:value="update"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
<div
|
<div
|
||||||
v-if="errors.length"
|
v-if="errors.length"
|
||||||
class="error"
|
class="error"
|
||||||
|
|||||||
@ -69,6 +69,15 @@ export default {
|
|||||||
:mode="mode"
|
:mode="mode"
|
||||||
label-key="harvester.setting.vmForceDeletionPolicy.period"
|
label-key="harvester.setting.vmForceDeletionPolicy.period"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
<LabeledInput
|
||||||
|
v-if="parseDefaultValue.enable"
|
||||||
|
v-model:value.number="parseDefaultValue.vmMigrationTimeout"
|
||||||
|
class="mb-20"
|
||||||
|
type="number"
|
||||||
|
:mode="mode"
|
||||||
|
label-key="harvester.setting.vmForceDeletionPolicy.vmMigrationTimeout"
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@ -9,4 +9,5 @@ export const DOC = {
|
|||||||
SUPPORT_BUNDLE_NAMESPACES: `/advanced/index/#support-bundle-namespaces`,
|
SUPPORT_BUNDLE_NAMESPACES: `/advanced/index/#support-bundle-namespaces`,
|
||||||
VPC_CONFIGURATION_EXAMPLES: `/networking/kubeovn-vpc#vpc-peering-configuration-examples`,
|
VPC_CONFIGURATION_EXAMPLES: `/networking/kubeovn-vpc#vpc-peering-configuration-examples`,
|
||||||
NETWORK_POLICY: `/networking/kubeovn-vm-isolation/#network-policies`,
|
NETWORK_POLICY: `/networking/kubeovn-vm-isolation/#network-policies`,
|
||||||
|
TRANSPARENT_HUGEPAGES: `https://docs.kernel.org/admin-guide/mm/transhuge.html`,
|
||||||
};
|
};
|
||||||
|
|||||||
@ -53,8 +53,19 @@ const FEATURE_FLAGS = {
|
|||||||
'vmMachineTypeAuto',
|
'vmMachineTypeAuto',
|
||||||
'lhV2VolExpansion',
|
'lhV2VolExpansion',
|
||||||
'l2VlanTrunkMode',
|
'l2VlanTrunkMode',
|
||||||
'kubevirtMigration'
|
'kubevirtMigration',
|
||||||
]
|
'hotplugNic',
|
||||||
|
'resumeUpgradePausedNode',
|
||||||
|
],
|
||||||
|
'v1.7.1': [],
|
||||||
|
'v1.8.0': [
|
||||||
|
'hotplugCdRom',
|
||||||
|
'supportBundleFileNameSetting',
|
||||||
|
'clusterRegistrationTLSVerify',
|
||||||
|
'vGPUAsPCIDevice',
|
||||||
|
'instanceManagerResourcesSetting',
|
||||||
|
'rwxNetworkSetting',
|
||||||
|
],
|
||||||
};
|
};
|
||||||
|
|
||||||
const generateFeatureFlags = () => {
|
const generateFeatureFlags = () => {
|
||||||
|
|||||||
@ -35,8 +35,21 @@ import {
|
|||||||
SNAPSHOT_TARGET_VOLUME,
|
SNAPSHOT_TARGET_VOLUME,
|
||||||
IMAGE_VIRTUAL_SIZE,
|
IMAGE_VIRTUAL_SIZE,
|
||||||
IMAGE_STORAGE_CLASS,
|
IMAGE_STORAGE_CLASS,
|
||||||
HARVESTER_DESCRIPTION
|
HARVESTER_DESCRIPTION,
|
||||||
|
VM_IMPORT_SOURCE_VM,
|
||||||
|
VM_IMPORT_SOURCE_CLUSTER,
|
||||||
|
VM_IMPORT_STATUS,
|
||||||
|
VM_IMPORT_SOURCE_V_DC,
|
||||||
|
VM_IMPORT_SOURCE_V_ENDPOINT,
|
||||||
|
VM_IMPORT_SOURCE_V_STATUS,
|
||||||
|
VM_IMPORT_SOURCE_O_REGION,
|
||||||
|
VM_IMPORT_SOURCE_O_ENDPOINT,
|
||||||
|
VM_IMPORT_SOURCE_O_STATUS,
|
||||||
|
VM_IMPORT_SOURCE_OVA_URL,
|
||||||
|
VM_IMPORT_SOURCE_OVA_STATUS,
|
||||||
} from './table-headers';
|
} from './table-headers';
|
||||||
|
import { ADD_ONS } from './harvester-map';
|
||||||
|
import { registerAddonSideNav } from '../utils/dynamic-nav';
|
||||||
|
|
||||||
const TEMPLATE = HCI.VM_VERSION;
|
const TEMPLATE = HCI.VM_VERSION;
|
||||||
const MONITORING_GROUP = 'Monitoring & Logging::Monitoring';
|
const MONITORING_GROUP = 'Monitoring & Logging::Monitoring';
|
||||||
@ -195,6 +208,142 @@ export function init($plugin, store) {
|
|||||||
exact: false
|
exact: false
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// ===========================================================================
|
||||||
|
// VM Import Controller UI Flow
|
||||||
|
// ===========================================================================
|
||||||
|
// Define group (Hidden by default)
|
||||||
|
weightGroup('vmimport', 0, false);
|
||||||
|
|
||||||
|
// VirtualMachineImport
|
||||||
|
headers(HCI.VMIMPORT, [
|
||||||
|
STATE,
|
||||||
|
NAME_COL,
|
||||||
|
NAMESPACE_COL,
|
||||||
|
VM_IMPORT_SOURCE_VM,
|
||||||
|
VM_IMPORT_SOURCE_CLUSTER,
|
||||||
|
VM_IMPORT_STATUS,
|
||||||
|
AGE
|
||||||
|
]);
|
||||||
|
configureType(HCI.VMIMPORT, {
|
||||||
|
resource: HCI.VMIMPORT,
|
||||||
|
resourceDetail: HCI.VMIMPORT,
|
||||||
|
resourceEdit: HCI.VMIMPORT,
|
||||||
|
location: {
|
||||||
|
name: `${ PRODUCT_NAME }-c-cluster-resource`,
|
||||||
|
params: { resource: HCI.VMIMPORT }
|
||||||
|
}
|
||||||
|
});
|
||||||
|
virtualType({ // needed to avoid 404 on refresh when combined with registerAddonSideNav()
|
||||||
|
name: HCI.VMIMPORT,
|
||||||
|
labelKey: 'harvester.addons.vmImport.labels.vmimport',
|
||||||
|
group: 'vmimport',
|
||||||
|
namespaced: true,
|
||||||
|
route: {
|
||||||
|
name: `${ PRODUCT_NAME }-c-cluster-resource`,
|
||||||
|
params: { resource: HCI.VMIMPORT }
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Source: VMware
|
||||||
|
headers(HCI.VMIMPORT_SOURCE_V, [
|
||||||
|
STATE,
|
||||||
|
NAME_COL,
|
||||||
|
VM_IMPORT_SOURCE_V_ENDPOINT,
|
||||||
|
VM_IMPORT_SOURCE_V_DC,
|
||||||
|
VM_IMPORT_SOURCE_V_STATUS,
|
||||||
|
AGE
|
||||||
|
]);
|
||||||
|
configureType(HCI.VMIMPORT_SOURCE_V, {
|
||||||
|
resource: HCI.VMIMPORT_SOURCE_V,
|
||||||
|
resourceDetail: HCI.VMIMPORT_SOURCE_V,
|
||||||
|
resourceEdit: HCI.VMIMPORT_SOURCE_V,
|
||||||
|
location: {
|
||||||
|
name: `${ PRODUCT_NAME }-c-cluster-resource`,
|
||||||
|
params: { resource: HCI.VMIMPORT_SOURCE_V }
|
||||||
|
}
|
||||||
|
});
|
||||||
|
virtualType({ // needed to avoid 404 on refresh when combined with registerAddonSideNav()
|
||||||
|
name: HCI.VMIMPORT_SOURCE_V,
|
||||||
|
labelKey: 'harvester.addons.vmImport.labels.vmimportSourceVMWare',
|
||||||
|
group: 'vmimport',
|
||||||
|
namespaced: true,
|
||||||
|
route: {
|
||||||
|
name: `${ PRODUCT_NAME }-c-cluster-resource`,
|
||||||
|
params: { resource: HCI.VMIMPORT_SOURCE_V }
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Source: OpenStack
|
||||||
|
headers(HCI.VMIMPORT_SOURCE_O, [
|
||||||
|
STATE,
|
||||||
|
NAME_COL,
|
||||||
|
VM_IMPORT_SOURCE_O_ENDPOINT,
|
||||||
|
VM_IMPORT_SOURCE_O_REGION,
|
||||||
|
VM_IMPORT_SOURCE_O_STATUS,
|
||||||
|
AGE
|
||||||
|
]);
|
||||||
|
configureType(HCI.VMIMPORT_SOURCE_O, {
|
||||||
|
resource: HCI.VMIMPORT_SOURCE_O,
|
||||||
|
resourceDetail: HCI.VMIMPORT_SOURCE_O,
|
||||||
|
resourceEdit: HCI.VMIMPORT_SOURCE_O,
|
||||||
|
location: {
|
||||||
|
name: `${ PRODUCT_NAME }-c-cluster-resource`,
|
||||||
|
params: { resource: HCI.VMIMPORT_SOURCE_O }
|
||||||
|
}
|
||||||
|
});
|
||||||
|
virtualType({ // needed to avoid 404 on refresh when combined with registerAddonSideNav()
|
||||||
|
name: HCI.VMIMPORT_SOURCE_O,
|
||||||
|
labelKey: 'harvester.addons.vmImport.labels.vmimportSourceOpenStack',
|
||||||
|
group: 'vmimport',
|
||||||
|
namespaced: true,
|
||||||
|
route: {
|
||||||
|
name: `${ PRODUCT_NAME }-c-cluster-resource`,
|
||||||
|
params: { resource: HCI.VMIMPORT_SOURCE_O }
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Source: OVA
|
||||||
|
headers(HCI.VMIMPORT_SOURCE_OVA, [
|
||||||
|
STATE,
|
||||||
|
NAME_COL,
|
||||||
|
VM_IMPORT_SOURCE_OVA_URL,
|
||||||
|
VM_IMPORT_SOURCE_OVA_STATUS,
|
||||||
|
AGE
|
||||||
|
]);
|
||||||
|
configureType(HCI.VMIMPORT_SOURCE_OVA, {
|
||||||
|
resource: HCI.VMIMPORT_SOURCE_OVA,
|
||||||
|
resourceDetail: HCI.VMIMPORT_SOURCE_OVA,
|
||||||
|
resourceEdit: HCI.VMIMPORT_SOURCE_OVA,
|
||||||
|
location: {
|
||||||
|
name: `${ PRODUCT_NAME }-c-cluster-resource`,
|
||||||
|
params: { resource: HCI.VMIMPORT_SOURCE_OVA }
|
||||||
|
}
|
||||||
|
});
|
||||||
|
virtualType({ // needed to avoid 404 on refresh when combined with registerAddonSideNav()
|
||||||
|
name: HCI.VMIMPORT_SOURCE_OVA,
|
||||||
|
labelKey: 'harvester.addons.vmImport.labels.vmimportSourceOVA',
|
||||||
|
group: 'vmimport',
|
||||||
|
namespaced: true,
|
||||||
|
route: {
|
||||||
|
name: `${ PRODUCT_NAME }-c-cluster-resource`,
|
||||||
|
params: { resource: HCI.VMIMPORT_SOURCE_OVA }
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Enable SideNav based on Addon Status
|
||||||
|
registerAddonSideNav(store, PRODUCT_NAME, {
|
||||||
|
addonName: ADD_ONS.VM_IMPORT_CONTROLLER,
|
||||||
|
resourceType: HCI.ADD_ONS,
|
||||||
|
navGroup: 'vmimport',
|
||||||
|
types: [
|
||||||
|
HCI.VMIMPORT_SOURCE_V,
|
||||||
|
HCI.VMIMPORT_SOURCE_O,
|
||||||
|
HCI.VMIMPORT_SOURCE_OVA,
|
||||||
|
HCI.VMIMPORT
|
||||||
|
]
|
||||||
|
});
|
||||||
|
// ===========================================================================
|
||||||
|
|
||||||
basicType([HCI.VOLUME]);
|
basicType([HCI.VOLUME]);
|
||||||
configureType(HCI.VOLUME, {
|
configureType(HCI.VOLUME, {
|
||||||
location: {
|
location: {
|
||||||
|
|||||||
@ -39,6 +39,12 @@ export const VOLUME_TYPE = [{
|
|||||||
value: 'cd-rom'
|
value: 'cd-rom'
|
||||||
}];
|
}];
|
||||||
|
|
||||||
|
export const VOLUME_HOTPLUG_ACTION = {
|
||||||
|
INSERT_CDROM_IMAGE: 'INSERT_CDROM_IMAGE',
|
||||||
|
EJECT_CDROM_IMAGE: 'EJECT_CDROM_IMAGE',
|
||||||
|
DETACH_DISK: 'DETACH_DISK'
|
||||||
|
};
|
||||||
|
|
||||||
export const ACCESS_CREDENTIALS = {
|
export const ACCESS_CREDENTIALS = {
|
||||||
RESET_PWD: 'userPassword',
|
RESET_PWD: 'userPassword',
|
||||||
INJECT_SSH: 'sshPublicKey'
|
INJECT_SSH: 'sshPublicKey'
|
||||||
|
|||||||
@ -28,6 +28,7 @@ export const HCI = {
|
|||||||
NODE_ROLE_CONTROL_PLANE: 'node-role.kubernetes.io/control-plane',
|
NODE_ROLE_CONTROL_PLANE: 'node-role.kubernetes.io/control-plane',
|
||||||
NODE_ROLE_ETCD: 'node-role.harvesterhci.io/witness',
|
NODE_ROLE_ETCD: 'node-role.harvesterhci.io/witness',
|
||||||
PROMOTE_STATUS: 'harvesterhci.io/promote-status',
|
PROMOTE_STATUS: 'harvesterhci.io/promote-status',
|
||||||
|
CLONE_BACKEND_STORAGE_STATUS: 'harvesterhci.io/clone-backend-storage-status',
|
||||||
MIGRATION_STATE: 'harvesterhci.io/migrationState',
|
MIGRATION_STATE: 'harvesterhci.io/migrationState',
|
||||||
VOLUME_CLAIM_TEMPLATE: 'harvesterhci.io/volumeClaimTemplates',
|
VOLUME_CLAIM_TEMPLATE: 'harvesterhci.io/volumeClaimTemplates',
|
||||||
IMAGE_NAME: 'harvesterhci.io/image-name',
|
IMAGE_NAME: 'harvesterhci.io/image-name',
|
||||||
@ -50,6 +51,7 @@ export const HCI = {
|
|||||||
STORAGE_CLASS: 'harvesterhci.io/storageClassName',
|
STORAGE_CLASS: 'harvesterhci.io/storageClassName',
|
||||||
STORAGE_NETWORK: 'storage-network.settings.harvesterhci.io',
|
STORAGE_NETWORK: 'storage-network.settings.harvesterhci.io',
|
||||||
ADDON_EXPERIMENTAL: 'addon.harvesterhci.io/experimental',
|
ADDON_EXPERIMENTAL: 'addon.harvesterhci.io/experimental',
|
||||||
|
ADDON_DISPLAYNAME: 'addon.harvesterhci.io/displayName',
|
||||||
VOLUME_ERROR: 'longhorn.io/volume-scheduling-error',
|
VOLUME_ERROR: 'longhorn.io/volume-scheduling-error',
|
||||||
VOLUME_FOR_VM: 'harvesterhci.io/volumeForVirtualMachine',
|
VOLUME_FOR_VM: 'harvesterhci.io/volumeForVirtualMachine',
|
||||||
KVM_AMD_CPU: 'cpu-feature.node.kubevirt.io/svm',
|
KVM_AMD_CPU: 'cpu-feature.node.kubevirt.io/svm',
|
||||||
@ -76,4 +78,6 @@ export const HCI = {
|
|||||||
CLONE_STRATEGY: 'cdi.harvesterhci.io/storageProfileCloneStrategy',
|
CLONE_STRATEGY: 'cdi.harvesterhci.io/storageProfileCloneStrategy',
|
||||||
VOLUME_MODE_ACCESS_MODES: 'cdi.harvesterhci.io/storageProfileVolumeModeAccessModes',
|
VOLUME_MODE_ACCESS_MODES: 'cdi.harvesterhci.io/storageProfileVolumeModeAccessModes',
|
||||||
VOLUME_SNAPSHOT_CLASS: 'cdi.harvesterhci.io/storageProfileVolumeSnapshotClass',
|
VOLUME_SNAPSHOT_CLASS: 'cdi.harvesterhci.io/storageProfileVolumeSnapshotClass',
|
||||||
|
MAC_ADDRESS: 'harvesterhci.io/mac-address',
|
||||||
|
NODE_UPGRADE_PAUSE_MAP: 'harvesterhci.io/node-upgrade-pause-map',
|
||||||
};
|
};
|
||||||
|
|||||||
@ -16,9 +16,11 @@ export const HCI_SETTING = {
|
|||||||
DEFAULT_STORAGE_CLASS: 'default-storage-class',
|
DEFAULT_STORAGE_CLASS: 'default-storage-class',
|
||||||
SUPPORT_BUNDLE_TIMEOUT: 'support-bundle-timeout',
|
SUPPORT_BUNDLE_TIMEOUT: 'support-bundle-timeout',
|
||||||
SUPPORT_BUNDLE_EXPIRATION: 'support-bundle-expiration',
|
SUPPORT_BUNDLE_EXPIRATION: 'support-bundle-expiration',
|
||||||
|
SUPPORT_BUNDLE_FILE_NAME: 'support-bundle-file-name',
|
||||||
SUPPORT_BUNDLE_IMAGE: 'support-bundle-image',
|
SUPPORT_BUNDLE_IMAGE: 'support-bundle-image',
|
||||||
SUPPORT_BUNDLE_NODE_COLLECTION_TIMEOUT: 'support-bundle-node-collection-timeout',
|
SUPPORT_BUNDLE_NODE_COLLECTION_TIMEOUT: 'support-bundle-node-collection-timeout',
|
||||||
STORAGE_NETWORK: 'storage-network',
|
STORAGE_NETWORK: 'storage-network',
|
||||||
|
RWX_NETWORK: 'rwx-network',
|
||||||
VM_FORCE_RESET_POLICY: 'vm-force-reset-policy',
|
VM_FORCE_RESET_POLICY: 'vm-force-reset-policy',
|
||||||
SSL_CERTIFICATES: 'ssl-certificates',
|
SSL_CERTIFICATES: 'ssl-certificates',
|
||||||
SSL_PARAMETERS: 'ssl-parameters',
|
SSL_PARAMETERS: 'ssl-parameters',
|
||||||
@ -38,7 +40,8 @@ export const HCI_SETTING = {
|
|||||||
VM_MIGRATION_NETWORK: 'vm-migration-network',
|
VM_MIGRATION_NETWORK: 'vm-migration-network',
|
||||||
RANCHER_CLUSTER: 'rancher-cluster',
|
RANCHER_CLUSTER: 'rancher-cluster',
|
||||||
MAX_HOTPLUG_RATIO: 'max-hotplug-ratio',
|
MAX_HOTPLUG_RATIO: 'max-hotplug-ratio',
|
||||||
KUBEVIRT_MIGRATION: 'kubevirt-migration'
|
KUBEVIRT_MIGRATION: 'kubevirt-migration',
|
||||||
|
INSTANCE_MANAGER_RESOURCES: 'instance-manager-resources'
|
||||||
};
|
};
|
||||||
|
|
||||||
export const HCI_ALLOWED_SETTINGS = {
|
export const HCI_ALLOWED_SETTINGS = {
|
||||||
@ -71,11 +74,17 @@ export const HCI_ALLOWED_SETTINGS = {
|
|||||||
[HCI_SETTING.OVERCOMMIT_CONFIG]: { kind: 'json', from: 'import' },
|
[HCI_SETTING.OVERCOMMIT_CONFIG]: { kind: 'json', from: 'import' },
|
||||||
[HCI_SETTING.SUPPORT_BUNDLE_TIMEOUT]: { kind: 'number' },
|
[HCI_SETTING.SUPPORT_BUNDLE_TIMEOUT]: { kind: 'number' },
|
||||||
[HCI_SETTING.SUPPORT_BUNDLE_EXPIRATION]: { kind: 'number' },
|
[HCI_SETTING.SUPPORT_BUNDLE_EXPIRATION]: { kind: 'number' },
|
||||||
|
[HCI_SETTING.SUPPORT_BUNDLE_FILE_NAME]: {
|
||||||
|
kind: 'string', canReset: true, featureFlag: 'supportBundleFileNameSetting'
|
||||||
|
},
|
||||||
[HCI_SETTING.SUPPORT_BUNDLE_NODE_COLLECTION_TIMEOUT]: { kind: 'number', featureFlag: 'supportBundleNodeCollectionTimeoutSetting' },
|
[HCI_SETTING.SUPPORT_BUNDLE_NODE_COLLECTION_TIMEOUT]: { kind: 'number', featureFlag: 'supportBundleNodeCollectionTimeoutSetting' },
|
||||||
[HCI_SETTING.SUPPORT_BUNDLE_IMAGE]: { kind: 'json', from: 'import' },
|
[HCI_SETTING.SUPPORT_BUNDLE_IMAGE]: { kind: 'json', from: 'import' },
|
||||||
[HCI_SETTING.STORAGE_NETWORK]: {
|
[HCI_SETTING.STORAGE_NETWORK]: {
|
||||||
kind: 'custom', from: 'import', canReset: true
|
kind: 'custom', from: 'import', canReset: true
|
||||||
},
|
},
|
||||||
|
[HCI_SETTING.RWX_NETWORK]: {
|
||||||
|
kind: 'json', from: 'import', canReset: true, featureFlag: 'rwxNetworkSetting'
|
||||||
|
},
|
||||||
[HCI_SETTING.VM_FORCE_RESET_POLICY]: { kind: 'json', from: 'import' },
|
[HCI_SETTING.VM_FORCE_RESET_POLICY]: { kind: 'json', from: 'import' },
|
||||||
[HCI_SETTING.SSL_CERTIFICATES]: { kind: 'json', from: 'import' },
|
[HCI_SETTING.SSL_CERTIFICATES]: { kind: 'json', from: 'import' },
|
||||||
[HCI_SETTING.SSL_PARAMETERS]: {
|
[HCI_SETTING.SSL_PARAMETERS]: {
|
||||||
@ -118,12 +127,16 @@ export const HCI_ALLOWED_SETTINGS = {
|
|||||||
},
|
},
|
||||||
[HCI_SETTING.KUBEVIRT_MIGRATION]: {
|
[HCI_SETTING.KUBEVIRT_MIGRATION]: {
|
||||||
kind: 'json', from: 'import', canReset: true, featureFlag: 'kubevirtMigration',
|
kind: 'json', from: 'import', canReset: true, featureFlag: 'kubevirtMigration',
|
||||||
|
},
|
||||||
|
[HCI_SETTING.INSTANCE_MANAGER_RESOURCES]: {
|
||||||
|
kind: 'json', from: 'import', featureFlag: 'instanceManagerResourcesSetting'
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
export const HCI_SINGLE_CLUSTER_ALLOWED_SETTING = {
|
export const HCI_SINGLE_CLUSTER_ALLOWED_SETTING = {
|
||||||
[HCI_SETTING.CLUSTER_REGISTRATION_URL]: {
|
[HCI_SETTING.CLUSTER_REGISTRATION_URL]: {
|
||||||
kind: 'url',
|
kind: 'custom',
|
||||||
|
from: 'import',
|
||||||
canReset: true,
|
canReset: true,
|
||||||
},
|
},
|
||||||
[HCI_SETTING.UI_PL]: {
|
[HCI_SETTING.UI_PL]: {
|
||||||
|
|||||||
@ -131,3 +131,102 @@ export const PROVIDER = {
|
|||||||
value: 'spec.provider',
|
value: 'spec.provider',
|
||||||
align: 'left',
|
align: 'left',
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Source VM column in migration.harvesterhci.io.virtualmachineimport list page
|
||||||
|
export const VM_IMPORT_SOURCE_VM = {
|
||||||
|
name: 'sourceVm',
|
||||||
|
labelKey: 'harvester.tableHeaders.vmImportSourceVm',
|
||||||
|
value: 'spec.virtualMachineName',
|
||||||
|
sort: 'spec.virtualMachineName',
|
||||||
|
align: 'left',
|
||||||
|
};
|
||||||
|
|
||||||
|
// Source Cluster column in migration.harvesterhci.io.virtualmachineimport list page
|
||||||
|
export const VM_IMPORT_SOURCE_CLUSTER = {
|
||||||
|
name: 'sourceCluster',
|
||||||
|
labelKey: 'harvester.tableHeaders.vmImportSourceCluster',
|
||||||
|
value: 'spec.sourceCluster.name',
|
||||||
|
sort: 'spec.sourceCluster.name',
|
||||||
|
align: 'left',
|
||||||
|
};
|
||||||
|
|
||||||
|
// Import Status column in migration.harvesterhci.io.virtualmachineimport list page
|
||||||
|
export const VM_IMPORT_STATUS = {
|
||||||
|
name: 'importStatus',
|
||||||
|
labelKey: 'harvester.tableHeaders.vmImportStatus',
|
||||||
|
value: 'status.importStatus',
|
||||||
|
sort: 'status.importStatus',
|
||||||
|
align: 'left',
|
||||||
|
};
|
||||||
|
|
||||||
|
// Datacenter column in migration.harvesterhci.io.vmwaresource list page
|
||||||
|
export const VM_IMPORT_SOURCE_V_DC = {
|
||||||
|
name: 'datacenter',
|
||||||
|
labelKey: 'harvester.tableHeaders.vmImportSourceVDatacenter',
|
||||||
|
value: 'spec.dc',
|
||||||
|
sort: 'spec.dc',
|
||||||
|
align: 'left',
|
||||||
|
};
|
||||||
|
|
||||||
|
// Endpoint column in migration.harvesterhci.io.vmwaresource list page
|
||||||
|
export const VM_IMPORT_SOURCE_V_ENDPOINT = {
|
||||||
|
name: 'endpoint',
|
||||||
|
labelKey: 'harvester.tableHeaders.vmImportSourceVEndpoint',
|
||||||
|
value: 'spec.endpoint',
|
||||||
|
sort: 'spec.endpoint',
|
||||||
|
align: 'left',
|
||||||
|
};
|
||||||
|
|
||||||
|
// Cluster Status column in migration.harvesterhci.io.vmwaresource list page
|
||||||
|
export const VM_IMPORT_SOURCE_V_STATUS = {
|
||||||
|
name: 'clusterStatus',
|
||||||
|
labelKey: 'harvester.tableHeaders.vmImportSourceVClusterStatus',
|
||||||
|
value: 'status.status',
|
||||||
|
sort: 'status.status',
|
||||||
|
align: 'left',
|
||||||
|
};
|
||||||
|
|
||||||
|
// Region column in migration.harvesterhci.io.openstacksource list page
|
||||||
|
export const VM_IMPORT_SOURCE_O_REGION = {
|
||||||
|
name: 'region',
|
||||||
|
labelKey: 'harvester.tableHeaders.vmImportSourceORegion',
|
||||||
|
value: 'spec.region',
|
||||||
|
sort: 'spec.region',
|
||||||
|
align: 'left',
|
||||||
|
};
|
||||||
|
|
||||||
|
// Endpoint column in migration.harvesterhci.io.openstacksource list page
|
||||||
|
export const VM_IMPORT_SOURCE_O_ENDPOINT = {
|
||||||
|
name: 'endpoint',
|
||||||
|
labelKey: 'harvester.tableHeaders.vmImportSourceOEndpoint',
|
||||||
|
value: 'spec.endpoint',
|
||||||
|
sort: 'spec.endpoint',
|
||||||
|
align: 'left',
|
||||||
|
};
|
||||||
|
|
||||||
|
// Cluster Status column in migration.harvesterhci.io.openstacksource list page
|
||||||
|
export const VM_IMPORT_SOURCE_O_STATUS = {
|
||||||
|
name: 'clusterStatus',
|
||||||
|
labelKey: 'harvester.tableHeaders.vmImportSourceOClusterStatus',
|
||||||
|
value: 'status.status',
|
||||||
|
sort: 'status.status',
|
||||||
|
align: 'left',
|
||||||
|
};
|
||||||
|
|
||||||
|
// URL column in migration.harvesterhci.io.ovasource list page
|
||||||
|
export const VM_IMPORT_SOURCE_OVA_URL = {
|
||||||
|
name: 'url',
|
||||||
|
labelKey: 'harvester.tableHeaders.vmImportSourceOVAUrl',
|
||||||
|
value: 'spec.url',
|
||||||
|
sort: 'spec.url',
|
||||||
|
align: 'left',
|
||||||
|
};
|
||||||
|
|
||||||
|
// Status column in migration.harvesterhci.io.ovasource list page
|
||||||
|
export const VM_IMPORT_SOURCE_OVA_STATUS = {
|
||||||
|
name: 'status',
|
||||||
|
labelKey: 'harvester.tableHeaders.vmImportSourceOVAStatus',
|
||||||
|
value: 'status.status',
|
||||||
|
sort: 'status.status',
|
||||||
|
align: 'left',
|
||||||
|
};
|
||||||
|
|||||||
@ -29,3 +29,15 @@ export const L2VLAN_MODE = {
|
|||||||
ACCESS: 'access',
|
ACCESS: 'access',
|
||||||
TRUNK: 'trunk',
|
TRUNK: 'trunk',
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const VMIMPORT_SOURCE_PROVIDER = {
|
||||||
|
VMWARE: 'vmware',
|
||||||
|
OPENSTACK: 'openstack',
|
||||||
|
OVA: 'ova',
|
||||||
|
};
|
||||||
|
|
||||||
|
export const VMIMPORT_SOURCE_KINDS = {
|
||||||
|
VMWARE: 'VmwareSource',
|
||||||
|
OPENSTACK: 'OpenstackSource',
|
||||||
|
OVA: 'OvaSource',
|
||||||
|
};
|
||||||
|
|||||||
119
pkg/harvester/detail/harvesterhci.io.host/HarvesterHugepages.vue
Normal file
119
pkg/harvester/detail/harvesterhci.io.host/HarvesterHugepages.vue
Normal file
@ -0,0 +1,119 @@
|
|||||||
|
<script>
|
||||||
|
import LabelValue from '@shell/components/LabelValue';
|
||||||
|
import { HCI } from '../../types';
|
||||||
|
import { DOC } from '../../config/doc-links';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'HarvesterHugepages',
|
||||||
|
components: { LabelValue },
|
||||||
|
|
||||||
|
props: {
|
||||||
|
node: {
|
||||||
|
type: Object,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
computed: {
|
||||||
|
docsTransparentHugepagesLink() {
|
||||||
|
return DOC.TRANSPARENT_HUGEPAGES;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
async fetch() {
|
||||||
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
|
|
||||||
|
const hash = await this.$store.dispatch(`${ inStore }/findAll`, { type: HCI.HUGEPAGES });
|
||||||
|
|
||||||
|
this.hugepages = hash.find((node) => {
|
||||||
|
return node.id === this.node.id;
|
||||||
|
}) || {};
|
||||||
|
},
|
||||||
|
|
||||||
|
data() {
|
||||||
|
return { hugepages: {} };
|
||||||
|
},
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<template v-if="hugepages.status">
|
||||||
|
<h2>{{ t('harvester.host.hugepages.meminfo') }}</h2>
|
||||||
|
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-6">
|
||||||
|
<LabelValue
|
||||||
|
:name="t('harvester.host.hugepages.status.anon')"
|
||||||
|
:value="hugepages.status.meminfo.anonHugePages"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="col span-6">
|
||||||
|
<LabelValue
|
||||||
|
:name="t('harvester.host.hugepages.status.size')"
|
||||||
|
:value="hugepages.status.meminfo.hugepageSize"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-3">
|
||||||
|
<LabelValue
|
||||||
|
:name="t('harvester.host.hugepages.status.total')"
|
||||||
|
:value="hugepages.status.meminfo.hugePagesTotal"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="col span-3">
|
||||||
|
<LabelValue
|
||||||
|
:name="t('harvester.host.hugepages.status.free')"
|
||||||
|
:value="hugepages.status.meminfo.hugePagesFree"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="col span-3">
|
||||||
|
<LabelValue
|
||||||
|
:name="t('harvester.host.hugepages.status.rsvd')"
|
||||||
|
:value="hugepages.status.meminfo.hugePagesRsvd"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="col span-3">
|
||||||
|
<LabelValue
|
||||||
|
:name="t('harvester.host.hugepages.status.surp')"
|
||||||
|
:value="hugepages.status.meminfo.hugePagesSurp"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<hr class="divider" />
|
||||||
|
<h3>
|
||||||
|
<t
|
||||||
|
k="harvester.host.hugepages.transparent.title"
|
||||||
|
:raw="true"
|
||||||
|
:url="docsTransparentHugepagesLink"
|
||||||
|
/>
|
||||||
|
</h3>
|
||||||
|
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-4">
|
||||||
|
<LabelValue
|
||||||
|
:name="t('harvester.host.hugepages.transparent.enabled')"
|
||||||
|
:value="hugepages.spec.transparent.enabled"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="col span-4">
|
||||||
|
<LabelValue
|
||||||
|
:name="t('harvester.host.hugepages.transparent.shmemEnabled')"
|
||||||
|
:value="hugepages.spec.transparent.shmemEnabled"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="col span-4">
|
||||||
|
<LabelValue
|
||||||
|
:name="t('harvester.host.hugepages.transparent.defrag')"
|
||||||
|
:value="hugepages.spec.transparent.defrag"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
@ -27,6 +27,7 @@ import Instance from './VirtualMachineInstance';
|
|||||||
import Disk from './HarvesterHostDisk';
|
import Disk from './HarvesterHostDisk';
|
||||||
import VlanStatus from './VlanStatus';
|
import VlanStatus from './VlanStatus';
|
||||||
import HarvesterKsmtuned from './HarvesterKsmtuned.vue';
|
import HarvesterKsmtuned from './HarvesterKsmtuned.vue';
|
||||||
|
import HarvesterHugepages from './HarvesterHugepages.vue';
|
||||||
import HarvesterSeeder from './HarvesterSeeder';
|
import HarvesterSeeder from './HarvesterSeeder';
|
||||||
|
|
||||||
const LONGHORN_SYSTEM = 'longhorn-system';
|
const LONGHORN_SYSTEM = 'longhorn-system';
|
||||||
@ -46,6 +47,7 @@ export default {
|
|||||||
VlanStatus,
|
VlanStatus,
|
||||||
LabelValue,
|
LabelValue,
|
||||||
HarvesterKsmtuned,
|
HarvesterKsmtuned,
|
||||||
|
HarvesterHugepages,
|
||||||
Loading,
|
Loading,
|
||||||
SortableTable,
|
SortableTable,
|
||||||
HarvesterSeeder,
|
HarvesterSeeder,
|
||||||
@ -209,6 +211,12 @@ export default {
|
|||||||
return !!this.$store.getters[`${ inStore }/schemaFor`](HCI.KSTUNED);
|
return !!this.$store.getters[`${ inStore }/schemaFor`](HCI.KSTUNED);
|
||||||
},
|
},
|
||||||
|
|
||||||
|
hasHugepagesSchema() {
|
||||||
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
|
|
||||||
|
return !!this.$store.getters[`${ inStore }/schemaFor`](HCI.HUGEPAGES);
|
||||||
|
},
|
||||||
|
|
||||||
hasBlockDevicesSchema() {
|
hasBlockDevicesSchema() {
|
||||||
return !!this.$store.getters['harvester/schemaFor'](HCI.BLOCK_DEVICE);
|
return !!this.$store.getters['harvester/schemaFor'](HCI.BLOCK_DEVICE);
|
||||||
},
|
},
|
||||||
@ -468,6 +476,16 @@ export default {
|
|||||||
/>
|
/>
|
||||||
</Tab>
|
</Tab>
|
||||||
|
|
||||||
|
<Tab
|
||||||
|
v-if="hasHugepagesSchema"
|
||||||
|
name="hugepages"
|
||||||
|
:weight="0"
|
||||||
|
:show-header="false"
|
||||||
|
:label="t('harvester.host.tabs.hugepages')"
|
||||||
|
>
|
||||||
|
<HarvesterHugepages :node="value" />
|
||||||
|
</Tab>
|
||||||
|
|
||||||
<Tab
|
<Tab
|
||||||
v-if="seederEnabled"
|
v-if="seederEnabled"
|
||||||
name="seeder"
|
name="seeder"
|
||||||
|
|||||||
@ -75,7 +75,7 @@ export default {
|
|||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="col span-6 mb-20">
|
<div class="col span-6 mb-20">
|
||||||
<LabelValue
|
<LabelValue
|
||||||
:name="t('harvester.schedule.cron')"
|
:name="t('harvester.schedule.cron.label')"
|
||||||
:value="cronExpression"
|
:value="cronExpression"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
1446
pkg/harvester/detail/kubeovn.io.vpc.vue
Normal file
1446
pkg/harvester/detail/kubeovn.io.vpc.vue
Normal file
File diff suppressed because it is too large
Load Diff
@ -213,6 +213,7 @@ export default {
|
|||||||
const diskRows = this.getDiskRows(neu);
|
const diskRows = this.getDiskRows(neu);
|
||||||
|
|
||||||
this['diskRows'] = diskRows;
|
this['diskRows'] = diskRows;
|
||||||
|
this['networkRows'] = this.getNetworkRows(neu, { fromTemplate: false, init: false });
|
||||||
},
|
},
|
||||||
deep: true
|
deep: true
|
||||||
}
|
}
|
||||||
@ -265,6 +266,7 @@ export default {
|
|||||||
<Network
|
<Network
|
||||||
v-model:value="networkRows"
|
v-model:value="networkRows"
|
||||||
mode="view"
|
mode="view"
|
||||||
|
:vm="value"
|
||||||
/>
|
/>
|
||||||
</Tab>
|
</Tab>
|
||||||
|
|
||||||
|
|||||||
@ -32,7 +32,7 @@ export default {
|
|||||||
resources: {
|
resources: {
|
||||||
type: Array,
|
type: Array,
|
||||||
required: true
|
required: true
|
||||||
}
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
data() {
|
data() {
|
||||||
@ -43,7 +43,7 @@ export default {
|
|||||||
...mapState('action-menu', ['modalData']),
|
...mapState('action-menu', ['modalData']),
|
||||||
|
|
||||||
title() {
|
title() {
|
||||||
return this.modalData.title || 'dialog.promptRemove.title';
|
return this.modalData?.title || 'dialog.promptRemove.title';
|
||||||
},
|
},
|
||||||
|
|
||||||
formattedType() {
|
formattedType() {
|
||||||
@ -51,7 +51,7 @@ export default {
|
|||||||
},
|
},
|
||||||
|
|
||||||
warningMessage() {
|
warningMessage() {
|
||||||
if (this.modalData.warningMessage) return this.modalData.warningMessage;
|
if (this.modalData?.warningMessage) return this.modalData.warningMessage;
|
||||||
|
|
||||||
const isPlural = this.type.endsWith('s');
|
const isPlural = this.type.endsWith('s');
|
||||||
const thisOrThese = isPlural ? 'these' : 'this';
|
const thisOrThese = isPlural ? 'these' : 'this';
|
||||||
@ -145,6 +145,7 @@ export default {
|
|||||||
try {
|
try {
|
||||||
for (const resource of this.resources) {
|
for (const resource of this.resources) {
|
||||||
await resource.remove();
|
await resource.remove();
|
||||||
|
if (this.modalData?.extraActionAfterRemove) await this.modalData.extraActionAfterRemove();
|
||||||
}
|
}
|
||||||
buttonDone(true);
|
buttonDone(true);
|
||||||
this.close();
|
this.close();
|
||||||
|
|||||||
212
pkg/harvester/dialog/HarvesterAddHotplugNic.vue
Normal file
212
pkg/harvester/dialog/HarvesterAddHotplugNic.vue
Normal file
@ -0,0 +1,212 @@
|
|||||||
|
<script>
|
||||||
|
import { exceptionToErrorsArray } from '@shell/utils/error';
|
||||||
|
import { mapGetters } from 'vuex';
|
||||||
|
import { NETWORK_ATTACHMENT } from '@shell/config/types';
|
||||||
|
import { Card } from '@components/Card';
|
||||||
|
import { Banner } from '@components/Banner';
|
||||||
|
import AsyncButton from '@shell/components/AsyncButton';
|
||||||
|
import { LabeledInput } from '@components/Form/LabeledInput';
|
||||||
|
import LabeledSelect from '@shell/components/form/LabeledSelect';
|
||||||
|
import { HCI as HCI_ANNOTATIONS } from '@pkg/harvester/config/labels-annotations';
|
||||||
|
import { NETWORK_TYPE } from '../config/types';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'AddHotplugNic',
|
||||||
|
|
||||||
|
emits: ['close'],
|
||||||
|
|
||||||
|
components: {
|
||||||
|
AsyncButton,
|
||||||
|
Card,
|
||||||
|
LabeledInput,
|
||||||
|
LabeledSelect,
|
||||||
|
Banner
|
||||||
|
},
|
||||||
|
|
||||||
|
props: {
|
||||||
|
resources: {
|
||||||
|
type: Array,
|
||||||
|
required: true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
async fetch() {
|
||||||
|
try {
|
||||||
|
this.allVMNetworks = await this.$store.dispatch('harvester/findAll', { type: NETWORK_ATTACHMENT });
|
||||||
|
} catch (err) {
|
||||||
|
this.errors = exceptionToErrorsArray(err);
|
||||||
|
this.allVMNetworks = [];
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
interfaceName: '',
|
||||||
|
networkName: '',
|
||||||
|
macAddress: '',
|
||||||
|
allVMNetworks: [],
|
||||||
|
errors: [],
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
computed: {
|
||||||
|
...mapGetters({ t: 'i18n/t' }),
|
||||||
|
|
||||||
|
actionResource() {
|
||||||
|
return this.resources?.[0];
|
||||||
|
},
|
||||||
|
|
||||||
|
isFormValid() {
|
||||||
|
return this.interfaceName !== '' && this.networkName !== '';
|
||||||
|
},
|
||||||
|
|
||||||
|
vmNetworksOption() {
|
||||||
|
return this.allVMNetworks
|
||||||
|
.filter((network) => {
|
||||||
|
const labels = network.metadata?.labels || {};
|
||||||
|
const type = labels[HCI_ANNOTATIONS.NETWORK_TYPE];
|
||||||
|
|
||||||
|
const isValidType = [
|
||||||
|
NETWORK_TYPE.L2VLAN,
|
||||||
|
NETWORK_TYPE.UNTAGGED,
|
||||||
|
NETWORK_TYPE.L2TRUNK_VLAN,
|
||||||
|
].includes(type);
|
||||||
|
|
||||||
|
return isValidType && !network.isSystem;
|
||||||
|
})
|
||||||
|
.map((network) => {
|
||||||
|
const label = network.isNotReady ? `${ network.id } (${ this.t('generic.notReady') })` : network.id;
|
||||||
|
|
||||||
|
return ({
|
||||||
|
label,
|
||||||
|
value: network.id || '',
|
||||||
|
disabled: network.isNotReady,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
methods: {
|
||||||
|
close() {
|
||||||
|
this.interfaceName = '';
|
||||||
|
this.networkName = '';
|
||||||
|
this.macAddress = '';
|
||||||
|
this.errors = [];
|
||||||
|
this.$emit('close');
|
||||||
|
},
|
||||||
|
|
||||||
|
async save(buttonCb) {
|
||||||
|
if (!this.actionResource) {
|
||||||
|
buttonCb(false);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const payload = {
|
||||||
|
interfaceName: this.interfaceName,
|
||||||
|
networkName: this.networkName
|
||||||
|
};
|
||||||
|
|
||||||
|
if (this.macAddress) {
|
||||||
|
payload.macAddress = this.macAddress;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const res = await this.actionResource.doAction('addNic', payload);
|
||||||
|
|
||||||
|
if ([200, 204].includes(res?._status)) {
|
||||||
|
this.$store.dispatch('growl/success', {
|
||||||
|
title: this.t('generic.notification.title.succeed'),
|
||||||
|
message: this.t('harvester.modal.hotplugNic.success', {
|
||||||
|
interfaceName: this.interfaceName,
|
||||||
|
vm: this.actionResource.nameDisplay
|
||||||
|
})
|
||||||
|
}, { root: true });
|
||||||
|
|
||||||
|
this.close();
|
||||||
|
buttonCb(true);
|
||||||
|
} else {
|
||||||
|
this.errors = exceptionToErrorsArray(res);
|
||||||
|
buttonCb(false);
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
this.errors = exceptionToErrorsArray(err);
|
||||||
|
buttonCb(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<Card
|
||||||
|
ref="modal"
|
||||||
|
name="modal"
|
||||||
|
:show-highlight-border="false"
|
||||||
|
>
|
||||||
|
<template #title>
|
||||||
|
<h4
|
||||||
|
v-clean-html="t('harvester.modal.hotplugNic.title')"
|
||||||
|
class="text-default-text"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<template #body>
|
||||||
|
<LabeledInput
|
||||||
|
v-model:value="interfaceName"
|
||||||
|
:label="t('generic.name')"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
<LabeledSelect
|
||||||
|
v-model:value="networkName"
|
||||||
|
class="mt-20"
|
||||||
|
:label="t('harvester.modal.hotplugNic.vmNetwork')"
|
||||||
|
:options="vmNetworksOption"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
<LabeledInput
|
||||||
|
v-model:value="macAddress"
|
||||||
|
class="mt-20"
|
||||||
|
label-key="harvester.modal.hotplugNic.macAddress"
|
||||||
|
:tooltip="t('harvester.modal.hotplugNic.macAddressTooltip', _, true)"
|
||||||
|
/>
|
||||||
|
<Banner
|
||||||
|
v-for="(err, i) in errors"
|
||||||
|
:key="i"
|
||||||
|
:label="err"
|
||||||
|
color="error"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<template #actions>
|
||||||
|
<div class="actions">
|
||||||
|
<div class="buttons">
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="btn role-secondary mr-10"
|
||||||
|
@click="close"
|
||||||
|
>
|
||||||
|
{{ t('generic.cancel') }}
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<AsyncButton
|
||||||
|
mode="apply"
|
||||||
|
:disabled="!isFormValid"
|
||||||
|
@click="save"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</Card>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.actions {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
.buttons {
|
||||||
|
display: flex;
|
||||||
|
justify-content: flex-end;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@ -11,7 +11,7 @@ import { LabeledInput } from '@components/Form/LabeledInput';
|
|||||||
import LabeledSelect from '@shell/components/form/LabeledSelect';
|
import LabeledSelect from '@shell/components/form/LabeledSelect';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: 'HotplugModal',
|
name: 'HotplugVolumeModal',
|
||||||
|
|
||||||
emits: ['close'],
|
emits: ['close'],
|
||||||
|
|
||||||
@ -62,7 +62,7 @@ export default {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return !pvc.attachVM;
|
return true;
|
||||||
})
|
})
|
||||||
.map((pvc) => {
|
.map((pvc) => {
|
||||||
return {
|
return {
|
||||||
@ -90,7 +90,7 @@ export default {
|
|||||||
if (res._status === 200 || res._status === 204) {
|
if (res._status === 200 || res._status === 204) {
|
||||||
this.$store.dispatch('growl/success', {
|
this.$store.dispatch('growl/success', {
|
||||||
title: this.t('generic.notification.title.succeed'),
|
title: this.t('generic.notification.title.succeed'),
|
||||||
message: this.t('harvester.modal.hotplug.success', { diskName: this.diskName, vm: this.actionResource.nameDisplay })
|
message: this.t('harvester.modal.hotplugVolume.success', { diskName: this.diskName, vm: this.actionResource.nameDisplay })
|
||||||
}, { root: true });
|
}, { root: true });
|
||||||
|
|
||||||
this.close();
|
this.close();
|
||||||
@ -122,7 +122,7 @@ export default {
|
|||||||
>
|
>
|
||||||
<template #title>
|
<template #title>
|
||||||
<h4
|
<h4
|
||||||
v-clean-html="t('harvester.modal.hotplug.title')"
|
v-clean-html="t('harvester.modal.hotplugVolume.title')"
|
||||||
class="text-default-text"
|
class="text-default-text"
|
||||||
/>
|
/>
|
||||||
</template>
|
</template>
|
||||||
@ -1,13 +1,16 @@
|
|||||||
<script>
|
<script>
|
||||||
import { mapState, mapGetters } from 'vuex';
|
import { mapState, mapGetters } from 'vuex';
|
||||||
|
|
||||||
import { exceptionToErrorsArray } from '@shell/utils/error';
|
import { exceptionToErrorsArray } from '@shell/utils/error';
|
||||||
import { Card } from '@components/Card';
|
import { Card } from '@components/Card';
|
||||||
import { Banner } from '@components/Banner';
|
import { Banner } from '@components/Banner';
|
||||||
import AsyncButton from '@shell/components/AsyncButton';
|
import AsyncButton from '@shell/components/AsyncButton';
|
||||||
|
|
||||||
|
const VOLUME = 'volume';
|
||||||
|
const NETWORK = 'network';
|
||||||
|
const CDROM = 'cdrom';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: 'HarvesterHotUnplugModal',
|
name: 'HarvesterHotUnplug',
|
||||||
|
|
||||||
emits: ['close'],
|
emits: ['close'],
|
||||||
|
|
||||||
@ -35,8 +38,43 @@ export default {
|
|||||||
actionResource() {
|
actionResource() {
|
||||||
return this.resources[0];
|
return this.resources[0];
|
||||||
},
|
},
|
||||||
diskName() {
|
|
||||||
return this.modalData.diskName;
|
name() {
|
||||||
|
return this.modalData.name;
|
||||||
|
},
|
||||||
|
|
||||||
|
isVolume() {
|
||||||
|
return this.modalData.type === VOLUME;
|
||||||
|
},
|
||||||
|
|
||||||
|
titleKey() {
|
||||||
|
const keys = {
|
||||||
|
[VOLUME]: 'harvester.virtualMachine.hotUnplug.detachVolume.title',
|
||||||
|
[CDROM]: 'harvester.virtualMachine.hotUnplug.ejectCdRomVolume.title',
|
||||||
|
[NETWORK]: 'harvester.virtualMachine.hotUnplug.detachNIC.title',
|
||||||
|
};
|
||||||
|
|
||||||
|
return keys[this.modalData.type];
|
||||||
|
},
|
||||||
|
|
||||||
|
actionLabelKey() {
|
||||||
|
const keys = {
|
||||||
|
[VOLUME]: 'harvester.virtualMachine.hotUnplug.detachVolume.actionLabels',
|
||||||
|
[CDROM]: 'harvester.virtualMachine.hotUnplug.ejectCdRomVolume.actionLabels',
|
||||||
|
[NETWORK]: 'harvester.virtualMachine.hotUnplug.detachNIC.actionLabels',
|
||||||
|
};
|
||||||
|
|
||||||
|
return keys[this.modalData.type];
|
||||||
|
},
|
||||||
|
|
||||||
|
successMessageKey() {
|
||||||
|
const keys = {
|
||||||
|
[VOLUME]: 'harvester.virtualMachine.hotUnplug.detachVolume.success',
|
||||||
|
[CDROM]: 'harvester.virtualMachine.hotUnplug.ejectCdRomVolume.success',
|
||||||
|
[NETWORK]: 'harvester.virtualMachine.hotUnplug.detachNIC.success',
|
||||||
|
};
|
||||||
|
|
||||||
|
return keys[this.modalData.type];
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -47,14 +85,22 @@ export default {
|
|||||||
|
|
||||||
async save(buttonCb) {
|
async save(buttonCb) {
|
||||||
try {
|
try {
|
||||||
const res = await this.actionResource.doAction('removeVolume', { diskName: this.diskName });
|
let res;
|
||||||
|
|
||||||
|
if (this.modalData.type === VOLUME) {
|
||||||
|
res = await this.actionResource.doAction('removeVolume', { diskName: this.name });
|
||||||
|
} else if (this.modalData.type === NETWORK) {
|
||||||
|
res = await this.actionResource.doAction('removeNic', { interfaceName: this.name });
|
||||||
|
} else {
|
||||||
|
res = await this.actionResource.doAction('ejectCdRomVolume', { deviceName: this.name });
|
||||||
|
}
|
||||||
|
|
||||||
if (res._status === 200 || res._status === 204) {
|
if (res._status === 200 || res._status === 204) {
|
||||||
this.$store.dispatch(
|
this.$store.dispatch(
|
||||||
'growl/success',
|
'growl/success',
|
||||||
{
|
{
|
||||||
title: this.t('generic.notification.title.succeed'),
|
title: this.t('generic.notification.title.succeed'),
|
||||||
message: this.t('harvester.modal.hotunplug.success', { name: this.diskName })
|
message: this.t(this.successMessageKey, { name: this.name })
|
||||||
},
|
},
|
||||||
{ root: true }
|
{ root: true }
|
||||||
);
|
);
|
||||||
@ -64,14 +110,14 @@ export default {
|
|||||||
} else {
|
} else {
|
||||||
const error = [res?.data] || exceptionToErrorsArray(res);
|
const error = [res?.data] || exceptionToErrorsArray(res);
|
||||||
|
|
||||||
this['errors'] = error;
|
this.errors = error;
|
||||||
buttonCb(false);
|
buttonCb(false);
|
||||||
}
|
}
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
const error = err?.data || err;
|
const error = err?.data || err;
|
||||||
const message = exceptionToErrorsArray(error);
|
const message = exceptionToErrorsArray(error);
|
||||||
|
|
||||||
this['errors'] = message;
|
this.errors = message;
|
||||||
buttonCb(false);
|
buttonCb(false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -87,7 +133,7 @@ export default {
|
|||||||
>
|
>
|
||||||
<template #title>
|
<template #title>
|
||||||
<h4
|
<h4
|
||||||
v-clean-html="t('harvester.virtualMachine.unplug.title', { name: diskName })"
|
v-clean-html="t(titleKey, { name })"
|
||||||
class="text-default-text"
|
class="text-default-text"
|
||||||
/>
|
/>
|
||||||
<Banner
|
<Banner
|
||||||
@ -111,9 +157,9 @@ export default {
|
|||||||
|
|
||||||
<AsyncButton
|
<AsyncButton
|
||||||
mode="apply"
|
mode="apply"
|
||||||
:action-label="t('harvester.virtualMachine.unplug.actionLabel')"
|
:action-label="t(actionLabelKey)"
|
||||||
:waiting-label="t('harvester.virtualMachine.unplug.actionLabel')"
|
:waiting-label="t(actionLabelKey)"
|
||||||
:success-label="t('harvester.virtualMachine.unplug.actionLabel')"
|
:success-label="t(actionLabelKey)"
|
||||||
@click="save"
|
@click="save"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
@ -132,4 +178,8 @@ export default {
|
|||||||
justify-content: flex-end;
|
justify-content: flex-end;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
::v-deep(.card-title) {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
198
pkg/harvester/dialog/HarvesterInsertCdRomVolume.vue
Normal file
198
pkg/harvester/dialog/HarvesterInsertCdRomVolume.vue
Normal file
@ -0,0 +1,198 @@
|
|||||||
|
<script>
|
||||||
|
import { exceptionToErrorsArray } from '@shell/utils/error';
|
||||||
|
import { mapState, mapGetters } from 'vuex';
|
||||||
|
import { Card } from '@components/Card';
|
||||||
|
import { Banner } from '@components/Banner';
|
||||||
|
import AsyncButton from '@shell/components/AsyncButton';
|
||||||
|
import { LabeledInput } from '@components/Form/LabeledInput';
|
||||||
|
import LabeledSelect from '@shell/components/form/LabeledSelect';
|
||||||
|
import { HCI as HCI_ANNOTATIONS } from '@pkg/harvester/config/labels-annotations';
|
||||||
|
import { HCI } from '../types';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'HarvesterInsertCdRomVolume',
|
||||||
|
|
||||||
|
emits: ['close'],
|
||||||
|
|
||||||
|
components: {
|
||||||
|
AsyncButton,
|
||||||
|
Card,
|
||||||
|
LabeledInput,
|
||||||
|
LabeledSelect,
|
||||||
|
Banner
|
||||||
|
},
|
||||||
|
|
||||||
|
props: {
|
||||||
|
resources: {
|
||||||
|
type: Array,
|
||||||
|
required: true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
async fetch() {
|
||||||
|
try {
|
||||||
|
this.images = await this.$store.dispatch('harvester/findAll', { type: HCI.IMAGE });
|
||||||
|
} catch (err) {
|
||||||
|
this.errors = exceptionToErrorsArray(err);
|
||||||
|
this.images = [];
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
imageName: '',
|
||||||
|
images: [],
|
||||||
|
errors: [],
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
computed: {
|
||||||
|
...mapState('action-menu', ['modalData']),
|
||||||
|
...mapGetters({ t: 'i18n/t' }),
|
||||||
|
|
||||||
|
actionResource() {
|
||||||
|
return this.resources?.[0];
|
||||||
|
},
|
||||||
|
|
||||||
|
isFormValid() {
|
||||||
|
return this.imageName !== '';
|
||||||
|
},
|
||||||
|
|
||||||
|
deviceName() {
|
||||||
|
return this.modalData.name;
|
||||||
|
},
|
||||||
|
|
||||||
|
imagesOption() {
|
||||||
|
return this.images
|
||||||
|
.filter((image) => {
|
||||||
|
const labels = image.metadata?.labels || {};
|
||||||
|
const type = labels[HCI_ANNOTATIONS.IMAGE_SUFFIX];
|
||||||
|
|
||||||
|
return type === 'iso';
|
||||||
|
})
|
||||||
|
.map((image) => {
|
||||||
|
return ({
|
||||||
|
label: this.imageOptionLabel(image),
|
||||||
|
value: image.id,
|
||||||
|
disabled: image.isImportedImage
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
methods: {
|
||||||
|
close() {
|
||||||
|
this.imageName = '';
|
||||||
|
this.errors = [];
|
||||||
|
this.$emit('close');
|
||||||
|
},
|
||||||
|
|
||||||
|
imageOptionLabel(image) {
|
||||||
|
return `${ image.metadata.namespace }/${ image.spec.displayName }`;
|
||||||
|
},
|
||||||
|
|
||||||
|
async save(buttonCb) {
|
||||||
|
if (!this.actionResource) {
|
||||||
|
buttonCb(false);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const payload = {
|
||||||
|
deviceName: this.deviceName,
|
||||||
|
imageName: this.imageName
|
||||||
|
};
|
||||||
|
|
||||||
|
try {
|
||||||
|
const res = await this.actionResource.doAction('insertCdRomVolume', payload);
|
||||||
|
|
||||||
|
if ([200, 204].includes(res?._status)) {
|
||||||
|
this.$store.dispatch('growl/success', {
|
||||||
|
title: this.t('generic.notification.title.succeed'),
|
||||||
|
message: this.t('harvester.modal.insertCdRomVolume.success', {
|
||||||
|
deviceName: this.deviceName,
|
||||||
|
imageName: this.imageName,
|
||||||
|
})
|
||||||
|
}, { root: true });
|
||||||
|
|
||||||
|
this.close();
|
||||||
|
buttonCb(true);
|
||||||
|
} else {
|
||||||
|
this.errors = exceptionToErrorsArray(res);
|
||||||
|
buttonCb(false);
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
this.errors = exceptionToErrorsArray(err);
|
||||||
|
buttonCb(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<Card
|
||||||
|
ref="modal"
|
||||||
|
name="modal"
|
||||||
|
:show-highlight-border="false"
|
||||||
|
>
|
||||||
|
<template #title>
|
||||||
|
<h4
|
||||||
|
v-clean-html="t('harvester.modal.insertCdRomVolume.title')"
|
||||||
|
class="text-default-text"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<template #body>
|
||||||
|
<LabeledInput
|
||||||
|
v-model:value="deviceName"
|
||||||
|
:label="t('generic.name')"
|
||||||
|
disabled
|
||||||
|
/>
|
||||||
|
<LabeledSelect
|
||||||
|
v-model:value="imageName"
|
||||||
|
class="mt-20"
|
||||||
|
:label="t('harvester.modal.insertCdRomVolume.image')"
|
||||||
|
:options="imagesOption"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
<Banner
|
||||||
|
v-for="(err, i) in errors"
|
||||||
|
:key="i"
|
||||||
|
:label="err"
|
||||||
|
color="error"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<template #actions>
|
||||||
|
<div class="actions">
|
||||||
|
<div class="buttons">
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="btn role-secondary mr-10"
|
||||||
|
@click="close"
|
||||||
|
>
|
||||||
|
{{ t('generic.cancel') }}
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<AsyncButton
|
||||||
|
mode="apply"
|
||||||
|
:disabled="!isFormValid"
|
||||||
|
@click="save"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</Card>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.actions {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
.buttons {
|
||||||
|
display: flex;
|
||||||
|
justify-content: flex-end;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
239
pkg/harvester/dialog/HarvesterStorageMigrationDialog.vue
Normal file
239
pkg/harvester/dialog/HarvesterStorageMigrationDialog.vue
Normal file
@ -0,0 +1,239 @@
|
|||||||
|
<script>
|
||||||
|
import { mapGetters } from 'vuex';
|
||||||
|
|
||||||
|
import { PVC } from '@shell/config/types';
|
||||||
|
import { exceptionToErrorsArray } from '@shell/utils/error';
|
||||||
|
import { sortBy } from '@shell/utils/sort';
|
||||||
|
import { HCI } from '../types';
|
||||||
|
import { parseVolumeClaimTemplates } from '@pkg/harvester/utils/vm';
|
||||||
|
|
||||||
|
import { Card } from '@components/Card';
|
||||||
|
import { Banner } from '@components/Banner';
|
||||||
|
import AsyncButton from '@shell/components/AsyncButton';
|
||||||
|
import LabeledSelect from '@shell/components/form/LabeledSelect';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'HarvesterStorageMigrationDialog',
|
||||||
|
|
||||||
|
emits: ['close'],
|
||||||
|
|
||||||
|
components: {
|
||||||
|
AsyncButton, Banner, Card, LabeledSelect
|
||||||
|
},
|
||||||
|
|
||||||
|
props: {
|
||||||
|
resources: {
|
||||||
|
type: Array,
|
||||||
|
required: true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
async fetch() {
|
||||||
|
this.allPVCs = await this.$store.dispatch('harvester/findAll', { type: PVC });
|
||||||
|
},
|
||||||
|
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
sourceVolume: '',
|
||||||
|
targetVolume: '',
|
||||||
|
errors: [],
|
||||||
|
allPVCs: [],
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
computed: {
|
||||||
|
...mapGetters({ t: 'i18n/t' }),
|
||||||
|
|
||||||
|
actionResource() {
|
||||||
|
return this.resources[0];
|
||||||
|
},
|
||||||
|
|
||||||
|
sourceVolumeOptions() {
|
||||||
|
const volumes = this.actionResource.spec?.template?.spec?.volumes || [];
|
||||||
|
|
||||||
|
return sortBy(
|
||||||
|
volumes
|
||||||
|
.map((v) => v.persistentVolumeClaim?.claimName)
|
||||||
|
.filter((name) => !!name)
|
||||||
|
.map((name) => ({
|
||||||
|
label: name,
|
||||||
|
value: name
|
||||||
|
})),
|
||||||
|
'label'
|
||||||
|
);
|
||||||
|
},
|
||||||
|
|
||||||
|
namespacePVCs() {
|
||||||
|
return this.allPVCs.filter((pvc) => pvc.metadata.namespace === this.actionResource.metadata.namespace);
|
||||||
|
},
|
||||||
|
|
||||||
|
vmUsedVolumeNames() {
|
||||||
|
const allVMs = this.$store.getters['harvester/all'](HCI.VM) || [];
|
||||||
|
const names = new Set();
|
||||||
|
|
||||||
|
allVMs.forEach((vm) => {
|
||||||
|
// Collect volume names from spec.template.spec.volumes (both PVC and DataVolume references)
|
||||||
|
const volumes = vm.spec?.template?.spec?.volumes || [];
|
||||||
|
|
||||||
|
volumes.forEach((v) => {
|
||||||
|
const name = v.persistentVolumeClaim?.claimName || v.dataVolume?.name;
|
||||||
|
|
||||||
|
if (name) {
|
||||||
|
names.add(`${ vm.metadata.namespace }/${ name }`);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Collect volume names from volumeClaimTemplates annotation
|
||||||
|
const templates = parseVolumeClaimTemplates(vm);
|
||||||
|
|
||||||
|
templates.forEach((t) => {
|
||||||
|
if (t.metadata?.name) {
|
||||||
|
names.add(`${ vm.metadata.namespace }/${ t.metadata.name }`);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
return names;
|
||||||
|
},
|
||||||
|
|
||||||
|
targetVolumeOptions() {
|
||||||
|
return sortBy(
|
||||||
|
this.namespacePVCs
|
||||||
|
.filter((pvc) => {
|
||||||
|
// Exclude volumes used by any VM (via spec.volumes or volumeClaimTemplates)
|
||||||
|
if (this.vmUsedVolumeNames.has(`${ pvc.metadata.namespace }/${ pvc.metadata.name }`)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
})
|
||||||
|
.map((pvc) => ({
|
||||||
|
label: pvc.metadata.name,
|
||||||
|
value: pvc.metadata.name
|
||||||
|
})),
|
||||||
|
'label'
|
||||||
|
);
|
||||||
|
},
|
||||||
|
|
||||||
|
disableSave() {
|
||||||
|
return !this.sourceVolume || !this.targetVolume;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
methods: {
|
||||||
|
close() {
|
||||||
|
this.sourceVolume = '';
|
||||||
|
this.targetVolume = '';
|
||||||
|
this.errors = [];
|
||||||
|
this.$emit('close');
|
||||||
|
},
|
||||||
|
|
||||||
|
async apply(buttonDone) {
|
||||||
|
if (!this.actionResource) {
|
||||||
|
buttonDone(false);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!this.sourceVolume) {
|
||||||
|
const name = this.t('harvester.modal.storageMigration.fields.sourceVolume.label');
|
||||||
|
|
||||||
|
this['errors'] = [this.t('validation.required', { key: name })];
|
||||||
|
buttonDone(false);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!this.targetVolume) {
|
||||||
|
const name = this.t('harvester.modal.storageMigration.fields.targetVolume.label');
|
||||||
|
|
||||||
|
this['errors'] = [this.t('validation.required', { key: name })];
|
||||||
|
buttonDone(false);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await this.actionResource.doAction('storageMigration', {
|
||||||
|
sourceVolume: this.sourceVolume,
|
||||||
|
targetVolume: this.targetVolume
|
||||||
|
}, {}, false);
|
||||||
|
|
||||||
|
buttonDone(true);
|
||||||
|
this.close();
|
||||||
|
} catch (err) {
|
||||||
|
const error = err?.data || err;
|
||||||
|
|
||||||
|
this['errors'] = exceptionToErrorsArray(error);
|
||||||
|
buttonDone(false);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<Card :show-highlight-border="false">
|
||||||
|
<template #title>
|
||||||
|
{{ t('harvester.modal.storageMigration.title') }}
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<template #body>
|
||||||
|
<LabeledSelect
|
||||||
|
v-model:value="sourceVolume"
|
||||||
|
:label="t('harvester.modal.storageMigration.fields.sourceVolume.label')"
|
||||||
|
:placeholder="t('harvester.modal.storageMigration.fields.sourceVolume.placeholder')"
|
||||||
|
:options="sourceVolumeOptions"
|
||||||
|
class="mb-20"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
|
||||||
|
<LabeledSelect
|
||||||
|
v-model:value="targetVolume"
|
||||||
|
:label="t('harvester.modal.storageMigration.fields.targetVolume.label')"
|
||||||
|
:placeholder="t('harvester.modal.storageMigration.fields.targetVolume.placeholder')"
|
||||||
|
:options="targetVolumeOptions"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Banner
|
||||||
|
v-for="(err, i) in errors"
|
||||||
|
:key="i"
|
||||||
|
color="error"
|
||||||
|
:label="err"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<template
|
||||||
|
#actions
|
||||||
|
class="actions"
|
||||||
|
>
|
||||||
|
<div class="buttons">
|
||||||
|
<button
|
||||||
|
class="btn role-secondary mr-10"
|
||||||
|
@click="close"
|
||||||
|
>
|
||||||
|
{{ t('generic.cancel') }}
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<AsyncButton
|
||||||
|
mode="apply"
|
||||||
|
:disabled="disableSave"
|
||||||
|
@click="apply"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</Card>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.actions {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.buttons {
|
||||||
|
display: flex;
|
||||||
|
justify-content: flex-end;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@ -277,8 +277,9 @@ export default {
|
|||||||
:label="t('harvester.modal.bundle.namespaces.label')"
|
:label="t('harvester.modal.bundle.namespaces.label')"
|
||||||
:clearable="true"
|
:clearable="true"
|
||||||
:multiple="true"
|
:multiple="true"
|
||||||
|
:append-to-body="false"
|
||||||
:options="namespaceOptions"
|
:options="namespaceOptions"
|
||||||
class="mb-10 label-select"
|
class="mb-10 namespace-select"
|
||||||
:tooltip="t('harvester.modal.bundle.namespaces.tooltip', _, true)"
|
:tooltip="t('harvester.modal.bundle.namespaces.tooltip', _, true)"
|
||||||
@update:value="updateNamespaces"
|
@update:value="updateNamespaces"
|
||||||
/>
|
/>
|
||||||
@ -372,6 +373,11 @@ export default {
|
|||||||
padding: 10px 0;
|
padding: 10px 0;
|
||||||
height: 160px;
|
height: 160px;
|
||||||
}
|
}
|
||||||
|
.namespace-select {
|
||||||
|
:deep(.vs__dropdown-menu) {
|
||||||
|
max-height: 210px;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
div {
|
div {
|
||||||
|
|||||||
157
pkg/harvester/edit/harvesterhci.io.host/HarvesterHugepages.vue
Normal file
157
pkg/harvester/edit/harvesterhci.io.host/HarvesterHugepages.vue
Normal file
@ -0,0 +1,157 @@
|
|||||||
|
<script>
|
||||||
|
import LabeledSelect from '@shell/components/form/LabeledSelect.vue';
|
||||||
|
import { HCI } from '../../types';
|
||||||
|
import { DOC } from '../../config/doc-links';
|
||||||
|
|
||||||
|
export const hugepagesTHPEnabledMode = [{
|
||||||
|
label: 'Always',
|
||||||
|
value: 'always',
|
||||||
|
}, {
|
||||||
|
label: 'Madvise',
|
||||||
|
value: 'madvise',
|
||||||
|
}, {
|
||||||
|
label: 'Never',
|
||||||
|
value: 'never',
|
||||||
|
}];
|
||||||
|
|
||||||
|
export const hugepagesTHPShmemEnabledMode = [{
|
||||||
|
label: 'Always',
|
||||||
|
value: 'always',
|
||||||
|
}, {
|
||||||
|
label: 'Within Size',
|
||||||
|
value: 'within_size',
|
||||||
|
}, {
|
||||||
|
label: 'Advise',
|
||||||
|
value: 'advise',
|
||||||
|
}, {
|
||||||
|
label: 'Never',
|
||||||
|
value: 'never',
|
||||||
|
}, {
|
||||||
|
label: 'Deny',
|
||||||
|
value: 'deny',
|
||||||
|
}, {
|
||||||
|
label: 'Force',
|
||||||
|
value: 'force',
|
||||||
|
}];
|
||||||
|
|
||||||
|
export const hugepagesTHPDefragMode = [{
|
||||||
|
label: 'Always',
|
||||||
|
value: 'always',
|
||||||
|
}, {
|
||||||
|
label: 'Defer',
|
||||||
|
value: 'defer',
|
||||||
|
}, {
|
||||||
|
label: 'Defer+Madvise',
|
||||||
|
value: 'defer+madvise',
|
||||||
|
}, {
|
||||||
|
label: 'Madvise',
|
||||||
|
value: 'madvise',
|
||||||
|
}, {
|
||||||
|
label: 'Never',
|
||||||
|
value: 'never'
|
||||||
|
}];
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'HarvesterHugepages',
|
||||||
|
components: { LabeledSelect },
|
||||||
|
|
||||||
|
props: {
|
||||||
|
node: {
|
||||||
|
type: Object,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
|
||||||
|
registerBeforeHook: {
|
||||||
|
type: Function,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
computed: {
|
||||||
|
docsTransparentHugepagesLink() {
|
||||||
|
return DOC.TRANSPARENT_HUGEPAGES;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
methods: {
|
||||||
|
async saveHugepages() {
|
||||||
|
this.hugepages['spec'] = this.spec;
|
||||||
|
|
||||||
|
await this.hugepages.save().catch((reason) => {
|
||||||
|
if (reason?.type === 'error') {
|
||||||
|
this.$store.dispatch('growl/error', {
|
||||||
|
title: this.t('harvester.notification.title.error'),
|
||||||
|
message: reason?.message,
|
||||||
|
}, { root: true });
|
||||||
|
|
||||||
|
return Promise.reject(new Error('saveHugepages error'));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
async fetch() {
|
||||||
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
|
|
||||||
|
const hash = await this.$store.dispatch(`${ inStore }/findAll`, { type: HCI.HUGEPAGES });
|
||||||
|
|
||||||
|
this.hugepages = hash.find((node) => {
|
||||||
|
return node.id === this.node.id;
|
||||||
|
});
|
||||||
|
this.spec = this.hugepages.spec;
|
||||||
|
},
|
||||||
|
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
hugepages: {},
|
||||||
|
spec: { transparent: {} },
|
||||||
|
hugepagesTHPEnabledMode,
|
||||||
|
hugepagesTHPShmemEnabledMode,
|
||||||
|
hugepagesTHPDefragMode,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
created() {
|
||||||
|
this.registerBeforeHook(this.saveHugepages, 'saveHugepages');
|
||||||
|
},
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<div>
|
||||||
|
<hr class="divider" />
|
||||||
|
<h3>
|
||||||
|
<t
|
||||||
|
k="harvester.host.hugepages.transparent.title"
|
||||||
|
:raw="true"
|
||||||
|
:url="docsTransparentHugepagesLink"
|
||||||
|
/>
|
||||||
|
</h3>
|
||||||
|
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-4">
|
||||||
|
<LabeledSelect
|
||||||
|
v-model:value="spec.transparent.enabled"
|
||||||
|
:label="t('harvester.host.hugepages.transparent.enabled')"
|
||||||
|
:options="hugepagesTHPEnabledMode"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="col span-4">
|
||||||
|
<LabeledSelect
|
||||||
|
v-model:value="spec.transparent.shmemEnabled"
|
||||||
|
:label="t('harvester.host.hugepages.transparent.shmemEnabled')"
|
||||||
|
:options="hugepagesTHPShmemEnabledMode"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="col span-4">
|
||||||
|
<LabeledSelect
|
||||||
|
v-model:value="spec.transparent.defrag"
|
||||||
|
:label="t('harvester.host.hugepages.transparent.defrag')"
|
||||||
|
:options="hugepagesTHPDefragMode"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
@ -28,6 +28,7 @@ import { HCI } from '../../types';
|
|||||||
import HarvesterDisk from './HarvesterDisk';
|
import HarvesterDisk from './HarvesterDisk';
|
||||||
import HarvesterSeeder from './HarvesterSeeder';
|
import HarvesterSeeder from './HarvesterSeeder';
|
||||||
import HarvesterKsmtuned from './HarvesterKsmtuned';
|
import HarvesterKsmtuned from './HarvesterKsmtuned';
|
||||||
|
import HarvesterHugepages from './HarvesterHugepages';
|
||||||
import Tags from '../../components/DiskTags';
|
import Tags from '../../components/DiskTags';
|
||||||
import { LVM_DRIVER } from '../../models/harvester/storage.k8s.io.storageclass';
|
import { LVM_DRIVER } from '../../models/harvester/storage.k8s.io.storageclass';
|
||||||
import isEqual from 'lodash/isEqual';
|
import isEqual from 'lodash/isEqual';
|
||||||
@ -50,6 +51,7 @@ export default {
|
|||||||
ArrayListGrouped,
|
ArrayListGrouped,
|
||||||
HarvesterDisk,
|
HarvesterDisk,
|
||||||
HarvesterKsmtuned,
|
HarvesterKsmtuned,
|
||||||
|
HarvesterHugepages,
|
||||||
ButtonDropdown,
|
ButtonDropdown,
|
||||||
KeyValue,
|
KeyValue,
|
||||||
Banner,
|
Banner,
|
||||||
@ -225,6 +227,12 @@ export default {
|
|||||||
return !!this.$store.getters[`${ inStore }/schemaFor`](HCI.KSTUNED);
|
return !!this.$store.getters[`${ inStore }/schemaFor`](HCI.KSTUNED);
|
||||||
},
|
},
|
||||||
|
|
||||||
|
hasHugepagesSchema() {
|
||||||
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
|
|
||||||
|
return !!this.$store.getters[`${ inStore }/schemaFor`](HCI.HUGEPAGES);
|
||||||
|
},
|
||||||
|
|
||||||
hasBlockDevicesSchema() {
|
hasBlockDevicesSchema() {
|
||||||
const inStore = this.$store.getters['currentProduct'].inStore;
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
|
|
||||||
@ -647,6 +655,17 @@ export default {
|
|||||||
</template>
|
</template>
|
||||||
</ArrayListGrouped>
|
</ArrayListGrouped>
|
||||||
</Tab>
|
</Tab>
|
||||||
|
<Tab
|
||||||
|
v-if="hasHugepagesSchema"
|
||||||
|
name="Hugepages"
|
||||||
|
:weight="70"
|
||||||
|
:label="t('harvester.host.tabs.hugepages')"
|
||||||
|
>
|
||||||
|
<HarvesterHugepages
|
||||||
|
:node="value"
|
||||||
|
:register-before-hook="registerBeforeHook"
|
||||||
|
/>
|
||||||
|
</Tab>
|
||||||
<Tab
|
<Tab
|
||||||
v-if="hasKsmtunedSchema"
|
v-if="hasKsmtunedSchema"
|
||||||
name="Ksmtuned"
|
name="Ksmtuned"
|
||||||
|
|||||||
@ -1,4 +1,5 @@
|
|||||||
<script>
|
<script>
|
||||||
|
import { mapGetters } from 'vuex';
|
||||||
import { RadioGroup } from '@components/Form/Radio';
|
import { RadioGroup } from '@components/Form/Radio';
|
||||||
import { Banner } from '@components/Banner';
|
import { Banner } from '@components/Banner';
|
||||||
import { LabeledInput } from '@components/Form/LabeledInput';
|
import { LabeledInput } from '@components/Form/LabeledInput';
|
||||||
@ -16,6 +17,7 @@ import { sortBy } from '@shell/utils/sort';
|
|||||||
import { BACKUP_TYPE } from '../config/types';
|
import { BACKUP_TYPE } from '../config/types';
|
||||||
import { _EDIT, _CREATE } from '@shell/config/query-params';
|
import { _EDIT, _CREATE } from '@shell/config/query-params';
|
||||||
import { isBackupTargetSettingEmpty, isBackupTargetSettingUnavailable } from '../utils/setting';
|
import { isBackupTargetSettingEmpty, isBackupTargetSettingUnavailable } from '../utils/setting';
|
||||||
|
import CronExpressionEditorModal from '@shell/components/Cron/CronExpressionEditorModal.vue';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: 'CreateVMSchedule',
|
name: 'CreateVMSchedule',
|
||||||
@ -28,6 +30,7 @@ export default {
|
|||||||
LabeledSelect,
|
LabeledSelect,
|
||||||
MessageLink,
|
MessageLink,
|
||||||
Banner,
|
Banner,
|
||||||
|
CronExpressionEditorModal
|
||||||
},
|
},
|
||||||
|
|
||||||
mixins: [CreateEditView],
|
mixins: [CreateEditView],
|
||||||
@ -86,10 +89,12 @@ export default {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return { settings: [] };
|
return { settings: [], showModel: false };
|
||||||
},
|
},
|
||||||
|
|
||||||
computed: {
|
computed: {
|
||||||
|
...mapGetters({ t: 'i18n/t' }),
|
||||||
|
|
||||||
backupTargetResource() {
|
backupTargetResource() {
|
||||||
return this.settings.find( (O) => O.id === 'backup-target');
|
return this.settings.find( (O) => O.id === 'backup-target');
|
||||||
},
|
},
|
||||||
@ -172,6 +177,9 @@ export default {
|
|||||||
this.value.spec['maxFailure'] = this.value.spec.retain;
|
this.value.spec['maxFailure'] = this.value.spec.retain;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
openModal() {
|
||||||
|
this.showModel = true;
|
||||||
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
@ -256,16 +264,28 @@ export default {
|
|||||||
:weight="99"
|
:weight="99"
|
||||||
class="bordered-table"
|
class="bordered-table"
|
||||||
>
|
>
|
||||||
<LabeledInput
|
<div class="cronEditor">
|
||||||
v-model:value="value.spec.cron"
|
<LabeledInput
|
||||||
class="mb-30"
|
v-model:value="value.spec.cron"
|
||||||
type="cron"
|
class="mb-30"
|
||||||
required
|
type="cron"
|
||||||
:mode="mode"
|
required
|
||||||
:label="t('harvester.schedule.cron')"
|
:mode="mode"
|
||||||
placeholder="0 * * * *"
|
:label="t('harvester.schedule.cron.label')"
|
||||||
:disabled="isBackupTargetUnAvailable || isView"
|
placeholder="0 * * * *"
|
||||||
/>
|
:disabled="isBackupTargetUnAvailable || isView"
|
||||||
|
/>
|
||||||
|
<button
|
||||||
|
class="editCronBtn btn role-primary"
|
||||||
|
@click="openModal"
|
||||||
|
>
|
||||||
|
{{ t('harvester.schedule.cron.editButton') }}
|
||||||
|
</button>
|
||||||
|
<CronExpressionEditorModal
|
||||||
|
v-model:show="showModel"
|
||||||
|
v-model:cron-expression="value.spec.cron"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
<LabeledInput
|
<LabeledInput
|
||||||
v-model:value.number="value.spec.retain"
|
v-model:value.number="value.spec.retain"
|
||||||
class="mb-30"
|
class="mb-30"
|
||||||
@ -292,3 +312,16 @@ export default {
|
|||||||
</Tabbed>
|
</Tabbed>
|
||||||
</CruResource>
|
</CruResource>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.cronEditor {
|
||||||
|
align-items: center;
|
||||||
|
display: flex;
|
||||||
|
}
|
||||||
|
|
||||||
|
.editCronBtn {
|
||||||
|
margin-bottom: 30px;
|
||||||
|
margin-left: 10px;
|
||||||
|
height: 60px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|||||||
@ -6,6 +6,7 @@ import { Checkbox } from '@components/Form/Checkbox';
|
|||||||
import CruResource from '@shell/components/CruResource';
|
import CruResource from '@shell/components/CruResource';
|
||||||
import NameNsDescription from '@shell/components/form/NameNsDescription';
|
import NameNsDescription from '@shell/components/form/NameNsDescription';
|
||||||
import LabeledSelect from '@shell/components/form/LabeledSelect';
|
import LabeledSelect from '@shell/components/form/LabeledSelect';
|
||||||
|
import { set } from '@shell/utils/object';
|
||||||
import { Banner } from '@components/Banner';
|
import { Banner } from '@components/Banner';
|
||||||
import KeyValue from '@shell/components/form/KeyValue';
|
import KeyValue from '@shell/components/form/KeyValue';
|
||||||
import NodeScheduling from '@shell/components/form/NodeScheduling';
|
import NodeScheduling from '@shell/components/form/NodeScheduling';
|
||||||
@ -22,6 +23,7 @@ import Reserved from './kubevirt.io.virtualmachine/VirtualMachineReserved';
|
|||||||
import Volume from './kubevirt.io.virtualmachine/VirtualMachineVolume';
|
import Volume from './kubevirt.io.virtualmachine/VirtualMachineVolume';
|
||||||
import Network from './kubevirt.io.virtualmachine/VirtualMachineNetwork';
|
import Network from './kubevirt.io.virtualmachine/VirtualMachineNetwork';
|
||||||
import CpuMemory from './kubevirt.io.virtualmachine/VirtualMachineCpuMemory';
|
import CpuMemory from './kubevirt.io.virtualmachine/VirtualMachineCpuMemory';
|
||||||
|
import CpuModel from './kubevirt.io.virtualmachine/VirtualMachineCpuModel';
|
||||||
import CloudConfig from './kubevirt.io.virtualmachine/VirtualMachineCloudConfig';
|
import CloudConfig from './kubevirt.io.virtualmachine/VirtualMachineCloudConfig';
|
||||||
import SSHKey from './kubevirt.io.virtualmachine/VirtualMachineSSHKey';
|
import SSHKey from './kubevirt.io.virtualmachine/VirtualMachineSSHKey';
|
||||||
|
|
||||||
@ -38,6 +40,7 @@ export default {
|
|||||||
Network,
|
Network,
|
||||||
Checkbox,
|
Checkbox,
|
||||||
CpuMemory,
|
CpuMemory,
|
||||||
|
CpuModel,
|
||||||
CruResource,
|
CruResource,
|
||||||
CloudConfig,
|
CloudConfig,
|
||||||
LabeledSelect,
|
LabeledSelect,
|
||||||
@ -70,12 +73,12 @@ export default {
|
|||||||
|
|
||||||
return {
|
return {
|
||||||
templateId,
|
templateId,
|
||||||
templateValue: null,
|
templateValue: null,
|
||||||
templateSpec: null,
|
templateSpec: null,
|
||||||
versionName: '',
|
versionName: '',
|
||||||
description: '',
|
description: '',
|
||||||
defaultVersion: null,
|
defaultVersion: null,
|
||||||
isDefaultVersion: false,
|
isDefaultVersion: false,
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -154,6 +157,18 @@ export default {
|
|||||||
},
|
},
|
||||||
|
|
||||||
methods: {
|
methods: {
|
||||||
|
updateCpuModel(value) {
|
||||||
|
if (!this.spec?.template?.spec?.domain?.cpu) {
|
||||||
|
set(this.spec, 'template.spec.domain.cpu', {});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (value && value !== '') {
|
||||||
|
set(this.spec.template.spec.domain.cpu, 'model', value);
|
||||||
|
} else {
|
||||||
|
delete this.spec.template.spec.domain.cpu.model;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
async saveVMT(buttonCb) {
|
async saveVMT(buttonCb) {
|
||||||
this.parseVM();
|
this.parseVM();
|
||||||
|
|
||||||
@ -436,6 +451,17 @@ export default {
|
|||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-6">
|
||||||
|
<CpuModel
|
||||||
|
:value="spec.template.spec.domain.cpu?.model || ''"
|
||||||
|
:mode="mode"
|
||||||
|
@update:value="updateCpuModel"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="row mb-20">
|
<div class="row mb-20">
|
||||||
<a
|
<a
|
||||||
v-if="showAdvanced"
|
v-if="showAdvanced"
|
||||||
|
|||||||
@ -371,7 +371,7 @@ export default {
|
|||||||
<div class="key">
|
<div class="key">
|
||||||
<input
|
<input
|
||||||
v-model="scope.row.value"
|
v-model="scope.row.value"
|
||||||
:placeholder="t('harvester.setting.storageNetwork.exclude.placeholder')"
|
:placeholder="t('harvester.subnet.excludeIPs.placeholder')"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@ -0,0 +1,133 @@
|
|||||||
|
<script>
|
||||||
|
import YAML from 'yaml';
|
||||||
|
import LabeledSelect from '@shell/components/form/LabeledSelect';
|
||||||
|
import { CONFIG_MAP } from '@shell/config/types';
|
||||||
|
import { Banner } from '@components/Banner';
|
||||||
|
|
||||||
|
const CPU_MODEL_CONFIG_MAP_ID = 'harvester-system/node-cpu-model-configuration';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'HarvesterCpuModel',
|
||||||
|
|
||||||
|
emits: ['update:value'],
|
||||||
|
|
||||||
|
components: {
|
||||||
|
LabeledSelect,
|
||||||
|
Banner
|
||||||
|
},
|
||||||
|
|
||||||
|
props: {
|
||||||
|
value: {
|
||||||
|
type: String,
|
||||||
|
default: ''
|
||||||
|
},
|
||||||
|
mode: {
|
||||||
|
type: String,
|
||||||
|
default: 'create',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
async fetch() {
|
||||||
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
|
|
||||||
|
try {
|
||||||
|
await this.$store.dispatch(`${ inStore }/find`, { type: CONFIG_MAP, id: CPU_MODEL_CONFIG_MAP_ID });
|
||||||
|
this.fetchError = null;
|
||||||
|
} catch (e) {
|
||||||
|
this.fetchError = this.t('harvester.virtualMachine.cpuModel.fetchError', { error: e.message || e });
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
data() {
|
||||||
|
return { fetchError: null };
|
||||||
|
},
|
||||||
|
|
||||||
|
computed: {
|
||||||
|
localValue: {
|
||||||
|
get() {
|
||||||
|
return this.value ?? '';
|
||||||
|
},
|
||||||
|
set(val) {
|
||||||
|
this.$emit('update:value', val ?? '');
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
cpuModelConfigMap() {
|
||||||
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
|
|
||||||
|
return this.$store.getters[`${ inStore }/byId`](
|
||||||
|
CONFIG_MAP,
|
||||||
|
CPU_MODEL_CONFIG_MAP_ID
|
||||||
|
);
|
||||||
|
},
|
||||||
|
|
||||||
|
cpuModelOptions() {
|
||||||
|
if (!this.cpuModelConfigMap?.data?.cpuModels) {
|
||||||
|
return [{ label: this.t('generic.default'), value: '' }];
|
||||||
|
}
|
||||||
|
|
||||||
|
let cpuModelsData;
|
||||||
|
|
||||||
|
try {
|
||||||
|
cpuModelsData = YAML.parse(this.cpuModelConfigMap.data?.cpuModels || '');
|
||||||
|
} catch (e) {
|
||||||
|
return [{ label: this.t('generic.default'), value: '' }];
|
||||||
|
}
|
||||||
|
|
||||||
|
const options = [];
|
||||||
|
|
||||||
|
options.push({
|
||||||
|
label: this.t('generic.default'),
|
||||||
|
value: ''
|
||||||
|
});
|
||||||
|
|
||||||
|
// Add global models (host-model, host-passthrough)
|
||||||
|
const globalModels = cpuModelsData.globalModels || [];
|
||||||
|
|
||||||
|
globalModels.forEach((modelName) => {
|
||||||
|
options.push({
|
||||||
|
label: modelName,
|
||||||
|
value: modelName
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// Add regular models with node count
|
||||||
|
const modelEntries = Object.entries(cpuModelsData.models || {});
|
||||||
|
|
||||||
|
// Sort models alphabetically for consistent display
|
||||||
|
modelEntries.sort((a, b) => a[0].localeCompare(b[0]));
|
||||||
|
|
||||||
|
modelEntries.forEach(([modelName, modelInfo]) => {
|
||||||
|
const readyCount = modelInfo.readyCount || 0;
|
||||||
|
const label = this.t('harvester.virtualMachine.cpuModel.optionLabel', { modelName, count: readyCount });
|
||||||
|
|
||||||
|
options.push({
|
||||||
|
label,
|
||||||
|
value: modelName
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
return options;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<Banner
|
||||||
|
v-if="fetchError"
|
||||||
|
color="error"
|
||||||
|
class="mb-20"
|
||||||
|
>
|
||||||
|
{{ fetchError }}
|
||||||
|
</Banner>
|
||||||
|
<LabeledSelect
|
||||||
|
v-model:value="localValue"
|
||||||
|
:label="t('harvester.virtualMachine.cpuModel.label')"
|
||||||
|
:options="cpuModelOptions"
|
||||||
|
:mode="mode"
|
||||||
|
:disabled="!!fetchError"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
@ -1,6 +1,5 @@
|
|||||||
<script>
|
<script>
|
||||||
import InfoBox from '@shell/components/InfoBox';
|
import InfoBox from '@shell/components/InfoBox';
|
||||||
|
|
||||||
import { NETWORK_ATTACHMENT } from '@shell/config/types';
|
import { NETWORK_ATTACHMENT } from '@shell/config/types';
|
||||||
import { sortBy } from '@shell/utils/sort';
|
import { sortBy } from '@shell/utils/sort';
|
||||||
import { clone } from '@shell/utils/object';
|
import { clone } from '@shell/utils/object';
|
||||||
@ -14,6 +13,13 @@ export default {
|
|||||||
components: { InfoBox, Base },
|
components: { InfoBox, Base },
|
||||||
|
|
||||||
props: {
|
props: {
|
||||||
|
vm: {
|
||||||
|
type: Object,
|
||||||
|
default: () => {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
mode: {
|
mode: {
|
||||||
type: String,
|
type: String,
|
||||||
default: 'create'
|
default: 'create'
|
||||||
@ -32,10 +38,15 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
async fetch() {
|
||||||
|
await this.fetchHotunplugData();
|
||||||
|
},
|
||||||
|
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
rows: this.addKeyId(clone(this.value)),
|
nameIdx: 1,
|
||||||
nameIdx: 1
|
rows: this.addKeyId(clone(this.value)),
|
||||||
|
hotunpluggableNics: new Set(),
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -64,15 +75,57 @@ export default {
|
|||||||
|
|
||||||
return out;
|
return out;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
canCheckHotunplug() {
|
||||||
|
return !!this.vm?.actions?.findHotunpluggableNics;
|
||||||
|
},
|
||||||
|
|
||||||
|
vmState() {
|
||||||
|
return this.vm?.stateDisplay;
|
||||||
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
watch: {
|
watch: {
|
||||||
value(neu) {
|
value(neu) {
|
||||||
this.rows = neu;
|
this.rows = this.mergeHotplugData(clone(neu));
|
||||||
},
|
},
|
||||||
|
|
||||||
|
vmState(newState, oldState) {
|
||||||
|
if (newState !== oldState) {
|
||||||
|
this.fetchHotunplugData();
|
||||||
|
}
|
||||||
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
methods: {
|
methods: {
|
||||||
|
async fetchHotunplugData() {
|
||||||
|
if (!this.canCheckHotunplug) {
|
||||||
|
this.rows = this.mergeHotplugData(clone(this.value));
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const resp = await this.vm.doAction('findHotunpluggableNics');
|
||||||
|
|
||||||
|
this.hotunpluggableNics = new Set(resp?.interfaces || []);
|
||||||
|
} catch (e) {
|
||||||
|
// eslint-disable-next-line no-console
|
||||||
|
console.error('Failed to fetch hot-unpluggable NICs:', e);
|
||||||
|
this.hotunpluggableNics = new Set();
|
||||||
|
}
|
||||||
|
|
||||||
|
this.rows = this.mergeHotplugData(clone(this.value));
|
||||||
|
},
|
||||||
|
|
||||||
|
mergeHotplugData(networks) {
|
||||||
|
return (networks || []).map((network) => ({
|
||||||
|
...network,
|
||||||
|
isHotunpluggable: this.hotunpluggableNics.has(network.name),
|
||||||
|
rowKeyId: network.rowKeyId || randomStr(10)
|
||||||
|
}));
|
||||||
|
},
|
||||||
|
|
||||||
add(type) {
|
add(type) {
|
||||||
const name = this.generateName();
|
const name = this.generateName();
|
||||||
|
|
||||||
@ -118,7 +171,11 @@ export default {
|
|||||||
|
|
||||||
update() {
|
update() {
|
||||||
this.$emit('update:value', this.rows);
|
this.$emit('update:value', this.rows);
|
||||||
}
|
},
|
||||||
|
|
||||||
|
unplugNIC(network) {
|
||||||
|
this.vm.unplugNIC(network.name);
|
||||||
|
},
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
@ -129,17 +186,28 @@ export default {
|
|||||||
v-for="(row, i) in rows"
|
v-for="(row, i) in rows"
|
||||||
:key="i"
|
:key="i"
|
||||||
>
|
>
|
||||||
<button
|
<div class="box-title mb-10">
|
||||||
v-if="!isView"
|
<h3>
|
||||||
type="button"
|
{{ t('harvester.virtualMachine.network.title') }}
|
||||||
class="role-link remove-vol"
|
</h3>
|
||||||
@click="remove(row)"
|
<button
|
||||||
>
|
v-if="!isView"
|
||||||
<i class="icon icon-x" />
|
type="button"
|
||||||
</button>
|
class="role-link btn btn-sm remove"
|
||||||
|
@click="remove(row)"
|
||||||
<h3> {{ t('harvester.virtualMachine.network.title') }} </h3>
|
>
|
||||||
|
<i class="icon icon-x" />
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
v-if="vm.hotplugNicFeatureEnabled && row.isHotunpluggable && isView"
|
||||||
|
type="button"
|
||||||
|
class="role-link btn btn-sm remove"
|
||||||
|
:disabled="!canCheckHotunplug"
|
||||||
|
@click="unplugNIC(row)"
|
||||||
|
>
|
||||||
|
{{ t('harvester.virtualMachine.hotUnplug.detachNIC.actionLabel') }}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
<Base
|
<Base
|
||||||
v-model:value="rows[i]"
|
v-model:value="rows[i]"
|
||||||
:rows="rows"
|
:rows="rows"
|
||||||
@ -162,16 +230,13 @@ export default {
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style lang='scss' scoped>
|
<style lang='scss' scoped>
|
||||||
.infoBox{
|
.box-title{
|
||||||
position: relative;
|
display: flex;
|
||||||
}
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
|
||||||
.remove-vol {
|
h3 {
|
||||||
position: absolute;
|
margin-bottom: 0;
|
||||||
top: 10px;
|
}
|
||||||
right: 16px;
|
|
||||||
padding:0px;
|
|
||||||
max-height: 28px;
|
|
||||||
min-height: 28px;
|
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@ -31,6 +31,7 @@ export default {
|
|||||||
const _hash = {
|
const _hash = {
|
||||||
pciclaims: this.$store.dispatch(`${ inStore }/findAll`, { type: HCI.PCI_CLAIM }),
|
pciclaims: this.$store.dispatch(`${ inStore }/findAll`, { type: HCI.PCI_CLAIM }),
|
||||||
sriovs: this.$store.dispatch(`${ inStore }/findAll`, { type: HCI.SR_IOV }),
|
sriovs: this.$store.dispatch(`${ inStore }/findAll`, { type: HCI.SR_IOV }),
|
||||||
|
srigpuovs: this.$store.dispatch(`${ inStore }/findAll`, { type: HCI.SR_IOVGPU_DEVICE }),
|
||||||
};
|
};
|
||||||
|
|
||||||
await allHash(_hash);
|
await allHash(_hash);
|
||||||
@ -106,19 +107,32 @@ export default {
|
|||||||
},
|
},
|
||||||
|
|
||||||
computed: {
|
computed: {
|
||||||
parentSriovOptions() {
|
allSriovs() {
|
||||||
const inStore = this.$store.getters['currentProduct'].inStore;
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
const allSriovs = this.$store.getters[`${ inStore }/all`](HCI.SR_IOV) || [];
|
|
||||||
|
|
||||||
return allSriovs.map((sriov) => {
|
return this.$store.getters[`${ inStore }/all`](HCI.SR_IOV) || [];
|
||||||
return sriov.id;
|
},
|
||||||
});
|
allSriovGPUs() {
|
||||||
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
|
|
||||||
|
return this.$store.getters[`${ inStore }/all`](HCI.SR_IOVGPU_DEVICE) || [];
|
||||||
|
},
|
||||||
|
parentSriovOptions() {
|
||||||
|
return this.allSriovs.map((sriov) => sriov.id);
|
||||||
|
},
|
||||||
|
parentSriovGPUOptions() {
|
||||||
|
return this.allSriovGPUs.map((sriovgpu) => sriovgpu.id);
|
||||||
},
|
},
|
||||||
parentSriovLabel() {
|
parentSriovLabel() {
|
||||||
return HCI_ANNOTATIONS.PARENT_SRIOV;
|
return HCI_ANNOTATIONS.PARENT_SRIOV;
|
||||||
}
|
},
|
||||||
|
parentSriovGPULabel() {
|
||||||
|
return HCI_ANNOTATIONS.PARENT_SRIOV_GPU;
|
||||||
|
},
|
||||||
|
vGPUAsPCIDeviceEnabled() {
|
||||||
|
return this.$store.getters['harvester-common/getFeatureEnabled']('vGPUAsPCIDevice');
|
||||||
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
methods: {
|
methods: {
|
||||||
enableGroup(rows = []) {
|
enableGroup(rows = []) {
|
||||||
const row = rows[0];
|
const row = rows[0];
|
||||||
@ -206,6 +220,15 @@ export default {
|
|||||||
:rows="rows"
|
:rows="rows"
|
||||||
@change-rows="changeRows"
|
@change-rows="changeRows"
|
||||||
/>
|
/>
|
||||||
|
<FilterBySriov
|
||||||
|
v-if="vGPUAsPCIDeviceEnabled"
|
||||||
|
ref="filterByParentSRIOVGPU"
|
||||||
|
:parent-sriov-options="parentSriovGPUOptions"
|
||||||
|
:parent-sriov-label="parentSriovGPULabel"
|
||||||
|
:label="t('harvester.sriov.parentSriovGPU')"
|
||||||
|
:rows="rows"
|
||||||
|
@change-rows="changeRows"
|
||||||
|
/>
|
||||||
</template>
|
</template>
|
||||||
</ResourceTable>
|
</ResourceTable>
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@ -8,6 +8,7 @@ import { set } from '@shell/utils/object';
|
|||||||
import { HCI } from '../../../types';
|
import { HCI } from '../../../types';
|
||||||
import DeviceList from './DeviceList';
|
import DeviceList from './DeviceList';
|
||||||
import CompatibilityMatrix from '../CompatibilityMatrix';
|
import CompatibilityMatrix from '../CompatibilityMatrix';
|
||||||
|
import MessageLink from '@shell/components/MessageLink';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: 'VirtualMachinePCIDevices',
|
name: 'VirtualMachinePCIDevices',
|
||||||
@ -15,7 +16,8 @@ export default {
|
|||||||
LabeledSelect,
|
LabeledSelect,
|
||||||
DeviceList,
|
DeviceList,
|
||||||
CompatibilityMatrix,
|
CompatibilityMatrix,
|
||||||
Banner
|
Banner,
|
||||||
|
MessageLink
|
||||||
},
|
},
|
||||||
props: {
|
props: {
|
||||||
mode: {
|
mode: {
|
||||||
@ -54,6 +56,11 @@ export default {
|
|||||||
|
|
||||||
const vmDevices = this.value?.domain?.devices?.hostDevices || [];
|
const vmDevices = this.value?.domain?.devices?.hostDevices || [];
|
||||||
const otherDevices = this.otherDevices(vmDevices).map(({ name }) => name);
|
const otherDevices = this.otherDevices(vmDevices).map(({ name }) => name);
|
||||||
|
const vmDeviceNames = vmDevices.map(({ name }) => name);
|
||||||
|
|
||||||
|
this.pciDevices.forEach((row) => {
|
||||||
|
row.allowDisable = !vmDeviceNames.includes(row.metadata.name);
|
||||||
|
});
|
||||||
|
|
||||||
vmDevices.forEach(({ name, deviceName }) => {
|
vmDevices.forEach(({ name, deviceName }) => {
|
||||||
const checkName = (deviceName || '').split('/')?.[1];
|
const checkName = (deviceName || '').split('/')?.[1];
|
||||||
@ -133,6 +140,13 @@ export default {
|
|||||||
return inUse;
|
return inUse;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
toVGpuDevicesPage() {
|
||||||
|
return {
|
||||||
|
name: 'harvester-c-cluster-resource',
|
||||||
|
params: { cluster: this.$store.getters['clusterId'], resource: HCI.VGPU_DEVICE },
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
devicesByNode() {
|
devicesByNode() {
|
||||||
return this.enabledDevices?.reduce((acc, device) => {
|
return this.enabledDevices?.reduce((acc, device) => {
|
||||||
const nodeName = device.status?.nodeName;
|
const nodeName = device.status?.nodeName;
|
||||||
@ -227,7 +241,12 @@ export default {
|
|||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="col span-12">
|
<div class="col span-12">
|
||||||
<Banner color="info">
|
<Banner color="info">
|
||||||
<t k="harvester.pci.howToUseDevice" />
|
<MessageLink
|
||||||
|
:to="toVGpuDevicesPage"
|
||||||
|
prefix-label="harvester.pci.howToUseDevice.prefix"
|
||||||
|
middle-label="harvester.pci.howToUseDevice.middle"
|
||||||
|
suffix-label="harvester.pci.howToUseDevice.suffix"
|
||||||
|
/>
|
||||||
</Banner>
|
</Banner>
|
||||||
<Banner
|
<Banner
|
||||||
v-if="selectedDevices.length > 0"
|
v-if="selectedDevices.length > 0"
|
||||||
|
|||||||
@ -48,6 +48,13 @@ export default {
|
|||||||
this[key] = res[key];
|
this[key] = res[key];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const vmDevices = this.value?.domain?.devices?.hostDevices || [];
|
||||||
|
const vmDeviceNames = vmDevices.map(({ name }) => name);
|
||||||
|
|
||||||
|
this.devices.forEach((row) => {
|
||||||
|
row.allowDisable = !vmDeviceNames.includes(row.metadata.name);
|
||||||
|
});
|
||||||
|
|
||||||
this.selectedDevices = (this.value?.domain?.devices?.hostDevices || [])
|
this.selectedDevices = (this.value?.domain?.devices?.hostDevices || [])
|
||||||
.map(({ name }) => name)
|
.map(({ name }) => name)
|
||||||
.filter((name) => this.enabledDevices.find((device) => device?.metadata?.name === name));
|
.filter((name) => this.enabledDevices.find((device) => device?.metadata?.name === name));
|
||||||
|
|||||||
@ -46,6 +46,13 @@ export default {
|
|||||||
this[key] = res[key];
|
this[key] = res[key];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const vmDevices = this.value?.domain?.devices?.gpus || [];
|
||||||
|
const vmDeviceNames = vmDevices.map(({ name }) => name);
|
||||||
|
|
||||||
|
this.devices.forEach((row) => {
|
||||||
|
row.allowDisable = !vmDeviceNames.includes(row.metadata.name);
|
||||||
|
});
|
||||||
|
|
||||||
const vGpus = this.vm.isOff ? [
|
const vGpus = this.vm.isOff ? [
|
||||||
...(this.value?.domain?.devices?.gpus || []).map(({ name }) => name),
|
...(this.value?.domain?.devices?.gpus || []).map(({ name }) => name),
|
||||||
] : [
|
] : [
|
||||||
|
|||||||
@ -13,11 +13,12 @@ import { ucFirst, randomStr } from '@shell/utils/string';
|
|||||||
import { removeObject } from '@shell/utils/array';
|
import { removeObject } from '@shell/utils/array';
|
||||||
import { _VIEW, _EDIT, _CREATE } from '@shell/config/query-params';
|
import { _VIEW, _EDIT, _CREATE } from '@shell/config/query-params';
|
||||||
import { PLUGIN_DEVELOPER, DEV } from '@shell/store/prefs';
|
import { PLUGIN_DEVELOPER, DEV } from '@shell/store/prefs';
|
||||||
import { SOURCE_TYPE } from '../../../config/harvester-map';
|
import { VOLUME_HOTPLUG_ACTION, SOURCE_TYPE } from '../../../config/harvester-map';
|
||||||
import { PRODUCT_NAME as HARVESTER_PRODUCT } from '../../../config/harvester';
|
import { PRODUCT_NAME as HARVESTER_PRODUCT } from '../../../config/harvester';
|
||||||
import { HCI } from '../../../types';
|
import { HCI } from '../../../types';
|
||||||
import { VOLUME_MODE } from '@pkg/harvester/config/types';
|
import { VOLUME_MODE } from '@pkg/harvester/config/types';
|
||||||
import { OFF } from '../../../models/kubevirt.io.virtualmachine';
|
import { OFF } from '../../../models/kubevirt.io.virtualmachine';
|
||||||
|
import { EMPTY_IMAGE } from '../../../utils/vm';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
emits: ['update:value'],
|
emits: ['update:value'],
|
||||||
@ -117,6 +118,10 @@ export default {
|
|||||||
return this.mode === _CREATE;
|
return this.mode === _CREATE;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
isHotplugCdRomFeatureEnabled() {
|
||||||
|
return this.$store.getters['harvester-common/getFeatureEnabled']('hotplugCdRom');
|
||||||
|
},
|
||||||
|
|
||||||
defaultStorageClass() {
|
defaultStorageClass() {
|
||||||
const defaultStorage = this.$store.getters['harvester/all'](STORAGE_CLASS).find((sc) => sc.isDefault);
|
const defaultStorage = this.$store.getters['harvester/all'](STORAGE_CLASS).find((sc) => sc.isDefault);
|
||||||
|
|
||||||
@ -146,7 +151,7 @@ export default {
|
|||||||
value: {
|
value: {
|
||||||
handler(neu) {
|
handler(neu) {
|
||||||
const rows = clone(neu).map((V) => {
|
const rows = clone(neu).map((V) => {
|
||||||
if (!this.isCreate && V.source !== SOURCE_TYPE.CONTAINER && !V.newCreateId) {
|
if (!this.isCreate && V.source !== SOURCE_TYPE.CONTAINER && !V.newCreateId && V.image !== EMPTY_IMAGE) {
|
||||||
V.to = {
|
V.to = {
|
||||||
name: `${ HARVESTER_PRODUCT }-c-cluster-resource-namespace-id`,
|
name: `${ HARVESTER_PRODUCT }-c-cluster-resource-namespace-id`,
|
||||||
params: {
|
params: {
|
||||||
@ -217,8 +222,48 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
unplugVolume(volume) {
|
canDoVolumeHotplugAction(volume) {
|
||||||
this.vm.unplugVolume(volume.name);
|
if (!this.isHotplugCdRomFeatureEnabled && volume.type === 'cd-rom') {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (volume.hotpluggable) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return volume.type === 'cd-rom' && volume.bus === 'sata' && volume.image === EMPTY_IMAGE;
|
||||||
|
},
|
||||||
|
|
||||||
|
getVolumeHotplugAction(volume) {
|
||||||
|
if (volume.type === 'cd-rom' && volume.bus === 'sata') {
|
||||||
|
if (volume.image === EMPTY_IMAGE) {
|
||||||
|
return VOLUME_HOTPLUG_ACTION.INSERT_CDROM_IMAGE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return VOLUME_HOTPLUG_ACTION.EJECT_CDROM_IMAGE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return VOLUME_HOTPLUG_ACTION.DETACH_DISK;
|
||||||
|
},
|
||||||
|
|
||||||
|
getVolumeHotplugActionLabel(volume) {
|
||||||
|
const labels = {
|
||||||
|
[VOLUME_HOTPLUG_ACTION.DETACH_DISK]: 'harvester.virtualMachine.hotUnplug.detachVolume.actionLabel',
|
||||||
|
[VOLUME_HOTPLUG_ACTION.INSERT_CDROM_IMAGE]: 'harvester.modal.insertCdRomVolume.actionLabel',
|
||||||
|
[VOLUME_HOTPLUG_ACTION.EJECT_CDROM_IMAGE]: 'harvester.virtualMachine.hotUnplug.ejectCdRomVolume.actionLabel',
|
||||||
|
};
|
||||||
|
|
||||||
|
return labels[this.getVolumeHotplugAction(volume)];
|
||||||
|
},
|
||||||
|
|
||||||
|
hotplugVolume(volume) {
|
||||||
|
const calls = {
|
||||||
|
[VOLUME_HOTPLUG_ACTION.DETACH_DISK]: () => this.vm.unplugVolume(volume.name),
|
||||||
|
[VOLUME_HOTPLUG_ACTION.INSERT_CDROM_IMAGE]: () => this.vm.insertCdRomVolume(volume.name),
|
||||||
|
[VOLUME_HOTPLUG_ACTION.EJECT_CDROM_IMAGE]: () => this.vm.ejectCdRomVolume(volume.name),
|
||||||
|
};
|
||||||
|
|
||||||
|
return calls[this.getVolumeHotplugAction(volume)]();
|
||||||
},
|
},
|
||||||
|
|
||||||
componentFor(type) {
|
componentFor(type) {
|
||||||
@ -303,55 +348,57 @@ export default {
|
|||||||
v-for="(volume, i) in rows"
|
v-for="(volume, i) in rows"
|
||||||
:key="volume.id"
|
:key="volume.id"
|
||||||
>
|
>
|
||||||
<InfoBox class="box">
|
<InfoBox>
|
||||||
<button
|
<div class="box-title mb-10">
|
||||||
v-if="!isView"
|
<h3>
|
||||||
type="button"
|
<span
|
||||||
class="role-link btn btn-sm remove"
|
v-if="volume.to && isVirtualType"
|
||||||
@click="removeVolume(volume)"
|
class="title"
|
||||||
>
|
|
||||||
<i class="icon icon-x" />
|
|
||||||
</button>
|
|
||||||
<button
|
|
||||||
v-if="volume.hotpluggable && isView"
|
|
||||||
type="button"
|
|
||||||
class="role-link btn remove"
|
|
||||||
@click="unplugVolume(volume)"
|
|
||||||
>
|
|
||||||
{{ t('harvester.virtualMachine.unplug.detachVolume') }}
|
|
||||||
</button>
|
|
||||||
<h3>
|
|
||||||
<span
|
|
||||||
v-if="volume.to && isVirtualType"
|
|
||||||
class="title"
|
|
||||||
>
|
|
||||||
<router-link :to="volume.to">
|
|
||||||
{{ t('harvester.virtualMachine.volume.edit') }} {{ headerFor(volume.source) }}
|
|
||||||
</router-link>
|
|
||||||
|
|
||||||
<BadgeStateFormatter
|
|
||||||
v-if="volume.pvc"
|
|
||||||
class="ml-10 state"
|
|
||||||
:arbitrary="true"
|
|
||||||
:row="volume.pvc"
|
|
||||||
:value="volume.pvc.state"
|
|
||||||
/>
|
|
||||||
<a
|
|
||||||
v-if="dev && !!volume.pvc && !!volume.pvc.resourceExternalLink"
|
|
||||||
v-clean-tooltip="t(volume.pvc.resourceExternalLink.tipsKey || 'generic.resourceExternalLinkTips')"
|
|
||||||
class="ml-5 resource-external"
|
|
||||||
rel="nofollow noopener noreferrer"
|
|
||||||
target="_blank"
|
|
||||||
:href="volume.pvc.resourceExternalLink.url"
|
|
||||||
>
|
>
|
||||||
<i class="icon icon-external-link" />
|
<router-link :to="volume.to">
|
||||||
</a>
|
{{ t('harvester.virtualMachine.volume.edit') }} {{ headerFor(volume.source) }}
|
||||||
</span>
|
</router-link>
|
||||||
|
|
||||||
<span v-else>
|
<BadgeStateFormatter
|
||||||
{{ headerFor(volume.source, !!volume?.volumeBackups) }}
|
v-if="volume.pvc"
|
||||||
</span>
|
class="ml-10 state"
|
||||||
</h3>
|
:arbitrary="true"
|
||||||
|
:row="volume.pvc"
|
||||||
|
:value="volume.pvc.state"
|
||||||
|
/>
|
||||||
|
<a
|
||||||
|
v-if="dev && !!volume.pvc && !!volume.pvc.resourceExternalLink"
|
||||||
|
v-clean-tooltip="t(volume.pvc.resourceExternalLink.tipsKey || 'generic.resourceExternalLinkTips')"
|
||||||
|
class="ml-5 resource-external"
|
||||||
|
rel="nofollow noopener noreferrer"
|
||||||
|
target="_blank"
|
||||||
|
:href="volume.pvc.resourceExternalLink.url"
|
||||||
|
>
|
||||||
|
<i class="icon icon-external-link" />
|
||||||
|
</a>
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<span v-else>
|
||||||
|
{{ headerFor(volume.source, !!volume?.volumeBackups) }}
|
||||||
|
</span>
|
||||||
|
</h3>
|
||||||
|
<button
|
||||||
|
v-if="!isView"
|
||||||
|
type="button"
|
||||||
|
class="role-link btn btn-sm remove"
|
||||||
|
@click="removeVolume(volume)"
|
||||||
|
>
|
||||||
|
<i class="icon icon-x" />
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
v-if="canDoVolumeHotplugAction(volume) && isView"
|
||||||
|
type="button"
|
||||||
|
class="role-link btn btn-sm remove"
|
||||||
|
@click="hotplugVolume(volume)"
|
||||||
|
>
|
||||||
|
{{ t(getVolumeHotplugActionLabel(volume)) }}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<component
|
<component
|
||||||
:is="componentFor(volume.source)"
|
:is="componentFor(volume.source)"
|
||||||
@ -495,25 +542,24 @@ export default {
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style lang='scss' scoped>
|
<style lang='scss' scoped>
|
||||||
.box {
|
.box-title {
|
||||||
position: relative;
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
|
||||||
|
h3 {
|
||||||
|
margin-bottom: 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.title {
|
.title {
|
||||||
display: flex;
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
|
||||||
.state {
|
.state {
|
||||||
font-size: 16px;
|
font-size: 16px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.remove {
|
|
||||||
position: absolute;
|
|
||||||
top: 10px;
|
|
||||||
right: 10px;
|
|
||||||
padding: 0px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.bootOrder {
|
.bootOrder {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
|
|||||||
@ -99,7 +99,6 @@ export default {
|
|||||||
this.allPVCs
|
this.allPVCs
|
||||||
.filter( (pvc) => {
|
.filter( (pvc) => {
|
||||||
let isAvailable = true;
|
let isAvailable = true;
|
||||||
let isBeingUsed = false;
|
|
||||||
|
|
||||||
this.rows.forEach( (O) => {
|
this.rows.forEach( (O) => {
|
||||||
if (O.volumeName === pvc.metadata.name) {
|
if (O.volumeName === pvc.metadata.name) {
|
||||||
@ -111,17 +110,16 @@ export default {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// already used as image volume
|
||||||
|
if (this.idx > 0 && pvc.metadata?.annotations?.[HCI_ANNOTATIONS.IMAGE_ID]) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
if (pvc.isGoldenImageVolume) {
|
if (pvc.isGoldenImageVolume) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (pvc.attachVM && isAvailable && pvc.attachVM?.id === this.vm?.id && this.isEdit) {
|
return isAvailable && pvc.isAvailable;
|
||||||
isBeingUsed = false;
|
|
||||||
} else if (pvc.attachVM) {
|
|
||||||
isBeingUsed = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return isAvailable && !isBeingUsed && pvc.isAvailable;
|
|
||||||
})
|
})
|
||||||
.map((pvc) => {
|
.map((pvc) => {
|
||||||
return {
|
return {
|
||||||
|
|||||||
@ -14,6 +14,7 @@ import { _VIEW } from '@shell/config/query-params';
|
|||||||
import LabelValue from '@shell/components/LabelValue';
|
import LabelValue from '@shell/components/LabelValue';
|
||||||
import { ucFirst } from '@shell/utils/string';
|
import { ucFirst } from '@shell/utils/string';
|
||||||
import { GIBIBYTE } from '../../../../utils/unit';
|
import { GIBIBYTE } from '../../../../utils/unit';
|
||||||
|
import { EMPTY_IMAGE } from '../../../../utils/vm';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: 'HarvesterEditVMImage',
|
name: 'HarvesterEditVMImage',
|
||||||
@ -96,8 +97,20 @@ export default {
|
|||||||
return this.mode === _VIEW;
|
return this.mode === _VIEW;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
isExistingCdrom() {
|
||||||
|
return this.value.type === 'cd-rom' && !this.value.newCreateId;
|
||||||
|
},
|
||||||
|
|
||||||
|
isEmptyImage() {
|
||||||
|
return this.value.image === EMPTY_IMAGE;
|
||||||
|
},
|
||||||
|
|
||||||
|
isHotplugCdRomFeatureEnabled() {
|
||||||
|
return this.$store.getters['harvester-common/getFeatureEnabled']('hotplugCdRom');
|
||||||
|
},
|
||||||
|
|
||||||
imagesOption() {
|
imagesOption() {
|
||||||
return this.images
|
const images = this.images
|
||||||
.filter((image) => {
|
.filter((image) => {
|
||||||
if (!image.isReady) return false;
|
if (!image.isReady) return false;
|
||||||
|
|
||||||
@ -114,6 +127,19 @@ export default {
|
|||||||
value: image.id,
|
value: image.id,
|
||||||
disabled: image.isImportedImage
|
disabled: image.isImportedImage
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
const options = [];
|
||||||
|
|
||||||
|
if (this.isHotplugCdRomFeatureEnabled) {
|
||||||
|
options.push({
|
||||||
|
label: this.t('harvester.virtualMachine.volume.emptyImage'),
|
||||||
|
value: EMPTY_IMAGE,
|
||||||
|
disabled: false
|
||||||
|
});
|
||||||
|
}
|
||||||
|
options.push(...images);
|
||||||
|
|
||||||
|
return options;
|
||||||
},
|
},
|
||||||
|
|
||||||
imageName() {
|
imageName() {
|
||||||
@ -179,6 +205,7 @@ export default {
|
|||||||
'value.type'(neu) {
|
'value.type'(neu) {
|
||||||
if (neu === 'cd-rom') {
|
if (neu === 'cd-rom') {
|
||||||
this.value['bus'] = 'sata';
|
this.value['bus'] = 'sata';
|
||||||
|
this.updateHotpluggable();
|
||||||
this.update();
|
this.update();
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@ -221,12 +248,48 @@ export default {
|
|||||||
|
|
||||||
return label;
|
return label;
|
||||||
},
|
},
|
||||||
|
|
||||||
update() {
|
update() {
|
||||||
this.value.hasDiskError = this.showDiskTooSmallError;
|
this.value.hasDiskError = this.showDiskTooSmallError;
|
||||||
this.$emit('update');
|
this.$emit('update');
|
||||||
},
|
},
|
||||||
|
|
||||||
|
updateHotpluggable() {
|
||||||
|
if (this.value.type !== 'cd-rom') {
|
||||||
|
this.value['hotpluggable'] = false;
|
||||||
|
} else {
|
||||||
|
this.value['hotpluggable'] = (this.value.bus === 'sata');
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
onTypeChange() {
|
||||||
|
if (this.value.image === EMPTY_IMAGE && this.value.type !== 'cd-rom') {
|
||||||
|
this.value['image'] = '';
|
||||||
|
}
|
||||||
|
|
||||||
|
this.updateHotpluggable();
|
||||||
|
this.update();
|
||||||
|
},
|
||||||
|
|
||||||
|
onBusChange() {
|
||||||
|
if (this.value.image === EMPTY_IMAGE && this.value.bus !== 'sata') {
|
||||||
|
this.value['image'] = '';
|
||||||
|
}
|
||||||
|
|
||||||
|
this.updateHotpluggable();
|
||||||
|
this.update();
|
||||||
|
},
|
||||||
|
|
||||||
onImageChange() {
|
onImageChange() {
|
||||||
|
if (this.value.image === EMPTY_IMAGE) {
|
||||||
|
this.value['type'] = 'cd-rom';
|
||||||
|
this.value['bus'] = 'sata';
|
||||||
|
this.value['size'] = `0${ GIBIBYTE }`;
|
||||||
|
this.update();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const imageResource = this.$store.getters['harvester/all'](HCI.IMAGE)?.find( (I) => this.value.image === I.id);
|
const imageResource = this.$store.getters['harvester/all'](HCI.IMAGE)?.find( (I) => this.value.image === I.id);
|
||||||
const isIsoImage = /iso$/i.test(imageResource?.imageSuffix);
|
const isIsoImage = /iso$/i.test(imageResource?.imageSuffix);
|
||||||
const imageSize = Math.max(imageResource?.status?.size, imageResource?.status?.virtualSize);
|
const imageSize = Math.max(imageResource?.status?.size, imageResource?.status?.virtualSize);
|
||||||
@ -239,6 +302,8 @@ export default {
|
|||||||
this.value['bus'] = 'virtio';
|
this.value['bus'] = 'virtio';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this.updateHotpluggable();
|
||||||
|
|
||||||
if (imageSize) {
|
if (imageSize) {
|
||||||
let imageSizeGiB = Math.ceil(imageSize / 1024 / 1024 / 1024);
|
let imageSizeGiB = Math.ceil(imageSize / 1024 / 1024 / 1024);
|
||||||
|
|
||||||
@ -256,6 +321,10 @@ export default {
|
|||||||
},
|
},
|
||||||
|
|
||||||
checkImageExists(imageId) {
|
checkImageExists(imageId) {
|
||||||
|
if (imageId === EMPTY_IMAGE) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (!!imageId && this.imagesOption.length > 0 && !findBy(this.imagesOption, 'value', imageId)) {
|
if (!!imageId && this.imagesOption.length > 0 && !findBy(this.imagesOption, 'value', imageId)) {
|
||||||
this.$store.dispatch('growl/error', {
|
this.$store.dispatch('growl/error', {
|
||||||
title: this.$store.getters['i18n/t']('harvester.vmTemplate.tips.notExistImage.title', { name: imageId }),
|
title: this.$store.getters['i18n/t']('harvester.vmTemplate.tips.notExistImage.title', { name: imageId }),
|
||||||
@ -283,6 +352,7 @@ export default {
|
|||||||
>
|
>
|
||||||
<LabeledInput
|
<LabeledInput
|
||||||
v-model:value="value.name"
|
v-model:value="value.name"
|
||||||
|
:disabled="!isCreate && isExistingCdrom"
|
||||||
:label="t('harvester.fields.name')"
|
:label="t('harvester.fields.name')"
|
||||||
required
|
required
|
||||||
:mode="mode"
|
:mode="mode"
|
||||||
@ -302,10 +372,11 @@ export default {
|
|||||||
>
|
>
|
||||||
<LabeledSelect
|
<LabeledSelect
|
||||||
v-model:value="value.type"
|
v-model:value="value.type"
|
||||||
|
:disabled="!isCreate && isExistingCdrom"
|
||||||
:label="t('harvester.fields.type')"
|
:label="t('harvester.fields.type')"
|
||||||
:options="VOLUME_TYPE"
|
:options="VOLUME_TYPE"
|
||||||
:mode="mode"
|
:mode="mode"
|
||||||
@update:value="update"
|
@update:value="onTypeChange"
|
||||||
/>
|
/>
|
||||||
</InputOrDisplay>
|
</InputOrDisplay>
|
||||||
</div>
|
</div>
|
||||||
@ -323,7 +394,7 @@ export default {
|
|||||||
>
|
>
|
||||||
<LabeledSelect
|
<LabeledSelect
|
||||||
v-model:value="value.image"
|
v-model:value="value.image"
|
||||||
:disabled="idx === 0 && !isCreate && !value.newCreateId && isVirtualType"
|
:disabled="(idx === 0 || isExistingCdrom) && (!isCreate && !value.newCreateId && isVirtualType)"
|
||||||
:label="t('harvester.fields.image')"
|
:label="t('harvester.fields.image')"
|
||||||
:options="imagesOption"
|
:options="imagesOption"
|
||||||
:mode="mode"
|
:mode="mode"
|
||||||
@ -351,7 +422,7 @@ export default {
|
|||||||
:label="t('harvester.fields.size')"
|
:label="t('harvester.fields.size')"
|
||||||
:mode="mode"
|
:mode="mode"
|
||||||
:required="validateRequired"
|
:required="validateRequired"
|
||||||
:disabled="isResizeDisabled"
|
:disabled="isResizeDisabled || isEmptyImage || (!isCreate && isExistingCdrom)"
|
||||||
:suffix="GIBIBYTE"
|
:suffix="GIBIBYTE"
|
||||||
@update:value="update"
|
@update:value="update"
|
||||||
/>
|
/>
|
||||||
@ -374,7 +445,8 @@ export default {
|
|||||||
:label="t('harvester.virtualMachine.volume.bus')"
|
:label="t('harvester.virtualMachine.volume.bus')"
|
||||||
:mode="mode"
|
:mode="mode"
|
||||||
:options="InterfaceOption"
|
:options="InterfaceOption"
|
||||||
@update:value="update"
|
:disabled="!isCreate && isExistingCdrom"
|
||||||
|
@update:value="onBusChange"
|
||||||
/>
|
/>
|
||||||
</InputOrDisplay>
|
</InputOrDisplay>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@ -2,7 +2,7 @@
|
|||||||
import { isEqual } from 'lodash';
|
import { isEqual } from 'lodash';
|
||||||
import { mapGetters } from 'vuex';
|
import { mapGetters } from 'vuex';
|
||||||
import Tabbed from '@shell/components/Tabbed';
|
import Tabbed from '@shell/components/Tabbed';
|
||||||
import { clone } from '@shell/utils/object';
|
import { clone, set } from '@shell/utils/object';
|
||||||
import Tab from '@shell/components/Tabbed/Tab';
|
import Tab from '@shell/components/Tabbed/Tab';
|
||||||
import { Checkbox } from '@components/Form/Checkbox';
|
import { Checkbox } from '@components/Form/Checkbox';
|
||||||
import CruResource from '@shell/components/CruResource';
|
import CruResource from '@shell/components/CruResource';
|
||||||
@ -32,6 +32,7 @@ import PciDevices from './VirtualMachinePciDevices/index';
|
|||||||
import AccessCredentials from './VirtualMachineAccessCredentials';
|
import AccessCredentials from './VirtualMachineAccessCredentials';
|
||||||
import CloudConfig from './VirtualMachineCloudConfig';
|
import CloudConfig from './VirtualMachineCloudConfig';
|
||||||
import CpuMemory from './VirtualMachineCpuMemory';
|
import CpuMemory from './VirtualMachineCpuMemory';
|
||||||
|
import CpuModel from './VirtualMachineCpuModel';
|
||||||
import Network from './VirtualMachineNetwork';
|
import Network from './VirtualMachineNetwork';
|
||||||
import Volume from './VirtualMachineVolume';
|
import Volume from './VirtualMachineVolume';
|
||||||
import SSHKey from './VirtualMachineSSHKey';
|
import SSHKey from './VirtualMachineSSHKey';
|
||||||
@ -57,6 +58,7 @@ export default {
|
|||||||
SSHKey,
|
SSHKey,
|
||||||
Network,
|
Network,
|
||||||
CpuMemory,
|
CpuMemory,
|
||||||
|
CpuModel,
|
||||||
CloudConfig,
|
CloudConfig,
|
||||||
NodeScheduling,
|
NodeScheduling,
|
||||||
PodAffinity,
|
PodAffinity,
|
||||||
@ -96,7 +98,6 @@ export default {
|
|||||||
templateVersionId: '',
|
templateVersionId: '',
|
||||||
namePrefix: '',
|
namePrefix: '',
|
||||||
isSingle: true,
|
isSingle: true,
|
||||||
useTemplate: false,
|
|
||||||
isOpen: false,
|
isOpen: false,
|
||||||
hostname,
|
hostname,
|
||||||
isRestartImmediately,
|
isRestartImmediately,
|
||||||
@ -210,6 +211,10 @@ export default {
|
|||||||
|
|
||||||
return false;
|
return false;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
vGPUAsPCIDeviceEnabled() {
|
||||||
|
return this.$store.getters['harvester-common/getFeatureEnabled']('vGPUAsPCIDevice');
|
||||||
|
},
|
||||||
usbPassthroughEnabled() {
|
usbPassthroughEnabled() {
|
||||||
return this.$store.getters['harvester-common/getFeatureEnabled']('usbPassthrough');
|
return this.$store.getters['harvester-common/getFeatureEnabled']('usbPassthrough');
|
||||||
},
|
},
|
||||||
@ -255,6 +260,7 @@ export default {
|
|||||||
return volume;
|
return volume;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
delete cloneVersionVM.metadata.annotations[HCI_ANNOTATIONS.MAC_ADDRESS];
|
||||||
cloneVersionVM.metadata.annotations[HCI_ANNOTATIONS.VOLUME_CLAIM_TEMPLATE] = JSON.stringify(deleteDataSource);
|
cloneVersionVM.metadata.annotations[HCI_ANNOTATIONS.VOLUME_CLAIM_TEMPLATE] = JSON.stringify(deleteDataSource);
|
||||||
|
|
||||||
// Update instance labels, labels and annotations
|
// Update instance labels, labels and annotations
|
||||||
@ -490,6 +496,7 @@ export default {
|
|||||||
if (this.isSingle) {
|
if (this.isSingle) {
|
||||||
if (!this.value.spec.template.spec.hostname) {
|
if (!this.value.spec.template.spec.hostname) {
|
||||||
this.value.spec.template.spec['hostname'] = this.value.metadata.name;
|
this.value.spec.template.spec['hostname'] = this.value.metadata.name;
|
||||||
|
this.spec.template.spec['hostname'] = this.value.metadata.name;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -537,6 +544,18 @@ export default {
|
|||||||
|
|
||||||
return out;
|
return out;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
updateCpuModel(value) {
|
||||||
|
if (!this.spec?.template?.spec?.domain?.cpu) {
|
||||||
|
set(this.spec, 'template.spec.domain.cpu', {});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (value && value !== '') {
|
||||||
|
set(this.spec.template.spec.domain.cpu, 'model', value);
|
||||||
|
} else {
|
||||||
|
delete this.spec.template.spec.domain.cpu.model;
|
||||||
|
}
|
||||||
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
@ -725,7 +744,7 @@ export default {
|
|||||||
</Tab>
|
</Tab>
|
||||||
|
|
||||||
<Tab
|
<Tab
|
||||||
v-if="enabledSriovgpu"
|
v-if="enabledSriovgpu && !vGPUAsPCIDeviceEnabled"
|
||||||
:label="t('harvester.tab.vGpuDevices')"
|
:label="t('harvester.tab.vGpuDevices')"
|
||||||
name="vGpuDevices"
|
name="vGpuDevices"
|
||||||
:weight="-6"
|
:weight="-6"
|
||||||
@ -869,6 +888,16 @@ export default {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-6">
|
||||||
|
<CpuModel
|
||||||
|
v-model:value="cpuModel"
|
||||||
|
:mode="mode"
|
||||||
|
@update:value="updateCpuModel"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="row mb-20">
|
<div class="row mb-20">
|
||||||
<a
|
<a
|
||||||
v-if="showAdvanced"
|
v-if="showAdvanced"
|
||||||
|
|||||||
357
pkg/harvester/edit/migration.harvesterhci.io.openstacksource.vue
Normal file
357
pkg/harvester/edit/migration.harvesterhci.io.openstacksource.vue
Normal file
@ -0,0 +1,357 @@
|
|||||||
|
<script>
|
||||||
|
import CruResource from '@shell/components/CruResource';
|
||||||
|
import Tabbed from '@shell/components/Tabbed';
|
||||||
|
import Tab from '@shell/components/Tabbed/Tab';
|
||||||
|
import { LabeledInput } from '@components/Form/LabeledInput';
|
||||||
|
import LabeledSelect from '@shell/components/form/LabeledSelect';
|
||||||
|
import NameNsDescription from '@shell/components/form/NameNsDescription';
|
||||||
|
import { RadioGroup } from '@components/Form/Radio';
|
||||||
|
import UnitInput from '@shell/components/form/UnitInput';
|
||||||
|
import CreateEditView from '@shell/mixins/create-edit-view';
|
||||||
|
import FormValidation from '@shell/mixins/form-validation';
|
||||||
|
import { SECRET } from '@shell/config/types';
|
||||||
|
import { randomStr } from '@shell/utils/string';
|
||||||
|
import { mapGetters } from 'vuex';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'EditOpenstackSource',
|
||||||
|
|
||||||
|
emits: ['update:value'],
|
||||||
|
|
||||||
|
components: {
|
||||||
|
CruResource,
|
||||||
|
Tabbed,
|
||||||
|
Tab,
|
||||||
|
LabeledInput,
|
||||||
|
LabeledSelect,
|
||||||
|
NameNsDescription,
|
||||||
|
RadioGroup,
|
||||||
|
UnitInput
|
||||||
|
},
|
||||||
|
|
||||||
|
mixins: [CreateEditView, FormValidation],
|
||||||
|
|
||||||
|
inheritAttrs: false,
|
||||||
|
|
||||||
|
props: {
|
||||||
|
value: {
|
||||||
|
type: Object,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
mode: {
|
||||||
|
type: String,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
async fetch() {
|
||||||
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
|
|
||||||
|
this.allSecrets = await this.$store.dispatch(`${ inStore }/findAll`, { type: SECRET });
|
||||||
|
},
|
||||||
|
|
||||||
|
data() {
|
||||||
|
if (!this.value.spec) this.value.spec = {};
|
||||||
|
if (!this.value.spec.credentials) this.value.spec.credentials = {};
|
||||||
|
|
||||||
|
const initialMode = this.value.spec.credentials.name ? 'existing' : 'new';
|
||||||
|
|
||||||
|
return {
|
||||||
|
allSecrets: [],
|
||||||
|
authMode: initialMode,
|
||||||
|
|
||||||
|
newUsername: '',
|
||||||
|
newPassword: '',
|
||||||
|
newProjectName: '',
|
||||||
|
newDomainName: '',
|
||||||
|
newCaCert: '',
|
||||||
|
|
||||||
|
// Rules for fields that exist in the value object (Model)
|
||||||
|
fvFormRuleSets: [
|
||||||
|
{ path: 'metadata.name', rules: ['nameRequired'] },
|
||||||
|
{ path: 'spec.endpoint', rules: ['endpointRequired'] },
|
||||||
|
{ path: 'spec.region', rules: ['regionRequired'] },
|
||||||
|
],
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
computed: {
|
||||||
|
...mapGetters({ t: 'i18n/t' }),
|
||||||
|
|
||||||
|
authModeOptions() {
|
||||||
|
return [
|
||||||
|
{ label: this.t('harvester.addons.vmImport.fields.createSecret'), value: 'new' },
|
||||||
|
{ label: this.t('harvester.addons.vmImport.fields.useSecret'), value: 'existing' }
|
||||||
|
];
|
||||||
|
},
|
||||||
|
|
||||||
|
secretOptions() {
|
||||||
|
const currentNamespace = this.value.metadata.namespace || 'default';
|
||||||
|
|
||||||
|
return this.allSecrets
|
||||||
|
.filter((s) => s.metadata.namespace === currentNamespace)
|
||||||
|
.map((s) => ({
|
||||||
|
label: s.nameDisplay,
|
||||||
|
value: s.metadata.name
|
||||||
|
}));
|
||||||
|
},
|
||||||
|
|
||||||
|
// Define custom rules for the FormValidation mixin
|
||||||
|
fvExtraRules() {
|
||||||
|
return {
|
||||||
|
nameRequired: (val) => !val ? this.t('validation.required', { key: this.t('harvester.fields.name') }) : undefined,
|
||||||
|
endpointRequired: (val) => !val ? this.t('validation.required', { key: this.t('harvester.addons.vmImport.openstack.fields.endpoint') }) : undefined,
|
||||||
|
regionRequired: (val) => !val ? this.t('validation.required', { key: this.t('harvester.addons.vmImport.openstack.fields.region') }) : undefined,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
// Combine mixin validation + conditional manual checks
|
||||||
|
isFormValid() {
|
||||||
|
// Check static fields via Mixin
|
||||||
|
if (!this.fvFormIsValid) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check conditional fields
|
||||||
|
if (this.authMode === 'new') {
|
||||||
|
if (!this.newUsername || !this.newPassword) return false;
|
||||||
|
if (!this.newProjectName || !this.newDomainName) return false;
|
||||||
|
} else {
|
||||||
|
if (!this.value.spec.credentials.name) return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
methods: {
|
||||||
|
usernameRule(val) {
|
||||||
|
return !val ? this.t('validation.required', { key: this.t('harvester.addons.vmImport.fields.username') }) : undefined;
|
||||||
|
},
|
||||||
|
passwordRule(val) {
|
||||||
|
return !val ? this.t('validation.required', { key: this.t('harvester.addons.vmImport.fields.password') }) : undefined;
|
||||||
|
},
|
||||||
|
projectRule(val) {
|
||||||
|
return !val ? this.t('validation.required', { key: this.t('harvester.addons.vmImport.openstack.fields.projectName') }) : undefined;
|
||||||
|
},
|
||||||
|
domainRule(val) {
|
||||||
|
return !val ? this.t('validation.required', { key: this.t('harvester.addons.vmImport.openstack.fields.domainName') }) : undefined;
|
||||||
|
},
|
||||||
|
secretRule(val) {
|
||||||
|
return !val ? this.t('validation.required', { key: this.t('harvester.addons.vmImport.fields.selectSecret') }) : undefined;
|
||||||
|
},
|
||||||
|
|
||||||
|
async saveSource(buttonCb) {
|
||||||
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (this.authMode === 'new') {
|
||||||
|
const secretName = `${ this.value.metadata.name }-creds-${ randomStr(4).toLowerCase() }`;
|
||||||
|
const namespace = this.value.metadata.namespace || 'default';
|
||||||
|
|
||||||
|
const newSecret = await this.$store.dispatch(`${ inStore }/create`, {
|
||||||
|
type: SECRET,
|
||||||
|
metadata: {
|
||||||
|
name: secretName,
|
||||||
|
namespace
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
newSecret['_type'] = 'Opaque';
|
||||||
|
newSecret['data'] = {
|
||||||
|
username: btoa(this.newUsername),
|
||||||
|
password: btoa(this.newPassword),
|
||||||
|
project_name: btoa(this.newProjectName),
|
||||||
|
domain_name: btoa(this.newDomainName),
|
||||||
|
ca_cert: this.newCaCert ? btoa(this.newCaCert) : undefined
|
||||||
|
};
|
||||||
|
|
||||||
|
await newSecret.save();
|
||||||
|
|
||||||
|
this.value.spec.credentials = {
|
||||||
|
name: secretName,
|
||||||
|
namespace
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
await this.save(buttonCb);
|
||||||
|
} catch (err) {
|
||||||
|
this.errors = [err];
|
||||||
|
buttonCb(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<CruResource
|
||||||
|
:done-route="doneRoute"
|
||||||
|
:resource="value"
|
||||||
|
:mode="mode"
|
||||||
|
:errors="errors"
|
||||||
|
:apply-hooks="applyHooks"
|
||||||
|
:validation-passed="isFormValid"
|
||||||
|
@finish="saveSource"
|
||||||
|
@error="e=>errors=e"
|
||||||
|
>
|
||||||
|
<NameNsDescription
|
||||||
|
:value="value"
|
||||||
|
:mode="mode"
|
||||||
|
:rules="{ name: fvGetAndReportPathRules('metadata.name') }"
|
||||||
|
@update:value="$emit('update:value', $event)"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Tabbed
|
||||||
|
v-bind="$attrs"
|
||||||
|
class="mt-15"
|
||||||
|
:side-tabs="true"
|
||||||
|
>
|
||||||
|
<Tab
|
||||||
|
name="basic"
|
||||||
|
:label="t('harvester.addons.vmImport.titles.basic')"
|
||||||
|
:weight="3"
|
||||||
|
>
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-6">
|
||||||
|
<LabeledInput
|
||||||
|
v-model:value="value.spec.endpoint"
|
||||||
|
:label="t('harvester.addons.vmImport.openstack.fields.endpoint')"
|
||||||
|
:placeholder="t('harvester.addons.vmImport.openstack.placeholders.endpoint')"
|
||||||
|
:mode="mode"
|
||||||
|
:rules="fvGetAndReportPathRules('spec.endpoint')"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="col span-6">
|
||||||
|
<LabeledInput
|
||||||
|
v-model:value="value.spec.region"
|
||||||
|
:label="t('harvester.addons.vmImport.openstack.fields.region')"
|
||||||
|
:placeholder="t('harvester.addons.vmImport.openstack.placeholders.region')"
|
||||||
|
:mode="mode"
|
||||||
|
:rules="fvGetAndReportPathRules('spec.region')"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Tab>
|
||||||
|
|
||||||
|
<Tab
|
||||||
|
name="auth"
|
||||||
|
:label="t('harvester.addons.vmImport.titles.auth')"
|
||||||
|
:weight="2"
|
||||||
|
>
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-12">
|
||||||
|
<RadioGroup
|
||||||
|
v-model:value="authMode"
|
||||||
|
name="authMode"
|
||||||
|
:options="authModeOptions"
|
||||||
|
:mode="mode"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-if="authMode === 'new'">
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-6">
|
||||||
|
<LabeledInput
|
||||||
|
v-model:value="newUsername"
|
||||||
|
:label="t('harvester.addons.vmImport.fields.username')"
|
||||||
|
:mode="mode"
|
||||||
|
:rules="[usernameRule]"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="col span-6">
|
||||||
|
<LabeledInput
|
||||||
|
v-model:value="newPassword"
|
||||||
|
type="password"
|
||||||
|
:label="t('harvester.addons.vmImport.fields.password')"
|
||||||
|
:mode="mode"
|
||||||
|
:rules="[passwordRule]"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-6">
|
||||||
|
<LabeledInput
|
||||||
|
v-model:value="newProjectName"
|
||||||
|
:label="t('harvester.addons.vmImport.openstack.fields.projectName')"
|
||||||
|
:placeholder="t('harvester.addons.vmImport.openstack.placeholders.projectName')"
|
||||||
|
:mode="mode"
|
||||||
|
:rules="[projectRule]"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="col span-6">
|
||||||
|
<LabeledInput
|
||||||
|
v-model:value="newDomainName"
|
||||||
|
:label="t('harvester.addons.vmImport.openstack.fields.domainName')"
|
||||||
|
:placeholder="t('harvester.addons.vmImport.openstack.placeholders.domainName')"
|
||||||
|
:mode="mode"
|
||||||
|
:rules="[domainRule]"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-12">
|
||||||
|
<LabeledInput
|
||||||
|
v-model:value="newCaCert"
|
||||||
|
type="multiline"
|
||||||
|
:label="t('harvester.addons.vmImport.fields.caCert')"
|
||||||
|
:placeholder="t('harvester.addons.vmImport.placeholders.caCert')"
|
||||||
|
:min-height="100"
|
||||||
|
:mode="mode"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-if="authMode === 'existing'">
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-6">
|
||||||
|
<LabeledSelect
|
||||||
|
v-model:value="value.spec.credentials.name"
|
||||||
|
:options="secretOptions"
|
||||||
|
:label="t('harvester.addons.vmImport.fields.selectSecret')"
|
||||||
|
:mode="mode"
|
||||||
|
:rules="[secretRule]"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Tab>
|
||||||
|
|
||||||
|
<Tab
|
||||||
|
name="advanced"
|
||||||
|
:label="t('harvester.addons.vmImport.titles.advanced')"
|
||||||
|
:weight="1"
|
||||||
|
>
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-6">
|
||||||
|
<UnitInput
|
||||||
|
v-model:value="value.spec.uploadImageRetryCount"
|
||||||
|
:label="t('harvester.addons.vmImport.openstack.fields.retryCount')"
|
||||||
|
:placeholder="t('harvester.addons.vmImport.openstack.placeholders.retryCount')"
|
||||||
|
suffix="Times"
|
||||||
|
:mode="mode"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="col span-6">
|
||||||
|
<UnitInput
|
||||||
|
v-model:value="value.spec.uploadImageRetryDelay"
|
||||||
|
:label="t('harvester.addons.vmImport.openstack.fields.retryDelay')"
|
||||||
|
:placeholder="t('harvester.addons.vmImport.openstack.placeholders.retryDelay')"
|
||||||
|
suffix="Seconds"
|
||||||
|
:mode="mode"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Tab>
|
||||||
|
</Tabbed>
|
||||||
|
</CruResource>
|
||||||
|
</template>
|
||||||
322
pkg/harvester/edit/migration.harvesterhci.io.ovasource.vue
Normal file
322
pkg/harvester/edit/migration.harvesterhci.io.ovasource.vue
Normal file
@ -0,0 +1,322 @@
|
|||||||
|
<script>
|
||||||
|
import CruResource from '@shell/components/CruResource';
|
||||||
|
import Tabbed from '@shell/components/Tabbed';
|
||||||
|
import Tab from '@shell/components/Tabbed/Tab';
|
||||||
|
import { LabeledInput } from '@components/Form/LabeledInput';
|
||||||
|
import LabeledSelect from '@shell/components/form/LabeledSelect';
|
||||||
|
import NameNsDescription from '@shell/components/form/NameNsDescription';
|
||||||
|
import { RadioGroup } from '@components/Form/Radio';
|
||||||
|
import UnitInput from '@shell/components/form/UnitInput';
|
||||||
|
import CreateEditView from '@shell/mixins/create-edit-view';
|
||||||
|
import FormValidation from '@shell/mixins/form-validation';
|
||||||
|
import { SECRET } from '@shell/config/types';
|
||||||
|
import { randomStr } from '@shell/utils/string';
|
||||||
|
import { mapGetters } from 'vuex';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'EditOvaSource',
|
||||||
|
|
||||||
|
emits: ['update:value'],
|
||||||
|
|
||||||
|
components: {
|
||||||
|
CruResource,
|
||||||
|
Tabbed,
|
||||||
|
Tab,
|
||||||
|
LabeledInput,
|
||||||
|
LabeledSelect,
|
||||||
|
NameNsDescription,
|
||||||
|
RadioGroup,
|
||||||
|
UnitInput
|
||||||
|
},
|
||||||
|
|
||||||
|
mixins: [CreateEditView, FormValidation],
|
||||||
|
|
||||||
|
inheritAttrs: false,
|
||||||
|
|
||||||
|
props: {
|
||||||
|
value: {
|
||||||
|
type: Object,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
mode: {
|
||||||
|
type: String,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
async fetch() {
|
||||||
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
|
|
||||||
|
this.allSecrets = await this.$store.dispatch(`${ inStore }/findAll`, { type: SECRET });
|
||||||
|
},
|
||||||
|
|
||||||
|
data() {
|
||||||
|
if (!this.value.spec) this.value.spec = {};
|
||||||
|
|
||||||
|
// Auth is optional for OVA (public URLs).
|
||||||
|
// If credentials.name exists -> Existing.
|
||||||
|
// If not -> None (default).
|
||||||
|
let initialMode = 'none';
|
||||||
|
|
||||||
|
if (this.value.spec.credentials?.name) {
|
||||||
|
initialMode = 'existing';
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
allSecrets: [],
|
||||||
|
authMode: initialMode,
|
||||||
|
|
||||||
|
newUsername: '',
|
||||||
|
newPassword: '',
|
||||||
|
newCaCert: '', // Key will be "ca.crt"
|
||||||
|
|
||||||
|
// Validation Rules for static fields
|
||||||
|
fvFormRuleSets: [
|
||||||
|
{ path: 'metadata.name', rules: ['nameRequired'] },
|
||||||
|
{ path: 'spec.url', rules: ['urlRequired'] },
|
||||||
|
],
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
computed: {
|
||||||
|
...mapGetters({ t: 'i18n/t' }),
|
||||||
|
|
||||||
|
authModeOptions() {
|
||||||
|
return [
|
||||||
|
{ label: this.t('harvester.addons.vmImport.fields.none'), value: 'none' },
|
||||||
|
{ label: this.t('harvester.addons.vmImport.fields.createSecret'), value: 'new' },
|
||||||
|
{ label: this.t('harvester.addons.vmImport.fields.useSecret'), value: 'existing' }
|
||||||
|
];
|
||||||
|
},
|
||||||
|
|
||||||
|
secretOptions() {
|
||||||
|
const currentNamespace = this.value.metadata.namespace || 'default';
|
||||||
|
|
||||||
|
return this.allSecrets
|
||||||
|
.filter((s) => s.metadata.namespace === currentNamespace)
|
||||||
|
.map((s) => ({
|
||||||
|
label: s.nameDisplay,
|
||||||
|
value: s.metadata.name
|
||||||
|
}));
|
||||||
|
},
|
||||||
|
|
||||||
|
// Define custom rules for the FormValidation mixin
|
||||||
|
fvExtraRules() {
|
||||||
|
return {
|
||||||
|
nameRequired: (val) => !val ? this.t('validation.required', { key: this.t('harvester.fields.name') }) : undefined,
|
||||||
|
urlRequired: (val) => !val ? this.t('validation.required', { key: this.t('harvester.addons.vmImport.ova.fields.url') }) : undefined,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
isFormValid() {
|
||||||
|
if (!this.fvFormIsValid) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.authMode === 'new') {
|
||||||
|
// At least a username/password OR a CA cert to be provided.
|
||||||
|
// If the user selected "Create New", they likely intend to enter something.
|
||||||
|
if (!this.newUsername && !this.newPassword && !this.newCaCert) return false;
|
||||||
|
} else if (this.authMode === 'existing') {
|
||||||
|
if (!this.value.spec.credentials?.name) return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
watch: {
|
||||||
|
authMode(newMode) {
|
||||||
|
if (newMode === 'existing') {
|
||||||
|
// Bind to value.spec.credentials.name for existing credential
|
||||||
|
// Ensure 'credentials' object exists first when selected
|
||||||
|
if (!this.value.spec.credentials) {
|
||||||
|
this.value.spec.credentials = {
|
||||||
|
name: '',
|
||||||
|
namespace: this.value.metadata.namespace || 'default'
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
methods: {
|
||||||
|
secretRule(val) {
|
||||||
|
return !val ? this.t('validation.required', { key: this.t('harvester.addons.vmImport.fields.selectSecret') }) : undefined;
|
||||||
|
},
|
||||||
|
|
||||||
|
async saveSource(buttonCb) {
|
||||||
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (this.authMode === 'none') {
|
||||||
|
// Clear any credential reference
|
||||||
|
delete this.value.spec.credentials;
|
||||||
|
} else if (this.authMode === 'new') {
|
||||||
|
const secretName = `${ this.value.metadata.name }-creds-${ randomStr(4).toLowerCase() }`;
|
||||||
|
const namespace = this.value.metadata.namespace || 'default';
|
||||||
|
|
||||||
|
const newSecret = await this.$store.dispatch(`${ inStore }/create`, {
|
||||||
|
type: SECRET,
|
||||||
|
metadata: {
|
||||||
|
name: secretName,
|
||||||
|
namespace
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
newSecret['_type'] = 'Opaque';
|
||||||
|
newSecret['data'] = {
|
||||||
|
// Optional fields logic
|
||||||
|
username: this.newUsername ? btoa(this.newUsername) : undefined,
|
||||||
|
password: this.newPassword ? btoa(this.newPassword) : undefined,
|
||||||
|
// vm-import-controller code specifies "ca.crt" with a dot.
|
||||||
|
'ca.crt': this.newCaCert ? btoa(this.newCaCert) : undefined
|
||||||
|
};
|
||||||
|
|
||||||
|
await newSecret.save();
|
||||||
|
|
||||||
|
this.value.spec.credentials = {
|
||||||
|
name: secretName,
|
||||||
|
namespace
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
await this.save(buttonCb);
|
||||||
|
} catch (err) {
|
||||||
|
this.errors = [err];
|
||||||
|
buttonCb(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<CruResource
|
||||||
|
:done-route="doneRoute"
|
||||||
|
:resource="value"
|
||||||
|
:mode="mode"
|
||||||
|
:errors="errors"
|
||||||
|
:apply-hooks="applyHooks"
|
||||||
|
:validation-passed="isFormValid"
|
||||||
|
@finish="saveSource"
|
||||||
|
@error="e=>errors=e"
|
||||||
|
>
|
||||||
|
<NameNsDescription
|
||||||
|
:value="value"
|
||||||
|
:mode="mode"
|
||||||
|
:rules="{ name: fvGetAndReportPathRules('metadata.name') }"
|
||||||
|
@update:value="$emit('update:value', $event)"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Tabbed
|
||||||
|
v-bind="$attrs"
|
||||||
|
class="mt-15"
|
||||||
|
:side-tabs="true"
|
||||||
|
>
|
||||||
|
<Tab
|
||||||
|
name="basic"
|
||||||
|
:label="t('harvester.addons.vmImport.titles.basic')"
|
||||||
|
:weight="3"
|
||||||
|
>
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-12">
|
||||||
|
<LabeledInput
|
||||||
|
v-model:value="value.spec.url"
|
||||||
|
:label="t('harvester.addons.vmImport.ova.fields.url')"
|
||||||
|
:placeholder="t('harvester.addons.vmImport.ova.placeholders.url')"
|
||||||
|
tooltip="Supports HTTP and HTTPS protocols."
|
||||||
|
:mode="mode"
|
||||||
|
:rules="fvGetAndReportPathRules('spec.url')"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Tab>
|
||||||
|
|
||||||
|
<Tab
|
||||||
|
name="auth"
|
||||||
|
:label="t('harvester.addons.vmImport.titles.auth')"
|
||||||
|
:weight="2"
|
||||||
|
>
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-12">
|
||||||
|
<RadioGroup
|
||||||
|
v-model:value="authMode"
|
||||||
|
name="authMode"
|
||||||
|
:options="authModeOptions"
|
||||||
|
:mode="mode"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-if="authMode === 'new'">
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-6">
|
||||||
|
<LabeledInput
|
||||||
|
v-model:value="newUsername"
|
||||||
|
:label="t('harvester.addons.vmImport.fields.username')"
|
||||||
|
placeholder="(Optional)"
|
||||||
|
:mode="mode"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="col span-6">
|
||||||
|
<LabeledInput
|
||||||
|
v-model:value="newPassword"
|
||||||
|
type="password"
|
||||||
|
:label="t('harvester.addons.vmImport.fields.password')"
|
||||||
|
placeholder="(Optional)"
|
||||||
|
:mode="mode"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-12">
|
||||||
|
<LabeledInput
|
||||||
|
v-model:value="newCaCert"
|
||||||
|
type="multiline"
|
||||||
|
:label="t('harvester.addons.vmImport.fields.caCert')"
|
||||||
|
:placeholder="t('harvester.addons.vmImport.placeholders.caCert')"
|
||||||
|
:min-height="100"
|
||||||
|
:mode="mode"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-if="authMode === 'existing'">
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-6">
|
||||||
|
<LabeledSelect
|
||||||
|
v-model:value="value.spec.credentials.name"
|
||||||
|
:options="secretOptions"
|
||||||
|
:label="t('harvester.addons.vmImport.fields.selectSecret')"
|
||||||
|
:mode="mode"
|
||||||
|
:rules="[secretRule]"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Tab>
|
||||||
|
|
||||||
|
<Tab
|
||||||
|
name="advanced"
|
||||||
|
:label="t('harvester.addons.vmImport.titles.advanced')"
|
||||||
|
:weight="1"
|
||||||
|
>
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-12">
|
||||||
|
<UnitInput
|
||||||
|
v-model:value="value.spec.httpTimeoutSeconds"
|
||||||
|
:label="t('harvester.addons.vmImport.ova.fields.httpTimeout')"
|
||||||
|
:placeholder="t('harvester.addons.vmImport.ova.placeholders.httpTimeout')"
|
||||||
|
suffix="Seconds"
|
||||||
|
:mode="mode"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Tab>
|
||||||
|
</Tabbed>
|
||||||
|
</CruResource>
|
||||||
|
</template>
|
||||||
@ -0,0 +1,570 @@
|
|||||||
|
<script>
|
||||||
|
import CruResource from '@shell/components/CruResource';
|
||||||
|
import Tabbed from '@shell/components/Tabbed';
|
||||||
|
import Tab from '@shell/components/Tabbed/Tab';
|
||||||
|
import { LabeledInput } from '@components/Form/LabeledInput';
|
||||||
|
import LabeledSelect from '@shell/components/form/LabeledSelect';
|
||||||
|
import NameNsDescription from '@shell/components/form/NameNsDescription';
|
||||||
|
import { Checkbox } from '@components/Form/Checkbox';
|
||||||
|
import CreateEditView from '@shell/mixins/create-edit-view';
|
||||||
|
import FormValidation from '@shell/mixins/form-validation';
|
||||||
|
import { STORAGE_CLASS, NETWORK_ATTACHMENT } from '@shell/config/types';
|
||||||
|
import { allHash } from '@shell/utils/promise';
|
||||||
|
import { MANAGEMENT_NETWORK } from '../mixins/harvester-vm';
|
||||||
|
import { VMIMPORT_SOURCE_PROVIDER, VMIMPORT_SOURCE_KINDS } from '../config/types';
|
||||||
|
import { HCI } from '../types';
|
||||||
|
import { isValidDNSLabelName } from '@pkg/utils/regular';
|
||||||
|
import { mapGetters } from 'vuex';
|
||||||
|
|
||||||
|
// Full API types for the fetch dispatch
|
||||||
|
const VMWARE_SOURCE_TYPE = `${ HCI.MIGRATION }.${ VMIMPORT_SOURCE_KINDS.VMWARE.toLowerCase() }`;
|
||||||
|
const OPENSTACK_SOURCE_TYPE = `${ HCI.MIGRATION }.${ VMIMPORT_SOURCE_KINDS.OPENSTACK.toLowerCase() }`;
|
||||||
|
const OVA_SOURCE_TYPE = `${ HCI.MIGRATION }.${ VMIMPORT_SOURCE_KINDS.OVA.toLowerCase() }`;
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'EditVirtualMachineImport',
|
||||||
|
|
||||||
|
components: {
|
||||||
|
CruResource,
|
||||||
|
Tabbed,
|
||||||
|
Tab,
|
||||||
|
LabeledInput,
|
||||||
|
LabeledSelect,
|
||||||
|
NameNsDescription,
|
||||||
|
Checkbox
|
||||||
|
},
|
||||||
|
|
||||||
|
mixins: [CreateEditView, FormValidation],
|
||||||
|
|
||||||
|
inheritAttrs: false,
|
||||||
|
|
||||||
|
props: {
|
||||||
|
value: {
|
||||||
|
type: Object,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
mode: {
|
||||||
|
type: String,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
async fetch() {
|
||||||
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
|
|
||||||
|
// Fetch all dependencies in parallel to speed up the page load
|
||||||
|
const hash = {
|
||||||
|
storageClasses: this.$store.dispatch(`${ inStore }/findAll`, { type: STORAGE_CLASS }),
|
||||||
|
networks: this.$store.dispatch(`${ inStore }/findAll`, { type: NETWORK_ATTACHMENT }),
|
||||||
|
vmwareSources: this.$store.dispatch(`${ inStore }/findAll`, { type: VMWARE_SOURCE_TYPE }),
|
||||||
|
openstackSources: this.$store.dispatch(`${ inStore }/findAll`, { type: OPENSTACK_SOURCE_TYPE }),
|
||||||
|
ovaSources: this.$store.dispatch(`${ inStore }/findAll`, { type: OVA_SOURCE_TYPE }).catch(() => []),
|
||||||
|
};
|
||||||
|
|
||||||
|
const res = await allHash(hash);
|
||||||
|
|
||||||
|
this.allStorageClasses = res.storageClasses;
|
||||||
|
this.allNetworks = res.networks;
|
||||||
|
this.vmwareSources = res.vmwareSources;
|
||||||
|
this.openstackSources = res.openstackSources;
|
||||||
|
this.ovaSources = res.ovaSources;
|
||||||
|
},
|
||||||
|
|
||||||
|
data() {
|
||||||
|
// Ensure the spec object exists to prevent 'undefined' errors during rendering
|
||||||
|
if (!this.value.spec) this.value.spec = {};
|
||||||
|
if (!this.value.spec.sourceCluster) this.value.spec.sourceCluster = {};
|
||||||
|
if (!this.value.spec.networkMapping) this.value.spec.networkMapping = [];
|
||||||
|
|
||||||
|
// Detect if in Edit mode by checking the existing kind
|
||||||
|
// This allows to pre-select the correct Provider Type tab
|
||||||
|
let initialProvider = '';
|
||||||
|
const existingKind = this.value.spec.sourceCluster.kind;
|
||||||
|
|
||||||
|
if (existingKind === VMIMPORT_SOURCE_KINDS.VMWARE) initialProvider = VMIMPORT_SOURCE_PROVIDER.VMWARE;
|
||||||
|
else if (existingKind === VMIMPORT_SOURCE_KINDS.OPENSTACK) initialProvider = VMIMPORT_SOURCE_PROVIDER.OPENSTACK;
|
||||||
|
else if (existingKind === VMIMPORT_SOURCE_KINDS.OVA) initialProvider = VMIMPORT_SOURCE_PROVIDER.OVA;
|
||||||
|
|
||||||
|
// Construct the unique key (Kind/Namespace/Name) if we are editing an existing resource
|
||||||
|
let initialSourceKey = null;
|
||||||
|
|
||||||
|
if (this.value.spec.sourceCluster.name) {
|
||||||
|
initialSourceKey = `${ existingKind }/${ this.value.spec.sourceCluster.namespace }/${ this.value.spec.sourceCluster.name }`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
allStorageClasses: [],
|
||||||
|
allNetworks: [],
|
||||||
|
vmwareSources: [],
|
||||||
|
openstackSources: [],
|
||||||
|
ovaSources: [],
|
||||||
|
|
||||||
|
// UI State
|
||||||
|
sourceProviderType: initialProvider,
|
||||||
|
selectedSourceKey: initialSourceKey,
|
||||||
|
|
||||||
|
// Static Options
|
||||||
|
providerTypeOptions: [
|
||||||
|
{ label: 'VMware', value: VMIMPORT_SOURCE_PROVIDER.VMWARE },
|
||||||
|
{ label: 'OpenStack', value: VMIMPORT_SOURCE_PROVIDER.OPENSTACK },
|
||||||
|
{ label: 'OVA', value: VMIMPORT_SOURCE_PROVIDER.OVA }
|
||||||
|
],
|
||||||
|
diskBusOptions: [
|
||||||
|
// Allow resetting selection / reset to the default behavior (sending null/empty)
|
||||||
|
{ label: this.t('harvester.addons.vmImport.options.useDefault'), value: '' },
|
||||||
|
{ label: 'VirtIO', value: 'virtio' },
|
||||||
|
{ label: 'SCSI', value: 'scsi' },
|
||||||
|
{ label: 'SATA', value: 'sata' },
|
||||||
|
{ label: 'USB', value: 'usb' },
|
||||||
|
],
|
||||||
|
interfaceModelOptions: [
|
||||||
|
// Allow resetting selection / reset to the default behavior (sending null/empty)
|
||||||
|
{ label: this.t('harvester.addons.vmImport.options.useDefault'), value: '' },
|
||||||
|
{ label: 'VirtIO', value: 'virtio' },
|
||||||
|
{ label: 'e1000', value: 'e1000' },
|
||||||
|
{ label: 'e1000e', value: 'e1000e' },
|
||||||
|
{ label: 'ne2k_pci', value: 'ne2k_pci' },
|
||||||
|
{ label: 'pcnet', value: 'pcnet' },
|
||||||
|
{ label: 'rtl8139', value: 'rtl8139' },
|
||||||
|
],
|
||||||
|
|
||||||
|
fvFormRuleSets: [
|
||||||
|
{ path: 'metadata.name', rules: ['nameRequired'] },
|
||||||
|
{ path: 'spec.virtualMachineName', rules: ['vmNameRequired', 'rfc1123'] },
|
||||||
|
],
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
created() {
|
||||||
|
if (this.registerBeforeHook) {
|
||||||
|
this.registerBeforeHook(this.updateBeforeSave);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
computed: {
|
||||||
|
...mapGetters({ t: 'i18n/t' }),
|
||||||
|
|
||||||
|
// Return only the sources that match the selected Provider Type (VMware or OpenStack)
|
||||||
|
sourceOptions() {
|
||||||
|
let list = [];
|
||||||
|
|
||||||
|
if (this.sourceProviderType === VMIMPORT_SOURCE_PROVIDER.VMWARE) {
|
||||||
|
list = this.vmwareSources;
|
||||||
|
} else if (this.sourceProviderType === VMIMPORT_SOURCE_PROVIDER.OPENSTACK) {
|
||||||
|
list = this.openstackSources;
|
||||||
|
} else if (this.sourceProviderType === VMIMPORT_SOURCE_PROVIDER.OVA) {
|
||||||
|
list = this.ovaSources;
|
||||||
|
}
|
||||||
|
|
||||||
|
return list.map((s) => {
|
||||||
|
// Fallback for API version/kind if missing on the object
|
||||||
|
let kind = s.kind;
|
||||||
|
|
||||||
|
if (!kind) {
|
||||||
|
if (this.sourceProviderType === VMIMPORT_SOURCE_PROVIDER.VMWARE) kind = VMIMPORT_SOURCE_KINDS.VMWARE;
|
||||||
|
else if (this.sourceProviderType === VMIMPORT_SOURCE_PROVIDER.OPENSTACK) kind = VMIMPORT_SOURCE_KINDS.OPENSTACK;
|
||||||
|
else if (this.sourceProviderType === VMIMPORT_SOURCE_PROVIDER.OVA) kind = VMIMPORT_SOURCE_KINDS.OVA;
|
||||||
|
}
|
||||||
|
|
||||||
|
const apiVersion = s.apiVersion || `${ HCI.MIGRATION }/v1beta1`;
|
||||||
|
|
||||||
|
return {
|
||||||
|
label: s.metadata.name,
|
||||||
|
value: `${ kind }/${ s.metadata.namespace }/${ s.metadata.name }`,
|
||||||
|
// We attach the raw metadata so we can easily populate the spec later without re-finding the object
|
||||||
|
raw: {
|
||||||
|
kind,
|
||||||
|
apiVersion,
|
||||||
|
name: s.metadata.name,
|
||||||
|
namespace: s.metadata.namespace
|
||||||
|
}
|
||||||
|
};
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
fvExtraRules() {
|
||||||
|
return {
|
||||||
|
nameRequired: (val) => !val ? this.t('validation.required', { key: this.t('harvester.fields.name') }) : undefined,
|
||||||
|
vmNameRequired: (val) => !val ? this.t('validation.required', { key: this.t('harvester.addons.vmImport.fields.vmName') }) : undefined,
|
||||||
|
rfc1123:
|
||||||
|
(val) => {
|
||||||
|
if (val && !isValidDNSLabelName(val)) {
|
||||||
|
return this.t('harvester.addons.vmImport.errors.rfc1123');
|
||||||
|
}
|
||||||
|
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
// Perform various form validations before allowing to submit
|
||||||
|
isFormValid() {
|
||||||
|
// Check VM Name is valid
|
||||||
|
const nameError = this.fvNameRule(this.value.spec.virtualMachineName);
|
||||||
|
|
||||||
|
if (nameError) return false;
|
||||||
|
|
||||||
|
// Check mandatory fields in Basics
|
||||||
|
if (!this.value.spec.virtualMachineName) return false;
|
||||||
|
if (!this.selectedSourceKey) return false;
|
||||||
|
|
||||||
|
// Check Network Mappings
|
||||||
|
// If any row is missing source or destination, the form is invalid.
|
||||||
|
const networks = this.value.spec.networkMapping || [];
|
||||||
|
const hasInvalidRow = networks.some((row) => !row.sourceNetwork || !row.destinationNetwork);
|
||||||
|
|
||||||
|
if (hasInvalidRow) return false;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
},
|
||||||
|
|
||||||
|
isNetworkTabInvalid() {
|
||||||
|
const networks = this.value.spec.networkMapping || [];
|
||||||
|
// Only error if a row exists AND it is missing fields
|
||||||
|
|
||||||
|
return networks.some((row) => !row.sourceNetwork || !row.destinationNetwork);
|
||||||
|
},
|
||||||
|
|
||||||
|
// Filter out internal storage classes
|
||||||
|
// to prevent selecting a class that might cause the import to fail
|
||||||
|
storageClassOptions() {
|
||||||
|
return this.allStorageClasses
|
||||||
|
.filter((sc) => {
|
||||||
|
const isInternal = sc.parameters?.['harvesterhci.io/isInternalStorageClass'] === 'true';
|
||||||
|
|
||||||
|
return !isInternal;
|
||||||
|
})
|
||||||
|
.map((sc) => ({
|
||||||
|
label: sc.nameDisplay,
|
||||||
|
value: sc.id
|
||||||
|
}));
|
||||||
|
},
|
||||||
|
|
||||||
|
networkOptions() {
|
||||||
|
const mgmtOption = {
|
||||||
|
label: 'Management Network',
|
||||||
|
value: MANAGEMENT_NETWORK
|
||||||
|
};
|
||||||
|
|
||||||
|
const vlanOptions = this.allNetworks.map((n) => ({
|
||||||
|
label: n.nameDisplay || n.metadata.name,
|
||||||
|
value: n.id
|
||||||
|
}));
|
||||||
|
|
||||||
|
return [mgmtOption, ...vlanOptions];
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
methods: {
|
||||||
|
// Clear the selected cluster if the user switches providers (e.g. VMware -> OpenStack)
|
||||||
|
// Prevents submitting a VMware cluster name while the kind is OpenStack
|
||||||
|
onProviderTypeChange(newType) {
|
||||||
|
this.selectedSourceKey = null;
|
||||||
|
this.value.spec.sourceCluster = {};
|
||||||
|
},
|
||||||
|
|
||||||
|
// Update the sourceCluster object based on the single dropdown selection
|
||||||
|
updateSource(key) {
|
||||||
|
this.selectedSourceKey = key;
|
||||||
|
const selectedOption = this.sourceOptions.find((o) => o.value === key);
|
||||||
|
|
||||||
|
if (selectedOption) {
|
||||||
|
const {
|
||||||
|
kind, apiVersion, name, namespace
|
||||||
|
} = selectedOption.raw;
|
||||||
|
|
||||||
|
this.value.spec.sourceCluster = {
|
||||||
|
kind,
|
||||||
|
apiVersion,
|
||||||
|
name,
|
||||||
|
namespace
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
this.value.spec.sourceCluster = {};
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
addNetworkMapping() {
|
||||||
|
this.value.spec.networkMapping.push({
|
||||||
|
sourceNetwork: '',
|
||||||
|
destinationNetwork: '',
|
||||||
|
networkInterfaceModel: ''
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
removeNetworkMapping(index) {
|
||||||
|
if (!this.value?.spec?.networkMapping) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (index >= 0 && index < this.value.spec.networkMapping.length) {
|
||||||
|
this.value.spec.networkMapping.splice(index, 1);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
requiredRule(val) {
|
||||||
|
if (!val) {
|
||||||
|
return this.t('validation.required', { key: this.t('generic.value') });
|
||||||
|
}
|
||||||
|
|
||||||
|
return undefined;
|
||||||
|
},
|
||||||
|
|
||||||
|
// Validates that the input follows Kubernetes Naming Rules (RFC 1123).
|
||||||
|
// If the source VM has uppercase letters or spaces, the user must be warned
|
||||||
|
// that they cannot import it until they rename it on the source. See:
|
||||||
|
// https://docs.harvesterhci.io/v1.6/advanced/addons/vmimport/#source-virtual-machine-name-is-not-rfc1123-compliant
|
||||||
|
fvNameRule(val) {
|
||||||
|
if (!val) return undefined; // 'Required' check handles empty state separately
|
||||||
|
|
||||||
|
// valid RFC 1123
|
||||||
|
if (!isValidDNSLabelName(val)) {
|
||||||
|
return this.t('harvester.addons.vmImport.errors.rfc1123');
|
||||||
|
}
|
||||||
|
|
||||||
|
return undefined;
|
||||||
|
},
|
||||||
|
|
||||||
|
updateBeforeSave() {
|
||||||
|
// If networkMapping exists, filter out the "Management Network" rows
|
||||||
|
// Let the vm-import-controller set the default network mapping
|
||||||
|
if (this.value.spec.networkMapping) {
|
||||||
|
this.value.spec.networkMapping = this.value.spec.networkMapping.filter((row) => {
|
||||||
|
return row.destinationNetwork !== MANAGEMENT_NETWORK;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
// Only handles complex logic that doesn't fit into simple field rules
|
||||||
|
async saveOverride(buttonCb) {
|
||||||
|
const errors = [];
|
||||||
|
|
||||||
|
this.errors = [];
|
||||||
|
|
||||||
|
// Validate Provider Type
|
||||||
|
if (!this.sourceProviderType) {
|
||||||
|
errors.push(this.t('validation.required', { key: this.t('harvester.addons.vmImport.fields.sourceProvider') }));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Validate Network Tab
|
||||||
|
if (this.isNetworkTabInvalid) {
|
||||||
|
errors.push(this.t('harvester.addons.vmImport.errors.networkMappingRequired'));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Return immediately in case of an error, avoid that `this.save()` runs, preventing `updateBeforeSave` from resetting data.
|
||||||
|
if (errors.length > 0) {
|
||||||
|
this.errors = errors;
|
||||||
|
buttonCb(false);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Only proceed if valid
|
||||||
|
this.save(buttonCb);
|
||||||
|
},
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<CruResource
|
||||||
|
:done-route="doneRoute"
|
||||||
|
:resource="value"
|
||||||
|
:mode="mode"
|
||||||
|
:errors="errors"
|
||||||
|
:apply-hooks="applyHooks"
|
||||||
|
:validation-passed="fvFormIsValid"
|
||||||
|
@finish="saveOverride"
|
||||||
|
@error="e=>errors=e"
|
||||||
|
>
|
||||||
|
<NameNsDescription
|
||||||
|
:value="value"
|
||||||
|
:mode="mode"
|
||||||
|
:rules="{ name: fvGetAndReportPathRules('metadata.name') }"
|
||||||
|
@update:value="$emit('update:value', $event)"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Tabbed
|
||||||
|
v-bind="$attrs"
|
||||||
|
class="mt-15"
|
||||||
|
:side-tabs="true"
|
||||||
|
>
|
||||||
|
<Tab
|
||||||
|
name="basic"
|
||||||
|
:label="t('harvester.addons.vmImport.titles.basic')"
|
||||||
|
:weight="3"
|
||||||
|
>
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-6">
|
||||||
|
<LabeledSelect
|
||||||
|
v-model:value="sourceProviderType"
|
||||||
|
:options="providerTypeOptions"
|
||||||
|
:label="t('harvester.addons.vmImport.fields.sourceProvider')"
|
||||||
|
:mode="mode"
|
||||||
|
:rules="[requiredRule]"
|
||||||
|
required
|
||||||
|
@update:value="onProviderTypeChange"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="col span-6">
|
||||||
|
<LabeledSelect
|
||||||
|
:value="selectedSourceKey"
|
||||||
|
:options="sourceOptions"
|
||||||
|
:label="t('harvester.addons.vmImport.fields.sourceCluster')"
|
||||||
|
:placeholder="sourceProviderType ? t('harvester.addons.vmImport.placeholders.selectCluster') : t('harvester.addons.vmImport.placeholders.selectProviderFirst')"
|
||||||
|
:disabled="!sourceProviderType"
|
||||||
|
:mode="mode"
|
||||||
|
:rules="fvGetAndReportPathRules('selectedSourceKey')"
|
||||||
|
required
|
||||||
|
@update:value="updateSource"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-6">
|
||||||
|
<LabeledInput
|
||||||
|
v-model:value="value.spec.virtualMachineName"
|
||||||
|
:label="t('harvester.addons.vmImport.fields.vmName')"
|
||||||
|
:placeholder="t('harvester.addons.vmImport.placeholders.matchSource')"
|
||||||
|
:mode="mode"
|
||||||
|
:rules="fvGetAndReportPathRules('spec.virtualMachineName')"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="col span-6">
|
||||||
|
<LabeledSelect
|
||||||
|
v-model:value="value.spec.storageClass"
|
||||||
|
:options="storageClassOptions"
|
||||||
|
:label="t('harvester.addons.vmImport.fields.targetStorageClass')"
|
||||||
|
:mode="mode"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Tab>
|
||||||
|
|
||||||
|
<Tab
|
||||||
|
name="networking"
|
||||||
|
:label="t('harvester.addons.vmImport.titles.networking')"
|
||||||
|
:weight="2"
|
||||||
|
:error="isNetworkTabInvalid"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
v-for="(row, i) in value.spec.networkMapping"
|
||||||
|
:key="i"
|
||||||
|
class="network-row box mb-10"
|
||||||
|
>
|
||||||
|
<div class="row">
|
||||||
|
<div class="col span-4">
|
||||||
|
<LabeledInput
|
||||||
|
v-model:value="row.sourceNetwork"
|
||||||
|
:label="t('harvester.addons.vmImport.fields.sourceNetwork')"
|
||||||
|
:mode="mode"
|
||||||
|
:rules="[requiredRule]"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="col span-4">
|
||||||
|
<LabeledSelect
|
||||||
|
v-model:value="row.destinationNetwork"
|
||||||
|
:options="networkOptions"
|
||||||
|
:label="t('harvester.addons.vmImport.fields.destNetwork')"
|
||||||
|
:mode="mode"
|
||||||
|
:rules="[requiredRule]"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="col span-3">
|
||||||
|
<LabeledSelect
|
||||||
|
v-model:value="row.networkInterfaceModel"
|
||||||
|
:options="interfaceModelOptions"
|
||||||
|
:label="t('harvester.addons.vmImport.fields.interfaceModel')"
|
||||||
|
:mode="mode"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="col span-1 remove-btn-container">
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="btn role-link"
|
||||||
|
@click="removeNetworkMapping(i)"
|
||||||
|
>
|
||||||
|
{{ t('harvester.addons.vmImport.actions.remove') }}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="btn role-secondary"
|
||||||
|
@click="addNetworkMapping"
|
||||||
|
>
|
||||||
|
{{ t('harvester.addons.vmImport.actions.addNetwork') }}
|
||||||
|
</button>
|
||||||
|
</Tab>
|
||||||
|
|
||||||
|
<Tab
|
||||||
|
name="advanced"
|
||||||
|
:label="t('harvester.addons.vmImport.titles.advanced')"
|
||||||
|
:weight="1"
|
||||||
|
>
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-6">
|
||||||
|
<LabeledInput
|
||||||
|
v-model:value="value.spec.folder"
|
||||||
|
:label="t('harvester.addons.vmImport.fields.folder')"
|
||||||
|
:placeholder="t('harvester.addons.vmImport.placeholders.folderExample')"
|
||||||
|
:mode="mode"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-6">
|
||||||
|
<LabeledSelect
|
||||||
|
v-model:value="value.spec.defaultDiskBusType"
|
||||||
|
:options="diskBusOptions"
|
||||||
|
:label="t('harvester.addons.vmImport.fields.diskBus')"
|
||||||
|
:mode="mode"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="col span-6">
|
||||||
|
<LabeledSelect
|
||||||
|
v-model:value="value.spec.defaultNetworkInterfaceModel"
|
||||||
|
:options="interfaceModelOptions"
|
||||||
|
:label="t('harvester.addons.vmImport.fields.defaultInterface')"
|
||||||
|
:mode="mode"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="row">
|
||||||
|
<div class="col span-12">
|
||||||
|
<Checkbox
|
||||||
|
v-model:value="value.spec.skipPreflightChecks"
|
||||||
|
:label="t('harvester.addons.vmImport.fields.skipPreflight')"
|
||||||
|
:mode="mode"
|
||||||
|
/>
|
||||||
|
<Checkbox
|
||||||
|
v-model:value="value.spec.forcePowerOff"
|
||||||
|
:label="t('harvester.addons.vmImport.fields.forcePowerOff')"
|
||||||
|
:mode="mode"
|
||||||
|
class="mt-10"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Tab>
|
||||||
|
</Tabbed>
|
||||||
|
</CruResource>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.network-row {
|
||||||
|
border: 1px solid var(--border);
|
||||||
|
padding: 10px;
|
||||||
|
border-radius: var(--border-radius);
|
||||||
|
background: var(--body-bg);
|
||||||
|
}
|
||||||
|
.remove-btn-container {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
302
pkg/harvester/edit/migration.harvesterhci.io.vmwaresource.vue
Normal file
302
pkg/harvester/edit/migration.harvesterhci.io.vmwaresource.vue
Normal file
@ -0,0 +1,302 @@
|
|||||||
|
<script>
|
||||||
|
import CruResource from '@shell/components/CruResource';
|
||||||
|
import Tabbed from '@shell/components/Tabbed';
|
||||||
|
import Tab from '@shell/components/Tabbed/Tab';
|
||||||
|
import { LabeledInput } from '@components/Form/LabeledInput';
|
||||||
|
import LabeledSelect from '@shell/components/form/LabeledSelect';
|
||||||
|
import NameNsDescription from '@shell/components/form/NameNsDescription';
|
||||||
|
import { RadioGroup } from '@components/Form/Radio';
|
||||||
|
import CreateEditView from '@shell/mixins/create-edit-view';
|
||||||
|
import FormValidation from '@shell/mixins/form-validation';
|
||||||
|
import { SECRET } from '@shell/config/types';
|
||||||
|
import { randomStr } from '@shell/utils/string';
|
||||||
|
import { mapGetters } from 'vuex';
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'EditVmwareSource',
|
||||||
|
|
||||||
|
// Declare the event, fixes a console warning
|
||||||
|
emits: ['update:value'],
|
||||||
|
|
||||||
|
components: {
|
||||||
|
CruResource,
|
||||||
|
Tabbed,
|
||||||
|
Tab,
|
||||||
|
LabeledInput,
|
||||||
|
LabeledSelect,
|
||||||
|
NameNsDescription,
|
||||||
|
RadioGroup,
|
||||||
|
},
|
||||||
|
|
||||||
|
mixins: [CreateEditView, FormValidation],
|
||||||
|
|
||||||
|
inheritAttrs: false,
|
||||||
|
|
||||||
|
props: {
|
||||||
|
value: {
|
||||||
|
type: Object,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
mode: {
|
||||||
|
type: String,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
async fetch() {
|
||||||
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
|
|
||||||
|
this.allSecrets = await this.$store.dispatch(`${ inStore }/findAll`, { type: SECRET });
|
||||||
|
},
|
||||||
|
|
||||||
|
data() {
|
||||||
|
if (!this.value.spec) this.value.spec = {};
|
||||||
|
if (!this.value.spec.credentials) this.value.spec.credentials = {};
|
||||||
|
|
||||||
|
const initialMode = this.value.spec.credentials.name ? 'existing' : 'new';
|
||||||
|
|
||||||
|
return {
|
||||||
|
allSecrets: [],
|
||||||
|
authMode: initialMode,
|
||||||
|
newUsername: '',
|
||||||
|
newPassword: '',
|
||||||
|
newCaCert: '',
|
||||||
|
|
||||||
|
fvFormRuleSets: [
|
||||||
|
{ path: 'metadata.name', rules: ['nameRequired'] },
|
||||||
|
{ path: 'spec.endpoint', rules: ['endpointRequired'] },
|
||||||
|
{ path: 'spec.dc', rules: ['dcRequired'] },
|
||||||
|
],
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
computed: {
|
||||||
|
...mapGetters({ t: 'i18n/t' }),
|
||||||
|
|
||||||
|
authModeOptions() {
|
||||||
|
return [
|
||||||
|
{ label: this.t('harvester.addons.vmImport.fields.createSecret'), value: 'new' },
|
||||||
|
{ label: this.t('harvester.addons.vmImport.fields.useSecret'), value: 'existing' }
|
||||||
|
];
|
||||||
|
},
|
||||||
|
|
||||||
|
secretOptions() {
|
||||||
|
const currentNamespace = this.value.metadata.namespace || 'default';
|
||||||
|
|
||||||
|
return this.allSecrets
|
||||||
|
.filter((s) => s.metadata.namespace === currentNamespace)
|
||||||
|
.map((s) => ({
|
||||||
|
label: s.nameDisplay,
|
||||||
|
value: s.metadata.name
|
||||||
|
}));
|
||||||
|
},
|
||||||
|
|
||||||
|
// Define custom rules for the FormValidation mixin
|
||||||
|
fvExtraRules() {
|
||||||
|
return {
|
||||||
|
nameRequired: (val) => !val ? this.t('validation.required', { key: this.t('harvester.fields.name') }) : undefined,
|
||||||
|
endpointRequired: (val) => !val ? this.t('validation.required', { key: this.t('harvester.addons.vmImport.vmware.fields.endpoint') }) : undefined,
|
||||||
|
dcRequired: (val) => !val ? this.t('validation.required', { key: this.t('harvester.addons.vmImport.vmware.fields.datacenter') }) : undefined,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
isFormValid() {
|
||||||
|
if (!this.fvFormIsValid) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.authMode === 'new') {
|
||||||
|
if (!this.newUsername || !this.newPassword) return false;
|
||||||
|
} else {
|
||||||
|
if (!this.value.spec.credentials.name) return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
methods: {
|
||||||
|
usernameRule(val) {
|
||||||
|
return !val ? this.t('validation.required', { key: this.t('harvester.addons.vmImport.fields.username') }) : undefined;
|
||||||
|
},
|
||||||
|
passwordRule(val) {
|
||||||
|
return !val ? this.t('validation.required', { key: this.t('harvester.addons.vmImport.fields.password') }) : undefined;
|
||||||
|
},
|
||||||
|
secretRule(val) {
|
||||||
|
return !val ? this.t('validation.required', { key: this.t('harvester.addons.vmImport.fields.selectSecret') }) : undefined;
|
||||||
|
},
|
||||||
|
|
||||||
|
async saveSource(buttonCb) {
|
||||||
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (this.authMode === 'new') {
|
||||||
|
const secretName = `${ this.value.metadata.name }-creds-${ randomStr(4).toLowerCase() }`;
|
||||||
|
const namespace = this.value.metadata.namespace || 'default';
|
||||||
|
|
||||||
|
// Create the model with the correct Schema ID (SECRET)
|
||||||
|
const newSecret = await this.$store.dispatch(`${ inStore }/create`, {
|
||||||
|
type: SECRET,
|
||||||
|
metadata: {
|
||||||
|
name: secretName,
|
||||||
|
namespace
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Use '_type' to set the Kubernetes 'type' field.
|
||||||
|
newSecret['_type'] = 'Opaque';
|
||||||
|
|
||||||
|
// base64 encode the data
|
||||||
|
newSecret['data'] = {
|
||||||
|
username: btoa(this.newUsername),
|
||||||
|
password: btoa(this.newPassword),
|
||||||
|
// Only include CA cert if the user provided one
|
||||||
|
caCert: this.newCaCert ? btoa(this.newCaCert) : undefined
|
||||||
|
};
|
||||||
|
|
||||||
|
await newSecret.save();
|
||||||
|
|
||||||
|
// Link the new secret to the Source
|
||||||
|
this.value.spec.credentials = {
|
||||||
|
name: secretName,
|
||||||
|
namespace
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
await this.save(buttonCb);
|
||||||
|
} catch (err) {
|
||||||
|
this.errors = [err];
|
||||||
|
buttonCb(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<CruResource
|
||||||
|
:done-route="doneRoute"
|
||||||
|
:resource="value"
|
||||||
|
:mode="mode"
|
||||||
|
:errors="errors"
|
||||||
|
:apply-hooks="applyHooks"
|
||||||
|
:validation-passed="isFormValid"
|
||||||
|
@finish="saveSource"
|
||||||
|
@error="e=>errors=e"
|
||||||
|
>
|
||||||
|
<NameNsDescription
|
||||||
|
:value="value"
|
||||||
|
:mode="mode"
|
||||||
|
:rules="{ name: fvGetAndReportPathRules('metadata.name') }"
|
||||||
|
@update:value="$emit('update:value', $event)"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Tabbed
|
||||||
|
v-bind="$attrs"
|
||||||
|
class="mt-15"
|
||||||
|
:side-tabs="true"
|
||||||
|
>
|
||||||
|
<Tab
|
||||||
|
name="basic"
|
||||||
|
:label="t('harvester.addons.vmImport.titles.basic')"
|
||||||
|
:weight="2"
|
||||||
|
>
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-6">
|
||||||
|
<LabeledInput
|
||||||
|
v-model:value="value.spec.endpoint"
|
||||||
|
:label="t('harvester.addons.vmImport.vmware.fields.endpoint')"
|
||||||
|
:placeholder="t('harvester.addons.vmImport.vmware.placeholders.endpoint')"
|
||||||
|
:mode="mode"
|
||||||
|
:rules="fvGetAndReportPathRules('spec.endpoint')"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="col span-6">
|
||||||
|
<LabeledInput
|
||||||
|
v-model:value="value.spec.dc"
|
||||||
|
:label="t('harvester.addons.vmImport.vmware.fields.datacenter')"
|
||||||
|
:placeholder="t('harvester.addons.vmImport.vmware.placeholders.datacenter')"
|
||||||
|
:tooltip="t('harvester.addons.vmImport.vmware.tooltips.datacenter')"
|
||||||
|
:mode="mode"
|
||||||
|
:rules="fvGetAndReportPathRules('spec.dc')"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Tab>
|
||||||
|
|
||||||
|
<Tab
|
||||||
|
name="auth"
|
||||||
|
:label="t('harvester.addons.vmImport.titles.auth')"
|
||||||
|
:weight="1"
|
||||||
|
>
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-12">
|
||||||
|
<RadioGroup
|
||||||
|
v-model:value="authMode"
|
||||||
|
name="authMode"
|
||||||
|
:options="authModeOptions"
|
||||||
|
:mode="mode"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-if="authMode === 'new'">
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-6">
|
||||||
|
<LabeledInput
|
||||||
|
v-model:value="newUsername"
|
||||||
|
:label="t('harvester.addons.vmImport.fields.username')"
|
||||||
|
:mode="mode"
|
||||||
|
:rules="[usernameRule]"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="col span-6">
|
||||||
|
<LabeledInput
|
||||||
|
v-model:value="newPassword"
|
||||||
|
type="password"
|
||||||
|
:label="t('harvester.addons.vmImport.fields.password')"
|
||||||
|
:mode="mode"
|
||||||
|
:rules="[passwordRule]"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-12">
|
||||||
|
<LabeledInput
|
||||||
|
v-model:value="newCaCert"
|
||||||
|
type="multiline"
|
||||||
|
:label="t('harvester.addons.vmImport.fields.caCert')"
|
||||||
|
:placeholder="t('harvester.addons.vmImport.placeholders.caCert')"
|
||||||
|
:min-height="100"
|
||||||
|
:mode="mode"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="text-muted">
|
||||||
|
Note: A new Kubernetes Secret will be created to store these credentials.
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-if="authMode === 'existing'">
|
||||||
|
<div class="row mb-20">
|
||||||
|
<div class="col span-6">
|
||||||
|
<LabeledSelect
|
||||||
|
v-model:value="value.spec.credentials.name"
|
||||||
|
:options="secretOptions"
|
||||||
|
:label="t('harvester.addons.vmImport.fields.selectSecret')"
|
||||||
|
:mode="mode"
|
||||||
|
:rules="[secretRule]"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Tab>
|
||||||
|
</Tabbed>
|
||||||
|
</CruResource>
|
||||||
|
</template>
|
||||||
@ -7,6 +7,7 @@ import harvesterStore from './store/harvester-store';
|
|||||||
import customValidators from './validators';
|
import customValidators from './validators';
|
||||||
import { PRODUCT_NAME } from './config/harvester';
|
import { PRODUCT_NAME } from './config/harvester';
|
||||||
import { defineAsyncComponent } from 'vue';
|
import { defineAsyncComponent } from 'vue';
|
||||||
|
import './styles/vue-flow.scss';
|
||||||
|
|
||||||
// Init the package
|
// Init the package
|
||||||
export default function (plugin: IPlugin) {
|
export default function (plugin: IPlugin) {
|
||||||
|
|||||||
@ -20,6 +20,7 @@ nav:
|
|||||||
Monitoring: Monitoring
|
Monitoring: Monitoring
|
||||||
Logging: Logging
|
Logging: Logging
|
||||||
'Monitoring and Logging': Monitoring and Logging
|
'Monitoring and Logging': Monitoring and Logging
|
||||||
|
vmimport: Virtual Machine Imports
|
||||||
|
|
||||||
resourceTable:
|
resourceTable:
|
||||||
groupBy:
|
groupBy:
|
||||||
@ -122,6 +123,15 @@ harvester:
|
|||||||
namespace: Namespace
|
namespace: Namespace
|
||||||
message:
|
message:
|
||||||
success: 'Image { name } created successfully.'
|
success: 'Image { name } created successfully.'
|
||||||
|
storageMigration:
|
||||||
|
title: Storage Migration
|
||||||
|
fields:
|
||||||
|
sourceVolume:
|
||||||
|
label: Source Volume
|
||||||
|
placeholder: Select a source volume
|
||||||
|
targetVolume:
|
||||||
|
label: Target Volume
|
||||||
|
placeholder: Select a target volume
|
||||||
migration:
|
migration:
|
||||||
failedMessage: Latest migration failed!
|
failedMessage: Latest migration failed!
|
||||||
title: Migration
|
title: Migration
|
||||||
@ -154,15 +164,24 @@ harvester:
|
|||||||
nodeTimeout:
|
nodeTimeout:
|
||||||
label: Node Collection Timeout
|
label: Node Collection Timeout
|
||||||
tooltip: Minutes allowed for collecting logs/configurations on nodes.<br/>See docs support-bundle-node-collection-timeout for detail.
|
tooltip: Minutes allowed for collecting logs/configurations on nodes.<br/>See docs support-bundle-node-collection-timeout for detail.
|
||||||
hotplug:
|
hotplugVolume:
|
||||||
success: 'Volume { diskName } is mounted to the virtual machine { vm }.'
|
success: 'Volume { diskName } is mounted to the virtual machine { vm }.'
|
||||||
title: Add Volume
|
title: Add Volume
|
||||||
|
hotplugNic:
|
||||||
|
success: 'The settings have been saved, but the network interface {interfaceName} will be attached only after the virtual machine is migrated.'
|
||||||
|
title: Add Network Interface
|
||||||
|
vmNetwork: Virtual Machine Network
|
||||||
|
macAddress: MAC Address
|
||||||
|
macAddressTooltip: If left blank, the MAC address will be automatically generated.
|
||||||
|
insertCdRomVolume:
|
||||||
|
success: '{ imageName } is inserted into device { deviceName }.'
|
||||||
|
title: Insert Image
|
||||||
|
image: Image
|
||||||
|
actionLabel: Insert Image
|
||||||
cpuMemoryHotplug:
|
cpuMemoryHotplug:
|
||||||
success: 'CPU and Memory are updated to the virtual machine { vm }.'
|
success: 'CPU and Memory are updated to the virtual machine { vm }.'
|
||||||
title: Edit CPU and Memory
|
title: Edit CPU and Memory
|
||||||
maxResourcesMessage: 'You can increase the CPU to maximum { maxCpu }C and memory to maximum { maxMemory }.'
|
maxResourcesMessage: 'You can increase the CPU to maximum { maxCpu }C and memory to maximum { maxMemory }.'
|
||||||
hotunplug:
|
|
||||||
success: 'Volume { name } is detached successfully.'
|
|
||||||
snapshot:
|
snapshot:
|
||||||
title: Take Snapshot
|
title: Take Snapshot
|
||||||
name: Name
|
name: Name
|
||||||
@ -191,6 +210,9 @@ harvester:
|
|||||||
info: Info
|
info: Info
|
||||||
warning: Warning
|
warning: Warning
|
||||||
error: Error
|
error: Error
|
||||||
|
restartRequired:
|
||||||
|
title: '{count} {count, plural, =1 {Virtual Machine is} other {Virtual Machines are}} Pending Restart'
|
||||||
|
message: 'Please restart { vmNames } to apply updated configurations'
|
||||||
action:
|
action:
|
||||||
createVM: Create Virtual Machine
|
createVM: Create Virtual Machine
|
||||||
start: Start
|
start: Start
|
||||||
@ -219,6 +241,8 @@ harvester:
|
|||||||
migrate: Migrate
|
migrate: Migrate
|
||||||
cpuAndMemoryHotplug: Edit CPU and Memory
|
cpuAndMemoryHotplug: Edit CPU and Memory
|
||||||
abortMigration: Abort Migration
|
abortMigration: Abort Migration
|
||||||
|
storageMigration: Storage Migration
|
||||||
|
cancelStorageMigration: Cancel Storage Migration
|
||||||
createTemplate: Generate Template
|
createTemplate: Generate Template
|
||||||
enableMaintenance: Enable Maintenance Mode
|
enableMaintenance: Enable Maintenance Mode
|
||||||
disableMaintenance: Disable Maintenance Mode
|
disableMaintenance: Disable Maintenance Mode
|
||||||
@ -226,7 +250,8 @@ harvester:
|
|||||||
disableCPUManager: Disable CPU Manager
|
disableCPUManager: Disable CPU Manager
|
||||||
cordon: Cordon
|
cordon: Cordon
|
||||||
uncordon: Uncordon
|
uncordon: Uncordon
|
||||||
addHotplug: Add Volume
|
addHotplugVolume: Add Volume
|
||||||
|
addHotplugNic: Hotplug Network Interface
|
||||||
exportImage: Export Image
|
exportImage: Export Image
|
||||||
viewlogs: View Logs
|
viewlogs: View Logs
|
||||||
cancelExpand: Cancel Expand
|
cancelExpand: Cancel Expand
|
||||||
@ -290,6 +315,17 @@ harvester:
|
|||||||
totalSnapshotQuota: Total Snapshot Quota
|
totalSnapshotQuota: Total Snapshot Quota
|
||||||
storageClass: Storage Class
|
storageClass: Storage Class
|
||||||
restore: Restore
|
restore: Restore
|
||||||
|
vmImportSourceVm: Source VM
|
||||||
|
vmImportSourceCluster: Source Cluster
|
||||||
|
vmImportStatus: Import Status
|
||||||
|
vmImportSourceVDatacenter: Datacenter
|
||||||
|
vmImportSourceVEndpoint: Endpoint
|
||||||
|
vmImportSourceVClusterStatus: Cluster Status
|
||||||
|
vmImportSourceORegion: Region
|
||||||
|
vmImportSourceOEndpoint: Endpoint
|
||||||
|
vmImportSourceOClusterStatus: Cluster Status
|
||||||
|
vmImportSourceOVAUrl: URL
|
||||||
|
vmImportSourceOVAStatus: Status
|
||||||
tab:
|
tab:
|
||||||
volume: Volumes
|
volume: Volumes
|
||||||
network: Networks
|
network: Networks
|
||||||
@ -330,7 +366,10 @@ harvester:
|
|||||||
available: Available Devices
|
available: Available Devices
|
||||||
compatibleNodes: Compatible Nodes
|
compatibleNodes: Compatible Nodes
|
||||||
impossibleSelection: 'There are no hosts with all of the selected devices.'
|
impossibleSelection: 'There are no hosts with all of the selected devices.'
|
||||||
howToUseDevice: 'Use the table below to enable PCI passthrough on each device you want to use in this virtual machine.'
|
howToUseDevice:
|
||||||
|
prefix: 'Use the table below to enable PCI passthrough on each device you want to use in this virtual machine.<br>For vGPU devices, please enable them on the'
|
||||||
|
middle: vGPU Devices
|
||||||
|
suffix: page first.
|
||||||
deviceInTheSameHost: 'You can only select devices on the same host.'
|
deviceInTheSameHost: 'You can only select devices on the same host.'
|
||||||
oldFormatDevices:
|
oldFormatDevices:
|
||||||
help: |-
|
help: |-
|
||||||
@ -352,6 +391,9 @@ harvester:
|
|||||||
claimError: Error enabling passthrough on {name}
|
claimError: Error enabling passthrough on {name}
|
||||||
unclaimError: Error disabling passthrough on {name}
|
unclaimError: Error disabling passthrough on {name}
|
||||||
cantUnclaim: You cannot disable passthrough on a device claimed by another user.
|
cantUnclaim: You cannot disable passthrough on a device claimed by another user.
|
||||||
|
detachWarning:
|
||||||
|
title: Cannot Disable Passthrough
|
||||||
|
message: Please detach the device from the VM and save it first before disabling passthrough.
|
||||||
enableGroup: Enable Group
|
enableGroup: Enable Group
|
||||||
disableGroup: Disable Group
|
disableGroup: Disable Group
|
||||||
labelRequired: "This rule should not be manually altered: it ensures that the PCI devices selected for this virtual machine are available on the virtual machine's host."
|
labelRequired: "This rule should not be manually altered: it ensures that the PCI devices selected for this virtual machine are available on the virtual machine's host."
|
||||||
@ -397,7 +439,7 @@ harvester:
|
|||||||
volume:
|
volume:
|
||||||
upperType: Volume name
|
upperType: Volume name
|
||||||
lowerType: volume name
|
lowerType: volume name
|
||||||
needImageOrExisting: 'At least an image volume or an existing root-disk volume is required!'
|
needAtLeastOneBootable: 'At least one bootable volume is required!'
|
||||||
image:
|
image:
|
||||||
ruleTip: 'The URL you have entered ends in an extension that we do not support. We only accept image files that end in .img, .iso, .qcow, .qcow2, .raw.'
|
ruleTip: 'The URL you have entered ends in an extension that we do not support. We only accept image files that end in .img, .iso, .qcow, .qcow2, .raw.'
|
||||||
ruleFileTip: 'The file you have chosen ends in an extension that we do not support. We only accept image files that end in .img, .iso, .qcow, .qcow2, .raw.'
|
ruleFileTip: 'The file you have chosen ends in an extension that we do not support. We only accept image files that end in .img, .iso, .qcow, .qcow2, .raw.'
|
||||||
@ -442,6 +484,7 @@ harvester:
|
|||||||
storage: Storage
|
storage: Storage
|
||||||
labels: Labels
|
labels: Labels
|
||||||
ksmtuned: Ksmtuned
|
ksmtuned: Ksmtuned
|
||||||
|
hugepages: Hugepages
|
||||||
seeder: Out-of-band Access
|
seeder: Out-of-band Access
|
||||||
detail:
|
detail:
|
||||||
kvm:
|
kvm:
|
||||||
@ -514,6 +557,20 @@ harvester:
|
|||||||
fullScans: Full Scans
|
fullScans: Full Scans
|
||||||
stableNodeChains: Stable Node Chains
|
stableNodeChains: Stable Node Chains
|
||||||
stableNodeDups: Stable Node Dups
|
stableNodeDups: Stable Node Dups
|
||||||
|
hugepages:
|
||||||
|
meminfo: Meminfo
|
||||||
|
transparent:
|
||||||
|
title: Transparent Hugepages <a href="{url}" target="_blank"><i class="icon icon-info" /></a>
|
||||||
|
enabled: Enabled
|
||||||
|
shmemEnabled: Shared Memory Enabled
|
||||||
|
defrag: Defragmentation
|
||||||
|
status:
|
||||||
|
anon: Anonymous Hugepages (bytes)
|
||||||
|
size: Default Hugepage Size (bytes)
|
||||||
|
total: Total Hugepages
|
||||||
|
free: Free Hugepages
|
||||||
|
rsvd: Reserved Hugepages
|
||||||
|
surp: Surplus Hugepages
|
||||||
disk:
|
disk:
|
||||||
add: Add Disk
|
add: Add Disk
|
||||||
path:
|
path:
|
||||||
@ -578,10 +635,28 @@ harvester:
|
|||||||
virtualMachine:
|
virtualMachine:
|
||||||
label: Virtual Machines
|
label: Virtual Machines
|
||||||
osType: OS Type
|
osType: OS Type
|
||||||
|
cpuModel:
|
||||||
|
label: CPU Model
|
||||||
|
fetchError: 'Failed to load CPU model configuration: {error}'
|
||||||
|
optionLabel: "{modelName} ({count} {count, plural, one {node} other {nodes}})"
|
||||||
hotplug:
|
hotplug:
|
||||||
title: Enable CPU and memory hotplug
|
title: Enable CPU and memory hotplug
|
||||||
tooltip: The default maximum CPU and maximum memory are {hotPlugTimes} times based on CPU and memory.
|
tooltip: The default maximum CPU and maximum memory are {hotPlugTimes} times based on CPU and memory.
|
||||||
restartVMMessage: Restart action is required for the virtual machine configuration change to take effect
|
restartVMMessage: Restart action is required for the virtual machine configuration change to take effect
|
||||||
|
hotUnplug:
|
||||||
|
actionLabel: Detach
|
||||||
|
detachVolume:
|
||||||
|
title: 'Are you sure that you want to detach volume {name}?'
|
||||||
|
actionLabel: Detach Volume
|
||||||
|
success: 'Volume { name } is detached successfully.'
|
||||||
|
ejectCdRomVolume:
|
||||||
|
title: 'Are you sure that you want to eject image from device {name}?'
|
||||||
|
actionLabel: Eject Image
|
||||||
|
success: 'Image from device { name } is ejected successfully.'
|
||||||
|
detachNIC:
|
||||||
|
title: 'Are you sure that you want to detach network interface {name}?'
|
||||||
|
actionLabel: Detach Network Interface
|
||||||
|
success: 'The settings have been saved, but the network interface {name} will be detached only after the virtual machine is migrated.'
|
||||||
instance:
|
instance:
|
||||||
singleInstance:
|
singleInstance:
|
||||||
multipleInstance:
|
multipleInstance:
|
||||||
@ -613,11 +688,6 @@ harvester:
|
|||||||
title: 'Select the volume you want to delete:'
|
title: 'Select the volume you want to delete:'
|
||||||
deleteAll: Delete All
|
deleteAll: Delete All
|
||||||
tips: "Warn: The snapshots of the virtual machine will be deleted with virtual machine and the snapshots of volume will be deleted with volume."
|
tips: "Warn: The snapshots of the virtual machine will be deleted with virtual machine and the snapshots of volume will be deleted with volume."
|
||||||
unplug:
|
|
||||||
title: 'Are you sure that you want to detach volume {name} ?'
|
|
||||||
actionLabel: Detach
|
|
||||||
detachVolume:
|
|
||||||
Detach Volume
|
|
||||||
restartTip: |-
|
restartTip: |-
|
||||||
{restart, select,
|
{restart, select,
|
||||||
true {Restart}
|
true {Restart}
|
||||||
@ -689,6 +759,7 @@ harvester:
|
|||||||
unmount:
|
unmount:
|
||||||
title: Are you sure?
|
title: Are you sure?
|
||||||
message: Are you sure you want to unmount this volume?
|
message: Are you sure you want to unmount this volume?
|
||||||
|
emptyImage: No media
|
||||||
network:
|
network:
|
||||||
title: Network
|
title: Network
|
||||||
addNetwork: Add Network
|
addNetwork: Add Network
|
||||||
@ -897,6 +968,8 @@ harvester:
|
|||||||
checksumTip: Validate the image using the SHA512 checksum, if specified.
|
checksumTip: Validate the image using the SHA512 checksum, if specified.
|
||||||
tooltip:
|
tooltip:
|
||||||
imported: Created automatically by the vm-import-controller
|
imported: Created automatically by the vm-import-controller
|
||||||
|
errors:
|
||||||
|
unsupportedBackend: 'Unsupported backend type: {backend}'
|
||||||
|
|
||||||
vmTemplate:
|
vmTemplate:
|
||||||
label: Templates
|
label: Templates
|
||||||
@ -958,7 +1031,9 @@ harvester:
|
|||||||
createTitle: Create Schedule
|
createTitle: Create Schedule
|
||||||
createButtonText: Create Schedule
|
createButtonText: Create Schedule
|
||||||
scheduleType: Virtual Machine Schedule Type
|
scheduleType: Virtual Machine Schedule Type
|
||||||
cron: Cron Schedule
|
cron:
|
||||||
|
label: Cron Schedule
|
||||||
|
editButton: Edit
|
||||||
detail:
|
detail:
|
||||||
namespace: Namespace
|
namespace: Namespace
|
||||||
sourceVM: Source Virtual Machine
|
sourceVM: Source Virtual Machine
|
||||||
@ -1064,6 +1139,8 @@ harvester:
|
|||||||
placeholder: e.g. 172.16.0.0/16
|
placeholder: e.g. 172.16.0.0/16
|
||||||
excludeIPs:
|
excludeIPs:
|
||||||
tooltip: The IP address list to reserve from automatic assignment. The gateway IP address is always excluded and will be automatically added to the list.
|
tooltip: The IP address list to reserve from automatic assignment. The gateway IP address is always excluded and will be automatically added to the list.
|
||||||
|
placeholder: single IP or 192.168.0.100..192.168.0.200
|
||||||
|
|
||||||
acl:
|
acl:
|
||||||
label: Access Control List
|
label: Access Control List
|
||||||
tooltip: The ACL to apply to this Subnet. Must be one of the ACLs in the same namespace.
|
tooltip: The ACL to apply to this Subnet. Must be one of the ACLs in the same namespace.
|
||||||
@ -1083,6 +1160,25 @@ harvester:
|
|||||||
banner: The supported field in ACL match can refer to <a href="https://kubeovn.github.io/docs/v1.14.x/en/guide/subnet/#subnet-acl" target="_blank">KubeOvn Subnet ACL document</a>
|
banner: The supported field in ACL match can refer to <a href="https://kubeovn.github.io/docs/v1.14.x/en/guide/subnet/#subnet-acl" target="_blank">KubeOvn Subnet ACL document</a>
|
||||||
|
|
||||||
vpc:
|
vpc:
|
||||||
|
viewTopology: Topology
|
||||||
|
topology:
|
||||||
|
loading: Loading topology...
|
||||||
|
empty: No resources found
|
||||||
|
visibility:
|
||||||
|
vpc: VPC
|
||||||
|
subnets: Subnets
|
||||||
|
overlayNetworks: Overlay Networks
|
||||||
|
vms: VMs
|
||||||
|
labels:
|
||||||
|
cidr: CIDR
|
||||||
|
provider: Provider
|
||||||
|
type: Type
|
||||||
|
clusterNetwork: Cluster Network
|
||||||
|
network: Network
|
||||||
|
subnet: Subnet
|
||||||
|
ip: IP
|
||||||
|
mac: MAC
|
||||||
|
peering: Peering
|
||||||
noAddonEnabled:
|
noAddonEnabled:
|
||||||
prefix: The kubeovn-operator add-on is not enabled, click
|
prefix: The kubeovn-operator add-on is not enabled, click
|
||||||
middle: here
|
middle: here
|
||||||
@ -1179,13 +1275,19 @@ harvester:
|
|||||||
vlan: VLAN ID
|
vlan: VLAN ID
|
||||||
exclude:
|
exclude:
|
||||||
label: Exclude IPs
|
label: Exclude IPs
|
||||||
placeholder: e.g. 172.16.0.1
|
placeholder: CIDR format, e.g. 172.16.0.10/32
|
||||||
invalid: '"Exclude list" is invalid.'
|
invalid: '"Exclude list" is invalid.'
|
||||||
addIp: Add Exclude IP
|
addIp: Add Exclude IP
|
||||||
warning: 'WARNING: <br/> Any change to storage-network requires shutting down all virtual machines before applying this setting. <br/> Users have to ensure the cluster network is configured and VLAN Configuration will cover all nodes and ensure the network connectivity is working and expected in all nodes.'
|
warning: 'WARNING: <br/> Any change to storage-network requires shutting down all virtual machines before applying this setting. <br/> Users have to ensure the cluster network is configured and VLAN Configuration will cover all nodes and ensure the network connectivity is working and expected in all nodes.'
|
||||||
tip: 'Specify an IP range in the IPv4 CIDR format. <code>Number of IPs Required = Number of Nodes * 2 + Number of Disks * 2 + Number of Images to Download/Upload </code>. For more information about storage network settings, see the <a href="{url}" target="_blank">documentation</a>.'
|
tip: 'Specify an IP range in the IPv4 CIDR format. <code>Number of IPs Required = Number of Nodes * 2 + Number of Disks * 2 + Number of Images to Download/Upload </code>. For more information about storage network settings, see the <a href="{url}" target="_blank">documentation</a>.'
|
||||||
|
rwxNetwork:
|
||||||
|
warning: 'WARNING: <br/> Any change to rwx-network requires longhorn RWX volumes detached before applying this setting.<br/>Users have to ensure the cluster network is configured and VLAN Configuration will cover all nodes and ensure the network connectivity is working and expected in all nodes.'
|
||||||
|
shareStorageNetwork: Share Storage Network
|
||||||
|
dedicatedRwxNetwork: Dedicated RWX Network
|
||||||
|
shareStorageNetworkWarning: The rwx-network is governed by storage-network, and changes here won't take effect until share-storage-network is set to false.
|
||||||
vmForceDeletionPolicy:
|
vmForceDeletionPolicy:
|
||||||
period: Period
|
period: Period
|
||||||
|
vmMigrationTimeout: VM Migration Timeout
|
||||||
vmMigrationNetwork:
|
vmMigrationNetwork:
|
||||||
parseError: "Failed to parse existing configuration."
|
parseError: "Failed to parse existing configuration."
|
||||||
fetchError: "Failed to load required network resources: {error}. Please refresh the page or try again later."
|
fetchError: "Failed to load required network resources: {error}. Please refresh the page or try again later."
|
||||||
@ -1249,7 +1351,10 @@ harvester:
|
|||||||
deleteImage: Please select an image to delete.
|
deleteImage: Please select an image to delete.
|
||||||
deleteSuccess: "{name} deleted successfully."
|
deleteSuccess: "{name} deleted successfully."
|
||||||
imagePreloadStrategy: Image Preload Strategy
|
imagePreloadStrategy: Image Preload Strategy
|
||||||
|
nodeUpgradeOption: Node Upgrade Option
|
||||||
restoreVM: Restore VM
|
restoreVM: Restore VM
|
||||||
|
strategy: Strategy
|
||||||
|
pauseNodes: Pause Nodes
|
||||||
strategyType: Strategy Type
|
strategyType: Strategy Type
|
||||||
concurrency: Concurrency
|
concurrency: Concurrency
|
||||||
harvesterMonitoring:
|
harvesterMonitoring:
|
||||||
@ -1265,10 +1370,20 @@ harvester:
|
|||||||
retention: How long to retain metrics
|
retention: How long to retain metrics
|
||||||
retentionSize: Maximum size of metrics
|
retentionSize: Maximum size of metrics
|
||||||
clusterRegistrationUrl:
|
clusterRegistrationUrl:
|
||||||
|
url: URL
|
||||||
|
insecureSkipTLSVerify: Insecure Skip TLS Verify
|
||||||
|
tip:
|
||||||
|
prefix: Harvester secures cluster registration via TLS by default. If opt out "Insecure Skip TLS Verify", you must provide custom CA certificates using the
|
||||||
|
middle: 'additional-ca'
|
||||||
|
suffix: setting.
|
||||||
message: To completely unset the imported Harvester cluster, please also remove it on the Rancher Dashboard UI via the <code> Virtualization Management </code> page.
|
message: To completely unset the imported Harvester cluster, please also remove it on the Rancher Dashboard UI via the <code> Virtualization Management </code> page.
|
||||||
ntpServers:
|
ntpServers:
|
||||||
isNotIPV4: The address you entered is not IPv4 or host. Please enter a valid IPv4 address or a host address.
|
isNotIPV4: The address you entered is not IPv4 or host. Please enter a valid IPv4 address or a host address.
|
||||||
isDuplicate: There are duplicate NTP server configurations.
|
isDuplicate: There are duplicate NTP server configurations.
|
||||||
|
instanceManagerResources:
|
||||||
|
parseError: "Failed to parse configuration: {error}"
|
||||||
|
v1: "V1 Data Engine"
|
||||||
|
v2: "V2 Data Engine"
|
||||||
kubevirtMigration:
|
kubevirtMigration:
|
||||||
parseError: "Failed to parse configuration: {error}"
|
parseError: "Failed to parse configuration: {error}"
|
||||||
parallelMigrationsPerCluster: "Parallel Migrations Per Cluster"
|
parallelMigrationsPerCluster: "Parallel Migrations Per Cluster"
|
||||||
@ -1534,10 +1649,86 @@ harvester:
|
|||||||
'harvester-seeder': harvester-seeder is an add-on that uses IPMI and Redfish to discover hardware information and perform out-of-band operations.
|
'harvester-seeder': harvester-seeder is an add-on that uses IPMI and Redfish to discover hardware information and perform out-of-band operations.
|
||||||
'harvester-system/harvester-seeder': harvester-seeder is an add-on that uses IPMI and Redfish to discover hardware information and perform out-of-band operations.
|
'harvester-system/harvester-seeder': harvester-seeder is an add-on that uses IPMI and Redfish to discover hardware information and perform out-of-band operations.
|
||||||
'harvester-csi-driver-lvm': harvester-csi-driver-lvm is an add-on allowing users to create PVC through the LVM with local devices.
|
'harvester-csi-driver-lvm': harvester-csi-driver-lvm is an add-on allowing users to create PVC through the LVM with local devices.
|
||||||
|
'descheduler': 'The virtual machine auto balance optimizes workload scheduling by evicting pods that are not optimally placed according to administrator-defined policies.'
|
||||||
|
|
||||||
vmImport:
|
vmImport:
|
||||||
titles:
|
titles:
|
||||||
basic: Basic
|
basic: Basic
|
||||||
|
auth: Authentication
|
||||||
pvc: Volume
|
pvc: Volume
|
||||||
|
networking: Network Mapping
|
||||||
|
advanced: Advanced
|
||||||
|
labels:
|
||||||
|
vmimport: Virtual Machine Import
|
||||||
|
vmimportSourceVMWare: Source VMWare
|
||||||
|
vmimportSourceOpenStack: Source OpenStack
|
||||||
|
vmimportSourceOVA: Source OVA
|
||||||
|
fields:
|
||||||
|
sourceProvider: Source Provider Type
|
||||||
|
sourceCluster: Source Cluster
|
||||||
|
vmName: VM Name
|
||||||
|
targetStorageClass: Target Storage Class
|
||||||
|
sourceNetwork: Source Network Name
|
||||||
|
destNetwork: Destination Network
|
||||||
|
interfaceModel: Interface Model
|
||||||
|
folder: Folder
|
||||||
|
diskBus: Default Disk Bus
|
||||||
|
defaultInterface: Default Network Interface
|
||||||
|
skipPreflight: Skip Preflight Checks
|
||||||
|
forcePowerOff: Force Power Off Source VM
|
||||||
|
username: Username
|
||||||
|
password: Password
|
||||||
|
caCert: CA Certificate (PEM)
|
||||||
|
selectSecret: Select Secret
|
||||||
|
createSecret: Create New Credentials
|
||||||
|
useSecret: Use Existing Secret
|
||||||
|
none: None (Public URL)
|
||||||
|
placeholders:
|
||||||
|
selectCluster: Select a cluster...
|
||||||
|
selectProviderFirst: Select a provider type first
|
||||||
|
matchSource: Must match the name in the source cluster
|
||||||
|
folderExample: e.g. /Datacenters/DC1/vm
|
||||||
|
caCert: "-----BEGIN CERTIFICATE----- ..."
|
||||||
|
options:
|
||||||
|
useDefault: Use Default
|
||||||
|
actions:
|
||||||
|
addNetwork: Add Network Mapping
|
||||||
|
remove: Remove
|
||||||
|
errors:
|
||||||
|
rfc1123: 'Invalid format. Name must be lowercase, alphanumeric, and cannot contain spaces (e.g. "my-vm-1"). If your Source VM name does not match this, you must rename it on the Source cluster first.'
|
||||||
|
networkMappingRequired: Every Network Mapping row must have a Source and Destination selected.
|
||||||
|
openstack:
|
||||||
|
fields:
|
||||||
|
endpoint: Identity Service Endpoint
|
||||||
|
region: Region
|
||||||
|
projectName: Project Name
|
||||||
|
domainName: Domain Name
|
||||||
|
retryCount: Upload Image Retry Count
|
||||||
|
retryDelay: Upload Image Retry Delay
|
||||||
|
placeholders:
|
||||||
|
endpoint: "e.g. https://devstack/identity"
|
||||||
|
region: e.g. RegionOne
|
||||||
|
projectName: e.g. admin
|
||||||
|
domainName: e.g. default
|
||||||
|
retryCount: "Default: 30"
|
||||||
|
retryDelay: "Default: 10"
|
||||||
|
vmware:
|
||||||
|
fields:
|
||||||
|
endpoint: vCenter Endpoint
|
||||||
|
datacenter: Datacenter
|
||||||
|
placeholders:
|
||||||
|
endpoint: "e.g. https://vscim/sdk"
|
||||||
|
datacenter: e.g. DC0
|
||||||
|
tooltips:
|
||||||
|
datacenter: The exact name of the Datacenter object in vCenter
|
||||||
|
ova:
|
||||||
|
fields:
|
||||||
|
url: URL
|
||||||
|
httpTimeout: HTTP Timeout
|
||||||
|
placeholders:
|
||||||
|
url: "e.g. https://download.example.com/images/my-vm.ova"
|
||||||
|
httpTimeout: "Default: 600"
|
||||||
|
|
||||||
rancherVcluster:
|
rancherVcluster:
|
||||||
accessRancher: Access the Rancher Dashboard
|
accessRancher: Access the Rancher Dashboard
|
||||||
hostname: Hostname
|
hostname: Hostname
|
||||||
@ -1664,7 +1855,8 @@ harvester:
|
|||||||
numVFs: Number Of Virtual Functions
|
numVFs: Number Of Virtual Functions
|
||||||
vfAddresses: Virtual Functions Addresses
|
vfAddresses: Virtual Functions Addresses
|
||||||
showMore: Show More
|
showMore: Show More
|
||||||
parentSriov: Filter By Parent SR-IOV
|
parentSriov: Filter By Parent SR-IOV Netork Device
|
||||||
|
parentSriovGPU: Filter By Parent SR-IOV GPU Device
|
||||||
|
|
||||||
sriovgpu:
|
sriovgpu:
|
||||||
label: SR-IOV GPU Devices
|
label: SR-IOV GPU Devices
|
||||||
@ -1695,7 +1887,7 @@ harvester:
|
|||||||
available: Available
|
available: Available
|
||||||
total: Total
|
total: Total
|
||||||
vGPUID: vGPU ID
|
vGPUID: vGPU ID
|
||||||
goSriovGPU:
|
goSriovGPU:
|
||||||
prefix: Please enable the supported GPU devices in
|
prefix: Please enable the supported GPU devices in
|
||||||
middle: SR-IOV GPU Devices
|
middle: SR-IOV GPU Devices
|
||||||
suffix: page to manage the vGPU MIG configurations.
|
suffix: page to manage the vGPU MIG configurations.
|
||||||
@ -1703,6 +1895,9 @@ harvester:
|
|||||||
vgpu:
|
vgpu:
|
||||||
label: vGPU Devices
|
label: vGPU Devices
|
||||||
noPermission: Please contact system administrator to add Harvester add-ons first.
|
noPermission: Please contact system administrator to add Harvester add-ons first.
|
||||||
|
detachWarning:
|
||||||
|
title: Cannot Disable vGPU
|
||||||
|
message: Please detach the device from the VM and save it first before disabling this vGPU device.
|
||||||
goSetting:
|
goSetting:
|
||||||
prefix: The nvidia-driver-toolkit add-on is not enabled, click
|
prefix: The nvidia-driver-toolkit add-on is not enabled, click
|
||||||
middle: here
|
middle: here
|
||||||
@ -1737,6 +1932,9 @@ harvester:
|
|||||||
claimError: Error enabling passthrough on {name}
|
claimError: Error enabling passthrough on {name}
|
||||||
unclaimError: Error disabling passthrough on {name}
|
unclaimError: Error disabling passthrough on {name}
|
||||||
cantUnclaim: You cannot disable passthrough on a device claimed by another user.
|
cantUnclaim: You cannot disable passthrough on a device claimed by another user.
|
||||||
|
detachWarning:
|
||||||
|
title: Cannot Disable Passthrough
|
||||||
|
message: Please detach the device from the VM and save it first before disabling passthrough.
|
||||||
enablePassthroughWarning: 'Please re-enable the USB device if the device path changes in the following situations:<br/> 1) Re-plugging the USB device.<br/> 2) Rebooting the node.<br/><br/>An incorrect device path may cause passthrough to fail.'
|
enablePassthroughWarning: 'Please re-enable the USB device if the device path changes in the following situations:<br/> 1) Re-plugging the USB device.<br/> 2) Rebooting the node.<br/><br/>An incorrect device path may cause passthrough to fail.'
|
||||||
|
|
||||||
harvesterVlanConfigMigrateDialog:
|
harvesterVlanConfigMigrateDialog:
|
||||||
@ -1799,11 +1997,13 @@ advancedSettings:
|
|||||||
'harv-additional-ca': 'Custom CA root certificates for TLS validation.'
|
'harv-additional-ca': 'Custom CA root certificates for TLS validation.'
|
||||||
'harv-overcommit-config': 'Resource overcommit configuration.'
|
'harv-overcommit-config': 'Resource overcommit configuration.'
|
||||||
'harv-support-bundle-timeout': 'Support bundle timeout configuration in minutes, use 0 to disable the timeout.'
|
'harv-support-bundle-timeout': 'Support bundle timeout configuration in minutes, use 0 to disable the timeout.'
|
||||||
|
'harv-support-bundle-file-name': 'Support bundle file name configuration.'
|
||||||
'harv-support-bundle-expiration': 'Support bundle expiration configuration in minutes.'
|
'harv-support-bundle-expiration': 'Support bundle expiration configuration in minutes.'
|
||||||
'harv-support-bundle-node-collection-timeout': 'Support bundle node collection timeout configuration in minutes.'
|
'harv-support-bundle-node-collection-timeout': 'Support bundle node collection timeout configuration in minutes.'
|
||||||
'harv-vm-force-reset-policy': Configuration for the force-reset action when a virtual machine is stuck on a node that is down.
|
'harv-vm-force-reset-policy': Configuration for the force-reset action when a virtual machine is stuck on a node that is down.
|
||||||
'harv-ssl-parameters': Custom SSL Parameters for TLS validation.
|
'harv-ssl-parameters': Custom SSL Parameters for TLS validation.
|
||||||
'harv-storage-network': 'Longhorn storage-network setting.'
|
'harv-storage-network': 'Longhorn storage-network setting.'
|
||||||
|
'harv-rwx-network': 'Configure RWX network behavior for shared or dedicated storage network usage.'
|
||||||
'harv-support-bundle-namespaces': Select additional namespaces to include in the support bundle.
|
'harv-support-bundle-namespaces': Select additional namespaces to include in the support bundle.
|
||||||
'harv-auto-disk-provision-paths': Specify the disks(using glob pattern) that Harvester will automatically add as virtual machine storage.
|
'harv-auto-disk-provision-paths': Specify the disks(using glob pattern) that Harvester will automatically add as virtual machine storage.
|
||||||
'harv-support-bundle-image': Support bundle image configuration. Find different versions in <a href="https://hub.docker.com/r/rancher/support-bundle-kit/tags" target="_blank">rancher/support-bundle-kit</a>.
|
'harv-support-bundle-image': Support bundle image configuration. Find different versions in <a href="https://hub.docker.com/r/rancher/support-bundle-kit/tags" target="_blank">rancher/support-bundle-kit</a>.
|
||||||
@ -1819,6 +2019,7 @@ advancedSettings:
|
|||||||
'harv-rancher-cluster': 'Configure Rancher cluster integration settings for guest cluster management.'
|
'harv-rancher-cluster': 'Configure Rancher cluster integration settings for guest cluster management.'
|
||||||
'harv-max-hotplug-ratio': 'The ratio for kubevirt to limit the maximum CPU and memory that can be hotplugged to a VM. The value could be an integer between 1 and 20, default to 4.'
|
'harv-max-hotplug-ratio': 'The ratio for kubevirt to limit the maximum CPU and memory that can be hotplugged to a VM. The value could be an integer between 1 and 20, default to 4.'
|
||||||
'harv-kubevirt-migration': 'Configure cluster-wide KubeVirt live migration parameters.'
|
'harv-kubevirt-migration': 'Configure cluster-wide KubeVirt live migration parameters.'
|
||||||
|
'harv-instance-manager-resources': 'Configure resource percentage reservations for Longhorn instance manager V1 and V2. Valid instance manager CPU range between 0 - 40.'
|
||||||
|
|
||||||
typeLabel:
|
typeLabel:
|
||||||
kubevirt.io.virtualmachine: |-
|
kubevirt.io.virtualmachine: |-
|
||||||
@ -2000,3 +2201,23 @@ typeLabel:
|
|||||||
one { IP Pool }
|
one { IP Pool }
|
||||||
other { IP Pools }
|
other { IP Pools }
|
||||||
}
|
}
|
||||||
|
migration.harvesterhci.io.openstacksource: |-
|
||||||
|
{count, plural,
|
||||||
|
one { OpenStack Source }
|
||||||
|
other { OpenStack Sources }
|
||||||
|
}
|
||||||
|
migration.harvesterhci.io.vmwaresource: |-
|
||||||
|
{count, plural,
|
||||||
|
one { VMware Source }
|
||||||
|
other { VMware Sources }
|
||||||
|
}
|
||||||
|
migration.harvesterhci.io.ovasource: |-
|
||||||
|
{count, plural,
|
||||||
|
one { OVA Source }
|
||||||
|
other { OVA Sources }
|
||||||
|
}
|
||||||
|
migration.harvesterhci.io.virtualmachineimport: |-
|
||||||
|
{count, plural,
|
||||||
|
one { Virtual Machine Import }
|
||||||
|
other { Virtual Machine Imports }
|
||||||
|
}
|
||||||
|
|||||||
@ -64,6 +64,10 @@ export default {
|
|||||||
const inStore = this.$store.getters['currentProduct'].inStore;
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
const rows = this.$store.getters[`${ inStore }/all`](HCI.PCI_DEVICE);
|
const rows = this.$store.getters[`${ inStore }/all`](HCI.PCI_DEVICE);
|
||||||
|
|
||||||
|
rows.forEach((row) => {
|
||||||
|
row.allowDisable = true;
|
||||||
|
});
|
||||||
|
|
||||||
return rows;
|
return rows;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|||||||
@ -54,7 +54,13 @@ export default {
|
|||||||
devices() {
|
devices() {
|
||||||
const inStore = this.$store.getters['currentProduct'].inStore;
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
|
|
||||||
return this.$store.getters[`${ inStore }/all`](HCI.USB_DEVICE) || [];
|
const data = this.$store.getters[`${ inStore }/all`](HCI.USB_DEVICE) || [];
|
||||||
|
|
||||||
|
data.forEach((row) => {
|
||||||
|
row.allowDisable = true;
|
||||||
|
});
|
||||||
|
|
||||||
|
return data;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|||||||
@ -65,6 +65,10 @@ export default {
|
|||||||
const vGpuDevices = this.$store.getters[`${ inStore }/all`](HCI.VGPU_DEVICE) || [];
|
const vGpuDevices = this.$store.getters[`${ inStore }/all`](HCI.VGPU_DEVICE) || [];
|
||||||
const srioVGpuDevices = this.$store.getters[`${ inStore }/all`](HCI.SR_IOVGPU_DEVICE) || [];
|
const srioVGpuDevices = this.$store.getters[`${ inStore }/all`](HCI.SR_IOVGPU_DEVICE) || [];
|
||||||
|
|
||||||
|
vGpuDevices.forEach((row) => {
|
||||||
|
row.allowDisable = true;
|
||||||
|
});
|
||||||
|
|
||||||
if (this.hasSRIOVGPUSchema) {
|
if (this.hasSRIOVGPUSchema) {
|
||||||
return vGpuDevices.filter((device) => !!srioVGpuDevices.find((s) => s.isEnabled && s.spec?.nodeName === device.spec?.nodeName));
|
return vGpuDevices.filter((device) => !!srioVGpuDevices.find((s) => s.isEnabled && s.spec?.nodeName === device.spec?.nodeName));
|
||||||
}
|
}
|
||||||
|
|||||||
@ -155,6 +155,15 @@ export default {
|
|||||||
return location;
|
return location;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
viewTopology(group) {
|
||||||
|
const vpc = group.key;
|
||||||
|
const resource = this.$store.getters[`harvester/byId`](HCI.VPC, vpc);
|
||||||
|
|
||||||
|
if (resource && resource.goToDetail) {
|
||||||
|
resource.goToDetail();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
showVpcAction(event, group) {
|
showVpcAction(event, group) {
|
||||||
const vpc = group.key;
|
const vpc = group.key;
|
||||||
|
|
||||||
@ -218,6 +227,14 @@ export default {
|
|||||||
>
|
>
|
||||||
{{ t('harvester.vpc.createSubnet') }}
|
{{ t('harvester.vpc.createSubnet') }}
|
||||||
</router-link>
|
</router-link>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
class="btn btn-sm role-secondary mr-5"
|
||||||
|
@click="viewTopology(group)"
|
||||||
|
>
|
||||||
|
<i class="icon icon-globe mr-5" />
|
||||||
|
{{ t('harvester.vpc.viewTopology') }}
|
||||||
|
</button>
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
class="btn btn-sm role-multi-action actions mr-10"
|
class="btn btn-sm role-multi-action actions mr-10"
|
||||||
|
|||||||
@ -110,11 +110,12 @@ export default {
|
|||||||
|
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
hasNode: false,
|
hasNode: false,
|
||||||
allVMs: [],
|
allVMs: [],
|
||||||
allVMIs: [],
|
allVMIs: [],
|
||||||
allNodeNetworks: [],
|
allNodeNetworks: [],
|
||||||
allClusterNetworks: [],
|
allClusterNetworks: [],
|
||||||
|
restartNotificationDisplayed: false,
|
||||||
HCI
|
HCI
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
@ -174,6 +175,48 @@ export default {
|
|||||||
this['allVMIs'] = vmis;
|
this['allVMIs'] = vmis;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
beforeUnmount() {
|
||||||
|
// clear restart message before component unmount
|
||||||
|
this.$store.dispatch('growl/clear');
|
||||||
|
},
|
||||||
|
|
||||||
|
watch: {
|
||||||
|
allVMs: {
|
||||||
|
handler(neu) {
|
||||||
|
const vmNames = [];
|
||||||
|
|
||||||
|
neu.forEach((vm) => {
|
||||||
|
if (vm.isRestartRequired) {
|
||||||
|
vmNames.push(vm.metadata.name);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
const count = vmNames.length;
|
||||||
|
|
||||||
|
if ( count === 0 && this.restartNotificationDisplayed) {
|
||||||
|
this.restartNotificationDisplayed = false;
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (count > 0) {
|
||||||
|
// clear old notification before showing new one
|
||||||
|
if (this.restartNotificationDisplayed) {
|
||||||
|
this.$store.dispatch('growl/clear');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (count > 0 && vmNames.length > 0) {
|
||||||
|
this.$store.dispatch('growl/warning', {
|
||||||
|
title: this.t('harvester.notification.restartRequired.title', { count }),
|
||||||
|
message: this.t('harvester.notification.restartRequired.message', { vmNames: vmNames.join(', ') }),
|
||||||
|
timeout: 10000,
|
||||||
|
}, { root: true });
|
||||||
|
this.restartNotificationDisplayed = true;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
deep: true,
|
||||||
|
}
|
||||||
|
},
|
||||||
methods: {
|
methods: {
|
||||||
lockIconTooltipMessage(row) {
|
lockIconTooltipMessage(row) {
|
||||||
const message = '';
|
const message = '';
|
||||||
@ -243,6 +286,12 @@ export default {
|
|||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
<style lang="scss">
|
||||||
|
.growl-container {
|
||||||
|
z-index: 56 !important; // set to be lower than the vm action menu (z-index: 57)
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
.state {
|
.state {
|
||||||
display: flex;
|
display: flex;
|
||||||
|
|||||||
@ -0,0 +1,60 @@
|
|||||||
|
<script>
|
||||||
|
import ResourceTable from '@shell/components/ResourceTable';
|
||||||
|
import Loading from '@shell/components/Loading';
|
||||||
|
import { SCHEMA } from '@shell/config/types';
|
||||||
|
import { HCI } from '../types';
|
||||||
|
|
||||||
|
const schema = {
|
||||||
|
id: HCI.VMIMPORT_SOURCE_O,
|
||||||
|
type: SCHEMA,
|
||||||
|
attributes: {
|
||||||
|
kind: HCI.VMIMPORT_SOURCE_O,
|
||||||
|
namespaced: true
|
||||||
|
},
|
||||||
|
metadata: { name: HCI.VMIMPORT_SOURCE_O },
|
||||||
|
};
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'HarvesterVMImportSourceO',
|
||||||
|
components: { ResourceTable, Loading },
|
||||||
|
inheritAttrs: false,
|
||||||
|
|
||||||
|
async fetch() {
|
||||||
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
|
|
||||||
|
this.rows = await this.$store.dispatch(`${ inStore }/findAll`, { type: HCI.VMIMPORT_SOURCE_O });
|
||||||
|
|
||||||
|
const configSchema = this.$store.getters[`${ inStore }/schemaFor`](HCI.VMIMPORT_SOURCE_O);
|
||||||
|
|
||||||
|
if (!configSchema?.collectionMethods.find((x) => x.toLowerCase() === 'post')) {
|
||||||
|
this.$store.dispatch('type-map/configureType', { match: HCI.VMIMPORT_SOURCE_O, isCreatable: false });
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
data() {
|
||||||
|
return { rows: [] };
|
||||||
|
},
|
||||||
|
|
||||||
|
computed: {
|
||||||
|
schema() {
|
||||||
|
return schema;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
typeDisplay() {
|
||||||
|
return this.$store.getters['type-map/labelFor'](schema, 99);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<Loading v-if="$fetchState.pending" />
|
||||||
|
<ResourceTable
|
||||||
|
v-else
|
||||||
|
v-bind="$attrs"
|
||||||
|
:groupable="true"
|
||||||
|
:schema="schema"
|
||||||
|
:rows="rows"
|
||||||
|
key-field="_key"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
60
pkg/harvester/list/migration.harvesterhci.io.ovasource.vue
Normal file
60
pkg/harvester/list/migration.harvesterhci.io.ovasource.vue
Normal file
@ -0,0 +1,60 @@
|
|||||||
|
<script>
|
||||||
|
import ResourceTable from '@shell/components/ResourceTable';
|
||||||
|
import Loading from '@shell/components/Loading';
|
||||||
|
import { SCHEMA } from '@shell/config/types';
|
||||||
|
import { HCI } from '../types';
|
||||||
|
|
||||||
|
const schema = {
|
||||||
|
id: HCI.VMIMPORT_SOURCE_OVA,
|
||||||
|
type: SCHEMA,
|
||||||
|
attributes: {
|
||||||
|
kind: HCI.VMIMPORT_SOURCE_OVA,
|
||||||
|
namespaced: true
|
||||||
|
},
|
||||||
|
metadata: { name: HCI.VMIMPORT_SOURCE_OVA },
|
||||||
|
};
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'HarvesterVMImportSourceOVA',
|
||||||
|
components: { ResourceTable, Loading },
|
||||||
|
inheritAttrs: false,
|
||||||
|
|
||||||
|
async fetch() {
|
||||||
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
|
|
||||||
|
this.rows = await this.$store.dispatch(`${ inStore }/findAll`, { type: HCI.VMIMPORT_SOURCE_OVA });
|
||||||
|
|
||||||
|
const configSchema = this.$store.getters[`${ inStore }/schemaFor`](HCI.VMIMPORT_SOURCE_OVA);
|
||||||
|
|
||||||
|
if (!configSchema?.collectionMethods.find((x) => x.toLowerCase() === 'post')) {
|
||||||
|
this.$store.dispatch('type-map/configureType', { match: HCI.VMIMPORT_SOURCE_OVA, isCreatable: false });
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
data() {
|
||||||
|
return { rows: [] };
|
||||||
|
},
|
||||||
|
|
||||||
|
computed: {
|
||||||
|
schema() {
|
||||||
|
return schema;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
typeDisplay() {
|
||||||
|
return this.$store.getters['type-map/labelFor'](schema, 99);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<Loading v-if="$fetchState.pending" />
|
||||||
|
<ResourceTable
|
||||||
|
v-else
|
||||||
|
v-bind="$attrs"
|
||||||
|
:groupable="true"
|
||||||
|
:schema="schema"
|
||||||
|
:rows="rows"
|
||||||
|
key-field="_key"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
@ -0,0 +1,60 @@
|
|||||||
|
<script>
|
||||||
|
import ResourceTable from '@shell/components/ResourceTable';
|
||||||
|
import Loading from '@shell/components/Loading';
|
||||||
|
import { SCHEMA } from '@shell/config/types';
|
||||||
|
import { HCI } from '../types';
|
||||||
|
|
||||||
|
const schema = {
|
||||||
|
id: HCI.VMIMPORT,
|
||||||
|
type: SCHEMA,
|
||||||
|
attributes: {
|
||||||
|
kind: HCI.VMIMPORT,
|
||||||
|
namespaced: true
|
||||||
|
},
|
||||||
|
metadata: { name: HCI.VMIMPORT },
|
||||||
|
};
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'HarvesterVMImportVirtualMachine',
|
||||||
|
components: { ResourceTable, Loading },
|
||||||
|
inheritAttrs: false,
|
||||||
|
|
||||||
|
async fetch() {
|
||||||
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
|
|
||||||
|
this.rows = await this.$store.dispatch(`${ inStore }/findAll`, { type: HCI.VMIMPORT });
|
||||||
|
|
||||||
|
const configSchema = this.$store.getters[`${ inStore }/schemaFor`](HCI.VMIMPORT);
|
||||||
|
|
||||||
|
if (!configSchema?.collectionMethods.find((x) => x.toLowerCase() === 'post')) {
|
||||||
|
this.$store.dispatch('type-map/configureType', { match: HCI.VMIMPORT, isCreatable: false });
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
data() {
|
||||||
|
return { rows: [] };
|
||||||
|
},
|
||||||
|
|
||||||
|
computed: {
|
||||||
|
schema() {
|
||||||
|
return schema;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
typeDisplay() {
|
||||||
|
return this.$store.getters['type-map/labelFor'](schema, 99);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<Loading v-if="$fetchState.pending" />
|
||||||
|
<ResourceTable
|
||||||
|
v-else
|
||||||
|
v-bind="$attrs"
|
||||||
|
:groupable="true"
|
||||||
|
:schema="schema"
|
||||||
|
:rows="rows"
|
||||||
|
key-field="_key"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
@ -0,0 +1,60 @@
|
|||||||
|
<script>
|
||||||
|
import ResourceTable from '@shell/components/ResourceTable';
|
||||||
|
import Loading from '@shell/components/Loading';
|
||||||
|
import { SCHEMA } from '@shell/config/types';
|
||||||
|
import { HCI } from '../types';
|
||||||
|
|
||||||
|
const schema = {
|
||||||
|
id: HCI.VMIMPORT_SOURCE_V,
|
||||||
|
type: SCHEMA,
|
||||||
|
attributes: {
|
||||||
|
kind: HCI.VMIMPORT_SOURCE_V,
|
||||||
|
namespaced: true
|
||||||
|
},
|
||||||
|
metadata: { name: HCI.VMIMPORT_SOURCE_V },
|
||||||
|
};
|
||||||
|
|
||||||
|
export default {
|
||||||
|
name: 'HarvesterVMImportSourceV',
|
||||||
|
components: { ResourceTable, Loading },
|
||||||
|
inheritAttrs: false,
|
||||||
|
|
||||||
|
async fetch() {
|
||||||
|
const inStore = this.$store.getters['currentProduct'].inStore;
|
||||||
|
|
||||||
|
this.rows = await this.$store.dispatch(`${ inStore }/findAll`, { type: HCI.VMIMPORT_SOURCE_V });
|
||||||
|
|
||||||
|
const configSchema = this.$store.getters[`${ inStore }/schemaFor`](HCI.VMIMPORT_SOURCE_V);
|
||||||
|
|
||||||
|
if (!configSchema?.collectionMethods.find((x) => x.toLowerCase() === 'post')) {
|
||||||
|
this.$store.dispatch('type-map/configureType', { match: HCI.VMIMPORT_SOURCE_V, isCreatable: false });
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
data() {
|
||||||
|
return { rows: [] };
|
||||||
|
},
|
||||||
|
|
||||||
|
computed: {
|
||||||
|
schema() {
|
||||||
|
return schema;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
typeDisplay() {
|
||||||
|
return this.$store.getters['type-map/labelFor'](schema, 99);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<Loading v-if="$fetchState.pending" />
|
||||||
|
<ResourceTable
|
||||||
|
v-else
|
||||||
|
v-bind="$attrs"
|
||||||
|
:groupable="true"
|
||||||
|
:schema="schema"
|
||||||
|
:rows="rows"
|
||||||
|
key-field="_key"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
@ -22,7 +22,7 @@ import {
|
|||||||
} from '../../config/harvester-map';
|
} from '../../config/harvester-map';
|
||||||
import { HCI_SETTING } from '../../config/settings';
|
import { HCI_SETTING } from '../../config/settings';
|
||||||
import { HCI } from '../../types';
|
import { HCI } from '../../types';
|
||||||
import { parseVolumeClaimTemplates } from '../../utils/vm';
|
import { parseVolumeClaimTemplates, EMPTY_IMAGE } from '../../utils/vm';
|
||||||
import impl, { QGA_JSON, USB_TABLET } from './impl';
|
import impl, { QGA_JSON, USB_TABLET } from './impl';
|
||||||
import { GIBIBYTE } from '../../utils/unit';
|
import { GIBIBYTE } from '../../utils/unit';
|
||||||
import { VOLUME_MODE } from '@pkg/harvester/config/types';
|
import { VOLUME_MODE } from '@pkg/harvester/config/types';
|
||||||
@ -142,6 +142,7 @@ export default {
|
|||||||
showYaml: false,
|
showYaml: false,
|
||||||
spec: null,
|
spec: null,
|
||||||
osType: 'linux',
|
osType: 'linux',
|
||||||
|
useTemplate: false,
|
||||||
sshKey: [],
|
sshKey: [],
|
||||||
maintenanceStrategies,
|
maintenanceStrategies,
|
||||||
maintenanceStrategy: 'Migrate',
|
maintenanceStrategy: 'Migrate',
|
||||||
@ -181,6 +182,7 @@ export default {
|
|||||||
immutableMode: this.realMode === _CREATE ? _CREATE : _VIEW,
|
immutableMode: this.realMode === _CREATE ? _CREATE : _VIEW,
|
||||||
terminationGracePeriodSeconds: '',
|
terminationGracePeriodSeconds: '',
|
||||||
cpuPinning: false,
|
cpuPinning: false,
|
||||||
|
cpuModel: '',
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -273,7 +275,7 @@ export default {
|
|||||||
|
|
||||||
needNewSecret() {
|
needNewSecret() {
|
||||||
// When creating a template it is always necessary to create a new secret.
|
// When creating a template it is always necessary to create a new secret.
|
||||||
return this.isCreate ? true : this.showYaml ? false : this.resourceType === HCI.VM_VERSION;
|
return this.isCreate || this.showYaml ? false : this.resourceType === HCI.VM_VERSION;
|
||||||
},
|
},
|
||||||
|
|
||||||
defaultTerminationSetting() {
|
defaultTerminationSetting() {
|
||||||
@ -393,6 +395,7 @@ export default {
|
|||||||
const efiPersistentStateEnabled = this.isEFIPersistentStateEnabled(spec);
|
const efiPersistentStateEnabled = this.isEFIPersistentStateEnabled(spec);
|
||||||
const secureBoot = this.isSecureBoot(spec);
|
const secureBoot = this.isSecureBoot(spec);
|
||||||
const cpuPinning = this.isCpuPinning(spec);
|
const cpuPinning = this.isCpuPinning(spec);
|
||||||
|
const cpuModel = spec.template.spec.domain.cpu?.model || '';
|
||||||
|
|
||||||
const secretRef = this.getSecret(spec);
|
const secretRef = this.getSecret(spec);
|
||||||
const accessCredentials = this.getAccessCredentials(spec);
|
const accessCredentials = this.getAccessCredentials(spec);
|
||||||
@ -430,6 +433,7 @@ export default {
|
|||||||
this['tpmPersistentStateEnabled'] = tpmPersistentStateEnabled;
|
this['tpmPersistentStateEnabled'] = tpmPersistentStateEnabled;
|
||||||
this['secureBoot'] = secureBoot;
|
this['secureBoot'] = secureBoot;
|
||||||
this['cpuPinning'] = cpuPinning;
|
this['cpuPinning'] = cpuPinning;
|
||||||
|
this['cpuModel'] = cpuModel;
|
||||||
|
|
||||||
this['hasCreateVolumes'] = hasCreateVolumes;
|
this['hasCreateVolumes'] = hasCreateVolumes;
|
||||||
this['networkRows'] = networkRows;
|
this['networkRows'] = networkRows;
|
||||||
@ -507,12 +511,15 @@ export default {
|
|||||||
|
|
||||||
const type = DISK?.cdrom ? CD_ROM : DISK?.disk ? HARD_DISK : '';
|
const type = DISK?.cdrom ? CD_ROM : DISK?.disk ? HARD_DISK : '';
|
||||||
|
|
||||||
if (volume?.containerDisk) { // SOURCE_TYPE.CONTAINER
|
if (type === CD_ROM && volume === undefined) {
|
||||||
|
// Empty CD_ROM
|
||||||
|
source = SOURCE_TYPE.IMAGE;
|
||||||
|
image = EMPTY_IMAGE;
|
||||||
|
size = `0${ GIBIBYTE }`;
|
||||||
|
} else if (volume.containerDisk) { // SOURCE_TYPE.CONTAINER
|
||||||
source = SOURCE_TYPE.CONTAINER;
|
source = SOURCE_TYPE.CONTAINER;
|
||||||
container = volume.containerDisk.image;
|
container = volume.containerDisk.image;
|
||||||
}
|
} else if (volume.persistentVolumeClaim && volume.persistentVolumeClaim?.claimName) {
|
||||||
|
|
||||||
if (volume.persistentVolumeClaim && volume.persistentVolumeClaim?.claimName) {
|
|
||||||
volumeName = volume.persistentVolumeClaim.claimName;
|
volumeName = volume.persistentVolumeClaim.claimName;
|
||||||
const DVT = _volumeClaimTemplates.find( (T) => T.metadata.name === volumeName);
|
const DVT = _volumeClaimTemplates.find( (T) => T.metadata.name === volumeName);
|
||||||
|
|
||||||
@ -689,7 +696,6 @@ export default {
|
|||||||
set(this.spec.template.spec, 'domain.memory.maxGuest', this.maxMemory);
|
set(this.spec.template.spec, 'domain.memory.maxGuest', this.maxMemory);
|
||||||
set(this.spec.template.spec, 'domain.resources.limits.memory', this.maxMemory);
|
set(this.spec.template.spec, 'domain.resources.limits.memory', this.maxMemory);
|
||||||
} else {
|
} else {
|
||||||
this.spec.template.spec.domain.cpu.maxSockets = 1;
|
|
||||||
this.spec.template.spec.domain.cpu.sockets = 1;
|
this.spec.template.spec.domain.cpu.sockets = 1;
|
||||||
this.spec.template.spec.domain.cpu.cores = this.cpu;
|
this.spec.template.spec.domain.cpu.cores = this.cpu;
|
||||||
this.spec.template.spec.domain.resources.limits.cpu = this.cpu?.toString();
|
this.spec.template.spec.domain.resources.limits.cpu = this.cpu?.toString();
|
||||||
@ -701,34 +707,41 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
needVolumeRelatedInfo(R) {
|
||||||
|
// return [needVolume, needVolumeClaimTemplate]
|
||||||
|
if (R.source === SOURCE_TYPE.CONTAINER) {
|
||||||
|
return [true, false];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (R.source === SOURCE_TYPE.IMAGE && R.image === EMPTY_IMAGE) {
|
||||||
|
return [false, false];
|
||||||
|
}
|
||||||
|
|
||||||
|
return [true, true];
|
||||||
|
},
|
||||||
|
|
||||||
parseDiskRows(disk) {
|
parseDiskRows(disk) {
|
||||||
const disks = [];
|
const disks = [];
|
||||||
const volumes = [];
|
const volumes = [];
|
||||||
const diskNameLabels = [];
|
|
||||||
const volumeClaimTemplates = [];
|
const volumeClaimTemplates = [];
|
||||||
|
|
||||||
disk.forEach( (R, index) => {
|
disk.forEach( (R, index) => {
|
||||||
const prefixName = this.value.metadata?.name || '';
|
|
||||||
|
|
||||||
let dataVolumeName = '';
|
|
||||||
|
|
||||||
if (R.source === SOURCE_TYPE.ATTACH_VOLUME) {
|
|
||||||
dataVolumeName = R.volumeName;
|
|
||||||
} else if (this.isClone || !this.hasCreateVolumes.includes(R.realName)) {
|
|
||||||
dataVolumeName = `${ prefixName }-${ R.name }-${ randomStr(5).toLowerCase() }`;
|
|
||||||
} else {
|
|
||||||
dataVolumeName = R.realName;
|
|
||||||
}
|
|
||||||
|
|
||||||
const _disk = this.parseDisk(R, index);
|
const _disk = this.parseDisk(R, index);
|
||||||
const _volume = this.parseVolume(R, dataVolumeName);
|
|
||||||
const _dataVolumeTemplate = this.parseVolumeClaimTemplate(R, dataVolumeName);
|
|
||||||
|
|
||||||
disks.push(_disk);
|
disks.push(_disk);
|
||||||
volumes.push(_volume);
|
|
||||||
diskNameLabels.push(dataVolumeName);
|
|
||||||
|
|
||||||
if (R.source !== SOURCE_TYPE.CONTAINER) {
|
const prefixName = this.value.metadata?.name || '';
|
||||||
|
const dataVolumeName = this.parseDataVolumeName(R, prefixName);
|
||||||
|
const [needVolume, needVolumeClaimTemplate] = this.needVolumeRelatedInfo(R);
|
||||||
|
|
||||||
|
if (needVolume) {
|
||||||
|
const _volume = this.parseVolume(R, dataVolumeName);
|
||||||
|
|
||||||
|
volumes.push(_volume);
|
||||||
|
}
|
||||||
|
if (needVolumeClaimTemplate) {
|
||||||
|
const _dataVolumeTemplate = this.parseVolumeClaimTemplate(R, dataVolumeName);
|
||||||
|
|
||||||
volumeClaimTemplates.push(_dataVolumeTemplate);
|
volumeClaimTemplates.push(_dataVolumeTemplate);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -1013,6 +1026,20 @@ export default {
|
|||||||
this['cpuMemoryHotplugEnabled'] = cpuMemoryHotplugEnabled;
|
this['cpuMemoryHotplugEnabled'] = cpuMemoryHotplugEnabled;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
parseDataVolumeName(R, prefixName) {
|
||||||
|
let dataVolumeName = '';
|
||||||
|
|
||||||
|
if (R.source === SOURCE_TYPE.ATTACH_VOLUME) {
|
||||||
|
dataVolumeName = R.volumeName;
|
||||||
|
} else if (this.isClone || !this.hasCreateVolumes.includes(R.realName)) {
|
||||||
|
dataVolumeName = `${ prefixName }-${ R.name }-${ randomStr(5).toLowerCase() }`;
|
||||||
|
} else {
|
||||||
|
dataVolumeName = R.realName;
|
||||||
|
}
|
||||||
|
|
||||||
|
return dataVolumeName;
|
||||||
|
},
|
||||||
|
|
||||||
parseDisk(R, index) {
|
parseDisk(R, index) {
|
||||||
const out = { name: R.name };
|
const out = { name: R.name };
|
||||||
|
|
||||||
@ -1638,7 +1665,8 @@ export default {
|
|||||||
|
|
||||||
secretRef: {
|
secretRef: {
|
||||||
handler(secret) {
|
handler(secret) {
|
||||||
if (secret && this.resourceType !== HCI.BACKUP) {
|
// we should not inherit the secret if it's from VM template.
|
||||||
|
if (secret && this.resourceType !== HCI.BACKUP && !this.useTemplate) {
|
||||||
this.secretName = secret?.metadata.name;
|
this.secretName = secret?.metadata.name;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|||||||
@ -1,12 +1,12 @@
|
|||||||
import SteveModel from '@shell/plugins/steve/steve-class';
|
|
||||||
import { escapeHtml } from '@shell/utils/string';
|
import { escapeHtml } from '@shell/utils/string';
|
||||||
import { colorForState } from '@shell/plugins/dashboard-store/resource-class';
|
import { colorForState } from '@shell/plugins/dashboard-store/resource-class';
|
||||||
|
import HarvesterResource from './harvester';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Class representing vGPU MIGConfiguration resource.
|
* Class representing vGPU MIGConfiguration resource.
|
||||||
* @extends SteveModal
|
* @extends HarvesterResource
|
||||||
*/
|
*/
|
||||||
export default class MIGCONFIGURATION extends SteveModel {
|
export default class MIGCONFIGURATION extends HarvesterResource {
|
||||||
get _availableActions() {
|
get _availableActions() {
|
||||||
let out = super._availableActions;
|
let out = super._availableActions;
|
||||||
|
|
||||||
|
|||||||
@ -1,6 +1,7 @@
|
|||||||
import SteveModel from '@shell/plugins/steve/steve-class';
|
import SteveModel from '@shell/plugins/steve/steve-class';
|
||||||
import { escapeHtml } from '@shell/utils/string';
|
import { escapeHtml } from '@shell/utils/string';
|
||||||
import { HCI } from '../types';
|
import { HCI } from '../types';
|
||||||
|
import { HCI as HCI_ANNOTATIONS } from '@pkg/harvester/config/labels-annotations';
|
||||||
|
|
||||||
const STATUS_DISPLAY = {
|
const STATUS_DISPLAY = {
|
||||||
enabled: {
|
enabled: {
|
||||||
@ -32,7 +33,7 @@ export default class PCIDevice extends SteveModel {
|
|||||||
out.push(
|
out.push(
|
||||||
{
|
{
|
||||||
action: 'enablePassthroughBulk',
|
action: 'enablePassthroughBulk',
|
||||||
enabled: !this.isEnabling,
|
enabled: !this.isEnabling && !this.isvGPUDevice,
|
||||||
icon: 'icon icon-fw icon-dot',
|
icon: 'icon icon-fw icon-dot',
|
||||||
label: 'Enable Passthrough',
|
label: 'Enable Passthrough',
|
||||||
bulkable: true,
|
bulkable: true,
|
||||||
@ -41,7 +42,7 @@ export default class PCIDevice extends SteveModel {
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
action: 'disablePassthrough',
|
action: 'disablePassthrough',
|
||||||
enabled: this.isEnabling && this.claimedByMe,
|
enabled: this.isEnabling && this.claimedByMe && !this.isvGPUDevice,
|
||||||
icon: 'icon icon-fw icon-dot-open',
|
icon: 'icon icon-fw icon-dot-open',
|
||||||
label: 'Disable Passthrough',
|
label: 'Disable Passthrough',
|
||||||
bulkable: true,
|
bulkable: true,
|
||||||
@ -52,6 +53,14 @@ export default class PCIDevice extends SteveModel {
|
|||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get isvGPUDevice() {
|
||||||
|
if (!this.vGPUAsPCIDeviceFeatureEnabled) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return !!this.metadata?.labels?.[HCI_ANNOTATIONS.PARENT_SRIOV_GPU];
|
||||||
|
}
|
||||||
|
|
||||||
get canYaml() {
|
get canYaml() {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -144,6 +153,12 @@ export default class PCIDevice extends SteveModel {
|
|||||||
// 'disable' passthrough deletes claim
|
// 'disable' passthrough deletes claim
|
||||||
// backend should return error if device is in use
|
// backend should return error if device is in use
|
||||||
async disablePassthrough() {
|
async disablePassthrough() {
|
||||||
|
if (!this.allowDisable) {
|
||||||
|
this.showDetachWarning();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
if (!this.claimedByMe) {
|
if (!this.claimedByMe) {
|
||||||
throw new Error(this.$rootGetters['i18n/t']('harvester.pci.cantUnclaim', { name: escapeHtml(this.metadata.name) }));
|
throw new Error(this.$rootGetters['i18n/t']('harvester.pci.cantUnclaim', { name: escapeHtml(this.metadata.name) }));
|
||||||
@ -169,4 +184,24 @@ export default class PCIDevice extends SteveModel {
|
|||||||
get groupByDevice() {
|
get groupByDevice() {
|
||||||
return this.status?.description;
|
return this.status?.description;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get vGPUAsPCIDeviceFeatureEnabled() {
|
||||||
|
return this.$rootGetters['harvester-common/getFeatureEnabled']('vGPUAsPCIDevice');
|
||||||
|
}
|
||||||
|
|
||||||
|
showDetachWarning() {
|
||||||
|
this.$dispatch('growl/warning', {
|
||||||
|
title: this.$rootGetters['i18n/t']('harvester.pci.detachWarning.title'),
|
||||||
|
message: this.$rootGetters['i18n/t']('harvester.pci.detachWarning.message'),
|
||||||
|
timeout: 5000
|
||||||
|
}, { root: true });
|
||||||
|
}
|
||||||
|
|
||||||
|
get allowDisable() {
|
||||||
|
return this._allowDisable;
|
||||||
|
}
|
||||||
|
|
||||||
|
set allowDisable(value) {
|
||||||
|
this._allowDisable = value;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -133,6 +133,12 @@ export default class USBDevice extends SteveModel {
|
|||||||
// 'disable' passthrough deletes claim
|
// 'disable' passthrough deletes claim
|
||||||
// backend should return error if device is in use
|
// backend should return error if device is in use
|
||||||
async disablePassthrough() {
|
async disablePassthrough() {
|
||||||
|
if (!this.allowDisable) {
|
||||||
|
this.showDetachWarning();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
if (!this.claimedByMe) {
|
if (!this.claimedByMe) {
|
||||||
throw new Error(this.$rootGetters['i18n/t']('harvester.usb.cantUnclaim', { name: escapeHtml(this.metadata.name) }));
|
throw new Error(this.$rootGetters['i18n/t']('harvester.usb.cantUnclaim', { name: escapeHtml(this.metadata.name) }));
|
||||||
@ -158,4 +164,20 @@ export default class USBDevice extends SteveModel {
|
|||||||
get groupByDevice() {
|
get groupByDevice() {
|
||||||
return this.status?.description;
|
return this.status?.description;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
showDetachWarning() {
|
||||||
|
this.$dispatch('growl/warning', {
|
||||||
|
title: this.$rootGetters['i18n/t']('harvester.usb.detachWarning.title'),
|
||||||
|
message: this.$rootGetters['i18n/t']('harvester.usb.detachWarning.message'),
|
||||||
|
timeout: 5000
|
||||||
|
}, { root: true });
|
||||||
|
}
|
||||||
|
|
||||||
|
get allowDisable() {
|
||||||
|
return this._allowDisable;
|
||||||
|
}
|
||||||
|
|
||||||
|
set allowDisable(value) {
|
||||||
|
this._allowDisable = value;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -100,6 +100,12 @@ export default class VGpuDevice extends SteveModel {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async disableVGpu() {
|
async disableVGpu() {
|
||||||
|
if (!this.allowDisable) {
|
||||||
|
this.showDetachWarning();
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const { vGPUTypeName, enabled } = this.spec;
|
const { vGPUTypeName, enabled } = this.spec;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
@ -126,4 +132,20 @@ export default class VGpuDevice extends SteveModel {
|
|||||||
get vGpuAvailableTypes() {
|
get vGpuAvailableTypes() {
|
||||||
return this.status?.availableTypes ? Object.keys(this.status.availableTypes) : [];
|
return this.status?.availableTypes ? Object.keys(this.status.availableTypes) : [];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
showDetachWarning() {
|
||||||
|
this.$dispatch('growl/warning', {
|
||||||
|
title: this.$rootGetters['i18n/t']('harvester.vgpu.detachWarning.title'),
|
||||||
|
message: this.$rootGetters['i18n/t']('harvester.vgpu.detachWarning.message'),
|
||||||
|
timeout: 5000
|
||||||
|
}, { root: true });
|
||||||
|
}
|
||||||
|
|
||||||
|
get allowDisable() {
|
||||||
|
return this._allowDisable;
|
||||||
|
}
|
||||||
|
|
||||||
|
set allowDisable(value) {
|
||||||
|
this._allowDisable = value;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -354,7 +354,7 @@ export default class HciNode extends HarvesterResource {
|
|||||||
}
|
}
|
||||||
|
|
||||||
get isCordoned() {
|
get isCordoned() {
|
||||||
return (this.isUnSchedulable && !this.isEtcd) || this.hasAction('uncordon');
|
return this.hasAction('uncordon');
|
||||||
}
|
}
|
||||||
|
|
||||||
get isEtcd() {
|
get isEtcd() {
|
||||||
@ -536,8 +536,7 @@ export default class HciNode extends HarvesterResource {
|
|||||||
get isStopped() {
|
get isStopped() {
|
||||||
const inventory = this.inventory || {};
|
const inventory = this.inventory || {};
|
||||||
|
|
||||||
return inventory.spec?.powerActionRequested === 'shutdown' &&
|
return inventory.status?.machinePowerState === 'off';
|
||||||
inventory.status?.powerAction?.actionStatus === 'complete';
|
|
||||||
}
|
}
|
||||||
|
|
||||||
get isStopping() {
|
get isStopping() {
|
||||||
@ -553,8 +552,7 @@ export default class HciNode extends HarvesterResource {
|
|||||||
get isStarted() {
|
get isStarted() {
|
||||||
const inventory = this.inventory || {};
|
const inventory = this.inventory || {};
|
||||||
|
|
||||||
return inventory.spec?.powerActionRequested === 'poweron' &&
|
return inventory.status?.machinePowerState === 'on';
|
||||||
inventory.status?.powerAction?.actionStatus === 'complete';
|
|
||||||
}
|
}
|
||||||
|
|
||||||
get isStarting() {
|
get isStarting() {
|
||||||
|
|||||||
@ -35,15 +35,10 @@ export default class HciPv extends HarvesterResource {
|
|||||||
get availableActions() {
|
get availableActions() {
|
||||||
let out = super._availableActions;
|
let out = super._availableActions;
|
||||||
|
|
||||||
// Longhorn V2 provisioner do not support volume clone feature yet
|
const clone = out.find((action) => action.action === 'goToClone');
|
||||||
if (this.isLonghornV2) {
|
|
||||||
out = out.filter((action) => action.action !== 'goToClone');
|
|
||||||
} else {
|
|
||||||
const clone = out.find((action) => action.action === 'goToClone');
|
|
||||||
|
|
||||||
if (clone) {
|
if (clone) {
|
||||||
clone.action = 'goToCloneVolume';
|
clone.action = 'goToCloneVolume';
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const exportImageAction = {
|
const exportImageAction = {
|
||||||
@ -65,10 +60,6 @@ export default class HciPv extends HarvesterResource {
|
|||||||
takeSnapshotAction,
|
takeSnapshotAction,
|
||||||
...out
|
...out
|
||||||
];
|
];
|
||||||
// TODO: remove this block if Longhorn V2 engine supports restore volume snapshot
|
|
||||||
if (this.isLonghornV2) {
|
|
||||||
out = out.filter((action) => action.action !== takeSnapshotAction.action);
|
|
||||||
}
|
|
||||||
} else { // v1.4 / v1.3
|
} else { // v1.4 / v1.3
|
||||||
if (!this.isLonghorn || !this.isLonghornV2) {
|
if (!this.isLonghorn || !this.isLonghornV2) {
|
||||||
out = [
|
out = [
|
||||||
|
|||||||
@ -113,8 +113,9 @@ export default class HciAddonConfig extends HarvesterResource {
|
|||||||
|
|
||||||
get displayName() {
|
get displayName() {
|
||||||
const isExperimental = this.metadata?.labels?.[HCI_ANNOTATIONS.ADDON_EXPERIMENTAL] === 'true';
|
const isExperimental = this.metadata?.labels?.[HCI_ANNOTATIONS.ADDON_EXPERIMENTAL] === 'true';
|
||||||
|
const name = this.metadata?.labels?.[HCI_ANNOTATIONS.ADDON_DISPLAYNAME] || this.metadata.name;
|
||||||
|
|
||||||
return isExperimental ? `${ this.metadata.name } (${ this.t('generic.experimental') })` : this.metadata.name;
|
return isExperimental ? `${ name } (${ this.t('generic.experimental') })` : name;
|
||||||
}
|
}
|
||||||
|
|
||||||
get customValidationRules() {
|
get customValidationRules() {
|
||||||
|
|||||||
@ -52,11 +52,19 @@ export default class HciSetting extends HarvesterResource {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get clusterRegistrationTLSVerifyFeatureEnabled() {
|
||||||
|
return this.$rootGetters['harvester-common/getFeatureEnabled']('clusterRegistrationTLSVerify');
|
||||||
|
}
|
||||||
|
|
||||||
get customValue() {
|
get customValue() {
|
||||||
if (this.metadata.name === HCI_SETTING.STORAGE_NETWORK) {
|
if (this.metadata.name === HCI_SETTING.STORAGE_NETWORK) {
|
||||||
try {
|
try {
|
||||||
return JSON.stringify(JSON.parse(this.value), null, 2);
|
return JSON.stringify(JSON.parse(this.value), null, 2);
|
||||||
} catch (e) {}
|
} catch (e) {}
|
||||||
|
} else if (this.metadata.name === HCI_SETTING.CLUSTER_REGISTRATION_URL) {
|
||||||
|
try {
|
||||||
|
return this.clusterRegistrationTLSVerifyFeatureEnabled ? JSON.stringify(JSON.parse(this.value), null, 2) : this.value;
|
||||||
|
} catch (e) {}
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
|
|||||||
@ -318,8 +318,21 @@ export default class HciVmImage extends HarvesterResource {
|
|||||||
get uploadImage() {
|
get uploadImage() {
|
||||||
return async(file, opt = {}) => {
|
return async(file, opt = {}) => {
|
||||||
const formData = new FormData();
|
const formData = new FormData();
|
||||||
|
const backend = this.spec?.backend || 'backingimage';
|
||||||
|
const backendFieldMap = {
|
||||||
|
cdi: 'file',
|
||||||
|
backingimage: 'chunk'
|
||||||
|
};
|
||||||
|
const fieldName = backendFieldMap[backend];
|
||||||
|
|
||||||
formData.append('chunk', file);
|
if (!fieldName) {
|
||||||
|
const error = this.t('harvester.image.errors.unsupportedBackend', { backend });
|
||||||
|
|
||||||
|
this.$ctx.commit('harvester-common/uploadError', { name: this.name, message: error }, { root: true });
|
||||||
|
throw new Error(error);
|
||||||
|
}
|
||||||
|
|
||||||
|
formData.append(fieldName, file);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
this.$ctx.commit('harvester-common/uploadStart', this.metadata.name, { root: true });
|
this.$ctx.commit('harvester-common/uploadStart', this.metadata.name, { root: true });
|
||||||
|
|||||||
@ -29,6 +29,16 @@ export default class NetworkAttachmentDef extends SteveModel {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get isSystem() {
|
||||||
|
const systemNamespaces = this.$rootGetters['systemNamespaces'];
|
||||||
|
|
||||||
|
if (systemNamespaces.includes(this.metadata?.namespace)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
get isIpamStatic() {
|
get isIpamStatic() {
|
||||||
return this.parseConfig.ipam?.type === 'static';
|
return this.parseConfig.ipam?.type === 'static';
|
||||||
}
|
}
|
||||||
|
|||||||
@ -87,17 +87,11 @@ const IgnoreMessages = ['pod has unbound immediate PersistentVolumeClaims'];
|
|||||||
|
|
||||||
export default class VirtVm extends HarvesterResource {
|
export default class VirtVm extends HarvesterResource {
|
||||||
get availableActions() {
|
get availableActions() {
|
||||||
let out = super._availableActions;
|
const out = super._availableActions;
|
||||||
|
const clone = out.find((action) => action.action === 'goToClone');
|
||||||
|
|
||||||
// VM attached with Longhorn V2 volume doesn't support clone feature
|
if (clone) {
|
||||||
if (this.longhornV2Volumes.length > 0) {
|
clone.action = 'goToCloneVM';
|
||||||
out = out.filter((action) => action.action !== 'goToClone');
|
|
||||||
} else {
|
|
||||||
const clone = out.find((action) => action.action === 'goToClone');
|
|
||||||
|
|
||||||
if (clone) {
|
|
||||||
clone.action = 'goToCloneVM';
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return [
|
return [
|
||||||
@ -159,7 +153,7 @@ export default class VirtVm extends HarvesterResource {
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
action: 'takeVMSnapshot',
|
action: 'takeVMSnapshot',
|
||||||
enabled: (!!this.actions?.snapshot || !!this.action?.backup) && !this.longhornV2Volumes.length,
|
enabled: (!!this.actions?.snapshot || !!this.actions?.backup),
|
||||||
icon: 'icon icon-snapshot',
|
icon: 'icon icon-snapshot',
|
||||||
label: this.t('harvester.action.vmSnapshot')
|
label: this.t('harvester.action.vmSnapshot')
|
||||||
},
|
},
|
||||||
@ -189,7 +183,7 @@ export default class VirtVm extends HarvesterResource {
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
action: 'ejectCDROM',
|
action: 'ejectCDROM',
|
||||||
enabled: !!this.actions?.ejectCdRom,
|
enabled: !this.hotplugCdRomEnabled && !!this.actions?.ejectCdRom,
|
||||||
icon: 'icon icon-delete',
|
icon: 'icon icon-delete',
|
||||||
label: this.t('harvester.action.ejectCDROM')
|
label: this.t('harvester.action.ejectCDROM')
|
||||||
},
|
},
|
||||||
@ -206,10 +200,28 @@ export default class VirtVm extends HarvesterResource {
|
|||||||
label: this.t('harvester.action.abortMigration')
|
label: this.t('harvester.action.abortMigration')
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
action: 'addHotplug',
|
action: 'storageMigration',
|
||||||
|
enabled: !!this.actions?.storageMigration,
|
||||||
|
icon: 'icon icon-copy',
|
||||||
|
label: this.t('harvester.action.storageMigration')
|
||||||
|
},
|
||||||
|
{
|
||||||
|
action: 'cancelStorageMigration',
|
||||||
|
enabled: !!this.actions?.cancelStorageMigration,
|
||||||
|
icon: 'icon icon-close',
|
||||||
|
label: this.t('harvester.action.cancelStorageMigration')
|
||||||
|
},
|
||||||
|
{
|
||||||
|
action: 'addHotplugVolume',
|
||||||
enabled: !!this.actions?.addVolume,
|
enabled: !!this.actions?.addVolume,
|
||||||
icon: 'icon icon-plus',
|
icon: 'icon icon-plus',
|
||||||
label: this.t('harvester.action.addHotplug')
|
label: this.t('harvester.action.addHotplugVolume')
|
||||||
|
},
|
||||||
|
{
|
||||||
|
action: 'addHotplugNic',
|
||||||
|
enabled: this.hotplugNicFeatureEnabled && !!this.actions?.addNic,
|
||||||
|
icon: 'icon icon-plus',
|
||||||
|
label: this.t('harvester.action.addHotplugNic')
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
action: 'createTemplate',
|
action: 'createTemplate',
|
||||||
@ -368,6 +380,13 @@ export default class VirtVm extends HarvesterResource {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
storageMigration(resources = this) {
|
||||||
|
this.$dispatch('promptModal', {
|
||||||
|
resources,
|
||||||
|
component: 'HarvesterStorageMigrationDialog'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
backupVM(resources = this) {
|
backupVM(resources = this) {
|
||||||
this.$dispatch('promptModal', {
|
this.$dispatch('promptModal', {
|
||||||
resources,
|
resources,
|
||||||
@ -395,8 +414,31 @@ export default class VirtVm extends HarvesterResource {
|
|||||||
|
|
||||||
this.$dispatch('promptModal', {
|
this.$dispatch('promptModal', {
|
||||||
resources,
|
resources,
|
||||||
diskName,
|
name: diskName,
|
||||||
component: 'HarvesterUnplugVolume'
|
type: 'volume',
|
||||||
|
component: 'HarvesterHotUnplug',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
ejectCdRomVolume(diskName) {
|
||||||
|
const resources = this;
|
||||||
|
|
||||||
|
this.$dispatch('promptModal', {
|
||||||
|
resources,
|
||||||
|
name: diskName,
|
||||||
|
type: 'cdrom',
|
||||||
|
component: 'HarvesterHotUnplug',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
unplugNIC(networkName) {
|
||||||
|
const resources = this;
|
||||||
|
|
||||||
|
this.$dispatch('promptModal', {
|
||||||
|
resources,
|
||||||
|
name: networkName,
|
||||||
|
type: 'network',
|
||||||
|
component: 'HarvesterHotUnplug',
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -497,6 +539,10 @@ export default class VirtVm extends HarvesterResource {
|
|||||||
this.doActionGrowl('abortMigration', {});
|
this.doActionGrowl('abortMigration', {});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
cancelStorageMigration() {
|
||||||
|
this.doActionGrowl('cancelStorageMigration', {});
|
||||||
|
}
|
||||||
|
|
||||||
createTemplate(resources = this) {
|
createTemplate(resources = this) {
|
||||||
this.$dispatch('promptModal', {
|
this.$dispatch('promptModal', {
|
||||||
resources,
|
resources,
|
||||||
@ -504,10 +550,27 @@ export default class VirtVm extends HarvesterResource {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
addHotplug(resources = this) {
|
addHotplugVolume(resources = this) {
|
||||||
this.$dispatch('promptModal', {
|
this.$dispatch('promptModal', {
|
||||||
resources,
|
resources,
|
||||||
component: 'HarvesterAddHotplugModal'
|
component: 'HarvesterAddHotplugVolumeModal'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
insertCdRomVolume(diskName) {
|
||||||
|
const resources = this;
|
||||||
|
|
||||||
|
this.$dispatch('promptModal', {
|
||||||
|
resources,
|
||||||
|
name: diskName,
|
||||||
|
component: 'HarvesterInsertCdRomVolume',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
addHotplugNic(resources = this) {
|
||||||
|
this.$dispatch('promptModal', {
|
||||||
|
resources,
|
||||||
|
component: 'HarvesterAddHotplugNic'
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -730,11 +793,11 @@ export default class VirtVm extends HarvesterResource {
|
|||||||
}
|
}
|
||||||
|
|
||||||
get isPending() {
|
get isPending() {
|
||||||
if (this &&
|
if ((this &&
|
||||||
!this.isVMExpectedRunning &&
|
!this.isVMExpectedRunning &&
|
||||||
this.isVMCreated &&
|
this.isVMCreated &&
|
||||||
this.vmi?.status?.phase === VMIPhase.Pending
|
this.vmi?.status?.phase === VMIPhase.Pending
|
||||||
) {
|
) || (this.metadata?.annotations?.[HCI_ANNOTATIONS.CLONE_BACKEND_STORAGE_STATUS] === 'cloning')) {
|
||||||
return { status: VMIPhase.Pending };
|
return { status: VMIPhase.Pending };
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1161,11 +1224,15 @@ export default class VirtVm extends HarvesterResource {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
get stateDescription() {
|
get isRestartRequired() {
|
||||||
const conditions = get(this, 'status.conditions');
|
const conditions = get(this, 'status.conditions');
|
||||||
const restartRequired = findBy(conditions, 'type', 'RestartRequired');
|
const restartRequired = findBy(conditions, 'type', 'RestartRequired');
|
||||||
|
|
||||||
if (restartRequired && restartRequired.status === 'True') {
|
return restartRequired && restartRequired.status === 'True';
|
||||||
|
}
|
||||||
|
|
||||||
|
get stateDescription() {
|
||||||
|
if (this.isRestartRequired) {
|
||||||
return this.t('harvester.virtualMachine.hotplug.restartVMMessage');
|
return this.t('harvester.virtualMachine.hotplug.restartVMMessage');
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1240,6 +1307,14 @@ export default class VirtVm extends HarvesterResource {
|
|||||||
return this.$rootGetters['harvester-common/getFeatureEnabled']('vmMachineTypeAuto');
|
return this.$rootGetters['harvester-common/getFeatureEnabled']('vmMachineTypeAuto');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get hotplugNicFeatureEnabled() {
|
||||||
|
return this.$rootGetters['harvester-common/getFeatureEnabled']('hotplugNic');
|
||||||
|
}
|
||||||
|
|
||||||
|
get hotplugCdRomEnabled() {
|
||||||
|
return this.$rootGetters['harvester-common/getFeatureEnabled']('hotplugCdRom');
|
||||||
|
}
|
||||||
|
|
||||||
get isBackupTargetUnavailable() {
|
get isBackupTargetUnavailable() {
|
||||||
const allSettings = this.$rootGetters['harvester/all'](HCI.SETTING) || [];
|
const allSettings = this.$rootGetters['harvester/all'](HCI.SETTING) || [];
|
||||||
const backupTargetSetting = allSettings.find( (O) => O.id === 'backup-target');
|
const backupTargetSetting = allSettings.find( (O) => O.id === 'backup-target');
|
||||||
|
|||||||
@ -1,13 +1,13 @@
|
|||||||
{
|
{
|
||||||
"name": "harvester",
|
"name": "harvester",
|
||||||
"description": "Rancher UI Extension for Harvester",
|
"description": "Rancher UI Extension for Harvester",
|
||||||
"version": "1.7.0-dev",
|
"version": "1.8.0-rc2",
|
||||||
"private": false,
|
"private": false,
|
||||||
"rancher": {
|
"rancher": {
|
||||||
"annotations": {
|
"annotations": {
|
||||||
"catalog.cattle.io/display-name": "Harvester",
|
"catalog.cattle.io/display-name": "Harvester",
|
||||||
"catalog.cattle.io/kube-version": ">= 1.16.0-0",
|
"catalog.cattle.io/kube-version": ">= 1.16.0-0",
|
||||||
"catalog.cattle.io/rancher-version": ">= 2.13.0-0",
|
"catalog.cattle.io/rancher-version": ">= 2.14.0-0",
|
||||||
"catalog.cattle.io/ui-extensions-version": ">= 3.0.0 < 4.0.0"
|
"catalog.cattle.io/ui-extensions-version": ">= 3.0.0 < 4.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|||||||
@ -79,7 +79,6 @@ export default {
|
|||||||
skipSingleReplicaDetachedVolFeatureEnabled() {
|
skipSingleReplicaDetachedVolFeatureEnabled() {
|
||||||
return this.$store.getters['harvester-common/getFeatureEnabled']('skipSingleReplicaDetachedVol');
|
return this.$store.getters['harvester-common/getFeatureEnabled']('skipSingleReplicaDetachedVol');
|
||||||
},
|
},
|
||||||
|
|
||||||
allOSImages() {
|
allOSImages() {
|
||||||
return this.$store.getters['harvester/all'](HCI.IMAGE).filter((I) => I.isOSImage) || [];
|
return this.$store.getters['harvester/all'](HCI.IMAGE).filter((I) => I.isOSImage) || [];
|
||||||
},
|
},
|
||||||
@ -116,7 +115,7 @@ export default {
|
|||||||
},
|
},
|
||||||
|
|
||||||
fileName() {
|
fileName() {
|
||||||
return this.file?.name || '';
|
return this.preprocessImageName(this.file?.name || '');
|
||||||
},
|
},
|
||||||
|
|
||||||
canEnableLogging() {
|
canEnableLogging() {
|
||||||
@ -191,6 +190,7 @@ export default {
|
|||||||
annotations: {}
|
annotations: {}
|
||||||
},
|
},
|
||||||
spec: {
|
spec: {
|
||||||
|
backend: 'cdi',
|
||||||
sourceType: UPLOAD,
|
sourceType: UPLOAD,
|
||||||
displayName: '',
|
displayName: '',
|
||||||
checksum: this.imageValue?.spec?.checksum || '',
|
checksum: this.imageValue?.spec?.checksum || '',
|
||||||
@ -203,8 +203,9 @@ export default {
|
|||||||
|
|
||||||
this.file = {};
|
this.file = {};
|
||||||
this.errors = [];
|
this.errors = [];
|
||||||
|
const imageDisplayName = this.imageValue?.spec?.displayName || '';
|
||||||
|
|
||||||
if (!this.imageValue.spec.displayName && this.createNewImage) {
|
if (!imageDisplayName && this.createNewImage) {
|
||||||
this.errors.push(this.$store.getters['i18n/t']('validation.required', { key: this.t('generic.name') }));
|
this.errors.push(this.$store.getters['i18n/t']('validation.required', { key: this.t('generic.name') }));
|
||||||
buttonCb(false);
|
buttonCb(false);
|
||||||
|
|
||||||
@ -212,24 +213,29 @@ export default {
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
// Save the image first if creating a new one
|
||||||
if (this.imageSource === IMAGE_METHOD.NEW) {
|
if (this.imageSource === IMAGE_METHOD.NEW) {
|
||||||
this.imageValue.metadata.annotations[HCI_ANNOTATIONS.OS_UPGRADE_IMAGE] = 'True';
|
this.imageValue.metadata.annotations[HCI_ANNOTATIONS.OS_UPGRADE_IMAGE] = 'True';
|
||||||
|
|
||||||
if (this.sourceType === UPLOAD && this.uploadImageId !== '') { // upload new image
|
if (this.sourceType === UPLOAD && this.uploadImageId !== '') { // upload new image
|
||||||
this.value.spec.image = this.uploadImageId;
|
this.value.spec.image = this.uploadImageId;
|
||||||
} else if (this.sourceType === DOWNLOAD) { // give URL to download new image
|
} else if (this.sourceType === DOWNLOAD) { // give URL to download new image
|
||||||
this.imageValue.spec.sourceType = DOWNLOAD;
|
// check if URL is provided
|
||||||
if (!this.imageValue.spec.url) {
|
if (!this.imageValue.spec.url) {
|
||||||
this.errors.push(this.$store.getters['i18n/t']('harvester.setting.upgrade.imageUrl'));
|
this.errors.push(this.$store.getters['i18n/t']('harvester.setting.upgrade.imageUrl'));
|
||||||
buttonCb(false);
|
buttonCb(false);
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this.imageValue.spec.sourceType = DOWNLOAD;
|
||||||
|
this.imageValue.spec.targetStorageClassName = 'longhorn-static';
|
||||||
|
|
||||||
res = await this.imageValue.save();
|
res = await this.imageValue.save();
|
||||||
|
|
||||||
this.value.spec.image = res.id;
|
this.value.spec.image = res.id;
|
||||||
}
|
}
|
||||||
} else if (this.imageSource === IMAGE_METHOD.EXIST) {
|
} else if (this.imageSource === IMAGE_METHOD.EXIST) { // select existing image
|
||||||
if (!this.imageId) {
|
if (!this.imageId) {
|
||||||
this.errors.push(this.$store.getters['i18n/t']('harvester.setting.upgrade.chooseFile'));
|
this.errors.push(this.$store.getters['i18n/t']('harvester.setting.upgrade.chooseFile'));
|
||||||
buttonCb(false);
|
buttonCb(false);
|
||||||
@ -239,7 +245,7 @@ export default {
|
|||||||
|
|
||||||
this.value.spec.image = this.imageId;
|
this.value.spec.image = this.imageId;
|
||||||
}
|
}
|
||||||
|
// enable logging or skip single replica detection if checked
|
||||||
if (this.canEnableLogging) {
|
if (this.canEnableLogging) {
|
||||||
this.value.spec.logEnabled = this.enableLogging;
|
this.value.spec.logEnabled = this.enableLogging;
|
||||||
}
|
}
|
||||||
@ -256,7 +262,7 @@ export default {
|
|||||||
},
|
},
|
||||||
|
|
||||||
async uploadFile(file) {
|
async uploadFile(file) {
|
||||||
const fileName = file.name;
|
const fileName = this.preprocessImageName(file.name);
|
||||||
|
|
||||||
if (!fileName) {
|
if (!fileName) {
|
||||||
this.errors.push(this.$store.getters['i18n/t']('harvester.setting.upgrade.unknownImageName'));
|
this.errors.push(this.$store.getters['i18n/t']('harvester.setting.upgrade.unknownImageName'));
|
||||||
@ -280,6 +286,8 @@ export default {
|
|||||||
this.imageValue.spec.url = '';
|
this.imageValue.spec.url = '';
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
this.imageValue.spec.targetStorageClassName = 'longhorn-static';
|
||||||
|
|
||||||
const res = await this.imageValue.save();
|
const res = await this.imageValue.save();
|
||||||
|
|
||||||
this.uploadImageId = res.id;
|
this.uploadImageId = res.id;
|
||||||
@ -301,15 +309,25 @@ export default {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
// replace _ to - to meet storage class name requirement
|
||||||
|
preprocessImageName(name) {
|
||||||
|
if (!name) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
return name.toLowerCase().replace(/[_]/g, '-');
|
||||||
|
},
|
||||||
|
|
||||||
handleImageDelete(imageId) {
|
handleImageDelete(imageId) {
|
||||||
const image = this.allOSImages.find((I) => I.id === imageId);
|
const image = this.allOSImages.find((I) => I.id === imageId);
|
||||||
|
const imageDisplayName = image?.spec?.displayName || '';
|
||||||
|
|
||||||
if (image) {
|
if (image && imageDisplayName) {
|
||||||
this.$store.dispatch('harvester/promptModal', {
|
this.$store.dispatch('harvester/promptModal', {
|
||||||
resources: [image],
|
resources: [image],
|
||||||
component: 'ConfirmRelatedToRemoveDialog',
|
component: 'ConfirmRelatedToRemoveDialog',
|
||||||
needConfirmation: false,
|
needConfirmation: false,
|
||||||
warningMessage: this.$store.getters['i18n/t']('harvester.modal.osImage.message', { name: image.displayName })
|
warningMessage: this.$store.getters['i18n/t']('harvester.modal.osImage.message', { name: imageDisplayName }),
|
||||||
});
|
});
|
||||||
this.deleteImageId = '';
|
this.deleteImageId = '';
|
||||||
}
|
}
|
||||||
@ -419,13 +437,13 @@ export default {
|
|||||||
v-if="showUploadSuccessBanner"
|
v-if="showUploadSuccessBanner"
|
||||||
color="success"
|
color="success"
|
||||||
class="mt-0 mb-30"
|
class="mt-0 mb-30"
|
||||||
:label="t('harvester.setting.upgrade.uploadSuccess', { name: file.name })"
|
:label="t('harvester.setting.upgrade.uploadSuccess', { name: fileName })"
|
||||||
/>
|
/>
|
||||||
<Banner
|
<Banner
|
||||||
v-if="showUploadingWarningBanner"
|
v-if="showUploadingWarningBanner"
|
||||||
color="warning"
|
color="warning"
|
||||||
class="mt-0 mb-30"
|
class="mt-0 mb-30"
|
||||||
:label="t('harvester.image.warning.osUpgrade.uploading', { name: file.name })"
|
:label="t('harvester.image.warning.osUpgrade.uploading', { name: fileName })"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<div
|
<div
|
||||||
|
|||||||
@ -98,6 +98,11 @@ export default {
|
|||||||
if (getters['schemaFor'](HCI.UPGRADE)) {
|
if (getters['schemaFor'](HCI.UPGRADE)) {
|
||||||
hash.upgrades = dispatch('findAll', { type: HCI.UPGRADE });
|
hash.upgrades = dispatch('findAll', { type: HCI.UPGRADE });
|
||||||
}
|
}
|
||||||
|
// Pre-fetch all HCI.UPGRADE_LOG data within loadCluster to ensure HarvesterUpgradeHeader has the necessary data. This is required because the header is dynamically loaded before the user enters the cluster in Rancher integration mode.
|
||||||
|
// See more details in https://github.com/harvester/harvester-ui-extension/pull/715
|
||||||
|
if (getters['schemaFor'](HCI.UPGRADE_LOG)) {
|
||||||
|
hash.upgradeLogs = dispatch('findAll', { type: HCI.UPGRADE_LOG });
|
||||||
|
}
|
||||||
|
|
||||||
const res: any = await allHash(hash);
|
const res: any = await allHash(hash);
|
||||||
|
|
||||||
|
|||||||
4
pkg/harvester/styles/vue-flow.scss
Normal file
4
pkg/harvester/styles/vue-flow.scss
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
@import '@vue-flow/core/dist/style.css';
|
||||||
|
@import '@vue-flow/core/dist/theme-default.css';
|
||||||
|
@import '@vue-flow/controls/dist/style.css';
|
||||||
|
@import '@vue-flow/minimap/dist/style.css';
|
||||||
@ -18,6 +18,7 @@ export const HCI = {
|
|||||||
CLUSTER_NETWORK: 'network.harvesterhci.io.clusternetwork',
|
CLUSTER_NETWORK: 'network.harvesterhci.io.clusternetwork',
|
||||||
SUBNET: 'kubeovn.io.subnet',
|
SUBNET: 'kubeovn.io.subnet',
|
||||||
VPC: 'kubeovn.io.vpc',
|
VPC: 'kubeovn.io.vpc',
|
||||||
|
IP: 'kubeovn.io.ip',
|
||||||
VM_IMAGE_DOWNLOADER: 'harvesterhci.io.virtualmachineimagedownloader',
|
VM_IMAGE_DOWNLOADER: 'harvesterhci.io.virtualmachineimagedownloader',
|
||||||
SUPPORT_BUNDLE: 'harvesterhci.io.supportbundle',
|
SUPPORT_BUNDLE: 'harvesterhci.io.supportbundle',
|
||||||
NETWORK_ATTACHMENT: 'harvesterhci.io.networkattachmentdefinition',
|
NETWORK_ATTACHMENT: 'harvesterhci.io.networkattachmentdefinition',
|
||||||
@ -37,6 +38,7 @@ export const HCI = {
|
|||||||
STORAGE: 'harvesterhci.io.storage',
|
STORAGE: 'harvesterhci.io.storage',
|
||||||
RESOURCE_QUOTA: 'harvesterhci.io.resourcequota',
|
RESOURCE_QUOTA: 'harvesterhci.io.resourcequota',
|
||||||
KSTUNED: 'node.harvesterhci.io.ksmtuned',
|
KSTUNED: 'node.harvesterhci.io.ksmtuned',
|
||||||
|
HUGEPAGES: 'node.harvesterhci.io.hugepage',
|
||||||
PCI_DEVICE: 'devices.harvesterhci.io.pcidevice',
|
PCI_DEVICE: 'devices.harvesterhci.io.pcidevice',
|
||||||
PCI_CLAIM: 'devices.harvesterhci.io.pcideviceclaim',
|
PCI_CLAIM: 'devices.harvesterhci.io.pcideviceclaim',
|
||||||
SR_IOV: 'devices.harvesterhci.io.sriovnetworkdevice',
|
SR_IOV: 'devices.harvesterhci.io.sriovnetworkdevice',
|
||||||
@ -55,6 +57,11 @@ export const HCI = {
|
|||||||
IP_POOL: 'loadbalancer.harvesterhci.io.ippool',
|
IP_POOL: 'loadbalancer.harvesterhci.io.ippool',
|
||||||
HARVESTER_CONFIG: 'rke-machine-config.cattle.io.harvesterconfig',
|
HARVESTER_CONFIG: 'rke-machine-config.cattle.io.harvesterconfig',
|
||||||
LVM_VOLUME_GROUP: 'harvesterhci.io.lvmvolumegroup',
|
LVM_VOLUME_GROUP: 'harvesterhci.io.lvmvolumegroup',
|
||||||
|
VMIMPORT_SOURCE_V: 'migration.harvesterhci.io.vmwaresource',
|
||||||
|
VMIMPORT_SOURCE_O: 'migration.harvesterhci.io.openstacksource',
|
||||||
|
VMIMPORT_SOURCE_OVA: 'migration.harvesterhci.io.ovasource',
|
||||||
|
VMIMPORT: 'migration.harvesterhci.io.virtualmachineimport',
|
||||||
|
MIGRATION: 'migration.harvesterhci.io',
|
||||||
};
|
};
|
||||||
|
|
||||||
export const VOLUME_SNAPSHOT = 'snapshot.storage.k8s.io.volumesnapshot';
|
export const VOLUME_SNAPSHOT = 'snapshot.storage.k8s.io.volumesnapshot';
|
||||||
|
|||||||
99
pkg/harvester/utils/dynamic-nav.js
Normal file
99
pkg/harvester/utils/dynamic-nav.js
Normal file
@ -0,0 +1,99 @@
|
|||||||
|
/**
|
||||||
|
* Dynamically toggles SideNav entries based on the enabled status of a specific Addon.
|
||||||
|
*
|
||||||
|
* @param {Object} store - The Vuex store instance.
|
||||||
|
* @param {String} productName - The product name (e.g. 'harvester').
|
||||||
|
* @param {Object} config - Configuration object.
|
||||||
|
* @param {String} config.addonName - The name of the addon to watch.
|
||||||
|
* @param {String} config.resourceType - The schema ID for addons.
|
||||||
|
* @param {String} config.navGroup - The group name in the side nav.
|
||||||
|
* @param {Array<String>} config.types - Array of Resource IDs to show/hide.
|
||||||
|
*/
|
||||||
|
export function registerAddonSideNav(store, productName, {
|
||||||
|
addonName, resourceType, navGroup, types
|
||||||
|
}) {
|
||||||
|
if (typeof window === 'undefined') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Forces the SideNav component to re-render by toggling a dummy user preference.
|
||||||
|
// Necessary because the menu component does not automatically detect
|
||||||
|
// changes to the allowed types list.
|
||||||
|
const kickSideNav = () => {
|
||||||
|
const TRIGGER = 'ui.refresh.trigger';
|
||||||
|
|
||||||
|
store.dispatch('type-map/addFavorite', TRIGGER);
|
||||||
|
|
||||||
|
// SideNav component seem to ignore rapid state changes.
|
||||||
|
// Wait 600ms to ensure the toggle event triggers a re-render.
|
||||||
|
setTimeout(() => {
|
||||||
|
store.dispatch('type-map/removeFavorite', TRIGGER);
|
||||||
|
}, 600);
|
||||||
|
};
|
||||||
|
|
||||||
|
// Adds or removes the resource IDs from the product visibility whitelist.
|
||||||
|
const setMenuVisibility = (visible) => {
|
||||||
|
if (visible) {
|
||||||
|
store.commit('type-map/basicType', {
|
||||||
|
product: productName,
|
||||||
|
group: navGroup,
|
||||||
|
types
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
// Manually delete the keys from the state object to hide them.
|
||||||
|
const basicTypes = store.state['type-map'].basicTypes[productName];
|
||||||
|
|
||||||
|
if (basicTypes) {
|
||||||
|
types.forEach((t) => delete basicTypes[t]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
kickSideNav();
|
||||||
|
};
|
||||||
|
|
||||||
|
// Start polling to check if the store is ready.
|
||||||
|
let attempts = 0;
|
||||||
|
const MAX_ATTEMPTS = 60;
|
||||||
|
|
||||||
|
const waitForStore = setInterval(() => {
|
||||||
|
attempts++;
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Check if the Schema definitions are loaded.
|
||||||
|
const hasSchema = store.getters[`${ productName }/schemaFor`] &&
|
||||||
|
store.getters[`${ productName }/schemaFor`](resourceType);
|
||||||
|
|
||||||
|
// Check if the resource list data is fully loaded to prevent race conditions.
|
||||||
|
const hasData = store.getters[`${ productName }/haveAll`] &&
|
||||||
|
store.getters[`${ productName }/haveAll`](resourceType);
|
||||||
|
|
||||||
|
if (hasSchema && hasData) {
|
||||||
|
// Store is ready. Stop polling.
|
||||||
|
clearInterval(waitForStore);
|
||||||
|
|
||||||
|
// Watch the specific addon resource for changes to its enabled status.
|
||||||
|
store.watch(
|
||||||
|
(state, getters) => {
|
||||||
|
const addons = getters[`${ productName }/all`](resourceType);
|
||||||
|
const addon = addons.find((a) => a.metadata.name === addonName);
|
||||||
|
|
||||||
|
return addon?.spec?.enabled === true;
|
||||||
|
},
|
||||||
|
(isEnabled) => {
|
||||||
|
setMenuVisibility(isEnabled);
|
||||||
|
},
|
||||||
|
{ immediate: true, deep: true }
|
||||||
|
);
|
||||||
|
} else if (hasSchema && !hasData) {
|
||||||
|
// If the schema is ready but the data is missing, request the list from the API.
|
||||||
|
// Ensures the script does not wait indefinitely if the UI has not loaded the addons yet.
|
||||||
|
store.dispatch(`${ productName }/findAll`, { type: resourceType });
|
||||||
|
} else if (attempts >= MAX_ATTEMPTS) {
|
||||||
|
// Stop checking if the store does not load within the timeout limit.
|
||||||
|
clearInterval(waitForStore);
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
// Ignore errors if the store module is not yet registered and wait for the next attempt.
|
||||||
|
if (attempts >= MAX_ATTEMPTS) clearInterval(waitForStore);
|
||||||
|
}
|
||||||
|
}, 1000);
|
||||||
|
}
|
||||||
@ -9,3 +9,5 @@ export function parseVolumeClaimTemplates(data) {
|
|||||||
|
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const EMPTY_IMAGE = 'EMPTY_IMAGE';
|
||||||
|
|||||||
@ -69,7 +69,7 @@ export function vmDisks(spec, getters, errors, validatorArgs, displayKey, value)
|
|||||||
validName(getters, errors, D.name, diskNames, prefix, type, lowerType, upperType);
|
validName(getters, errors, D.name, diskNames, prefix, type, lowerType, upperType);
|
||||||
});
|
});
|
||||||
|
|
||||||
let requiredVolume = false;
|
let hasBootableVolume = false;
|
||||||
|
|
||||||
_volumes.forEach((V, idx) => {
|
_volumes.forEach((V, idx) => {
|
||||||
const { type, typeValue } = getVolumeType(getters, V, _volumeClaimTemplates, value);
|
const { type, typeValue } = getVolumeType(getters, V, _volumeClaimTemplates, value);
|
||||||
@ -77,7 +77,7 @@ export function vmDisks(spec, getters, errors, validatorArgs, displayKey, value)
|
|||||||
const prefix = V.name || idx + 1;
|
const prefix = V.name || idx + 1;
|
||||||
|
|
||||||
if ([SOURCE_TYPE.IMAGE, SOURCE_TYPE.ATTACH_VOLUME, SOURCE_TYPE.CONTAINER].includes(type)) {
|
if ([SOURCE_TYPE.IMAGE, SOURCE_TYPE.ATTACH_VOLUME, SOURCE_TYPE.CONTAINER].includes(type)) {
|
||||||
requiredVolume = true;
|
hasBootableVolume = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (type === SOURCE_TYPE.NEW || type === SOURCE_TYPE.IMAGE) {
|
if (type === SOURCE_TYPE.NEW || type === SOURCE_TYPE.IMAGE) {
|
||||||
@ -137,10 +137,10 @@ export function vmDisks(spec, getters, errors, validatorArgs, displayKey, value)
|
|||||||
});
|
});
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* At least one volume must be create. (Verify only when create.)
|
* At least one bootable volume must be provided. (Verify only when create.)
|
||||||
*/
|
*/
|
||||||
if ((!requiredVolume || _volumes.length === 0) && !value.links) {
|
if (!hasBootableVolume && !value.links) {
|
||||||
errors.push(getters['i18n/t']('harvester.validation.vm.volume.needImageOrExisting'));
|
errors.push(getters['i18n/t']('harvester.validation.vm.volume.needAtLeastOneBootable'));
|
||||||
}
|
}
|
||||||
|
|
||||||
return errors;
|
return errors;
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user