mirror of
				https://github.com/apache/cloudstack.git
				synced 2025-10-26 01:32:18 +02:00 
			
		
		
		
	Compare commits
	
		
			56 Commits
		
	
	
		
			09b63bc2e8
			...
			8bbe253b00
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 8bbe253b00 | ||
|  | a6068a850c | ||
|  | f63118c011 | ||
|  | 4cdcde2fe7 | ||
|  | d8766418e0 | ||
|  | e7a55a766c | ||
|  | 8b9f5fd8f9 | ||
|  | 03a4b9f4fd | ||
|  | c8d44d92a7 | ||
|  | eee43e534f | ||
|  | 0e8b0b8e40 | ||
|  | b82369c241 | ||
|  | f4b6a74a94 | ||
|  | 4327871036 | ||
|  | 046014b4c5 | ||
|  | 6f931dbd00 | ||
|  | c0a4392b05 | ||
|  | f71d3a8e9f | ||
|  | 9e535e35d2 | ||
|  | dfcbd2e977 | ||
|  | a574f7ac99 | ||
|  | c9ce6e305c | ||
|  | 5e7ae227d3 | ||
|  | 0ca63f36a5 | ||
|  | 349feebd15 | ||
|  | cdb0604e7b | ||
|  | e27528f8b2 | ||
|  | 22ba8dd504 | ||
|  | 0ca267f516 | ||
|  | 8464e46b53 | ||
|  | 0e93ae3bdf | ||
|  | a5a934dac1 | ||
|  | 136ea3eafa | ||
|  | 1e23d6bc20 | ||
|  | 162c45f8fa | ||
|  | 0b9afe77ca | ||
|  | cc6ee906d5 | ||
|  | 86cad79c15 | ||
|  | b106d6e190 | ||
|  | b99a03092f | ||
|  | df49c4f14b | ||
|  | 67250d99d4 | ||
|  | 2b1f0bbbdb | ||
|  | 973819dad6 | ||
|  | f67b738eb3 | ||
|  | 4d95f08a3a | ||
|  | a6ef24d167 | ||
|  | 309b444205 | ||
|  | 6089c161a6 | ||
|  | 89d2b17461 | ||
|  | b143ddc405 | ||
|  | cc3170577c | ||
|  | 9f20979bce | ||
|  | a15fbd9bcc | ||
|  | 270d3f9a2d | ||
|  | 314c4591ec | 
| @ -59,6 +59,7 @@ github: | ||||
|     - abh1sar | ||||
|     - rosi-shapeblue | ||||
|     - sudo87 | ||||
|     - erikbocks | ||||
| 
 | ||||
|   protected_branches: ~ | ||||
| 
 | ||||
|  | ||||
							
								
								
									
										3
									
								
								.github/linters/.markdown-lint.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.github/linters/.markdown-lint.yml
									
									
									
									
										vendored
									
									
								
							| @ -18,9 +18,6 @@ | ||||
| # MD001/heading-increment Heading levels should only increment by one level at a time | ||||
| MD001: false | ||||
| 
 | ||||
| # MD003/heading-style Heading style | ||||
| MD003: false | ||||
| 
 | ||||
| # MD004/ul-style Unordered list style | ||||
| MD004: false | ||||
| 
 | ||||
|  | ||||
							
								
								
									
										2
									
								
								.github/linters/codespell.txt
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/linters/codespell.txt
									
									
									
									
										vendored
									
									
								
							| @ -375,6 +375,7 @@ propogate | ||||
| provison | ||||
| psudo | ||||
| pyhsical | ||||
| re-use | ||||
| readabilty | ||||
| readd | ||||
| reccuring | ||||
| @ -411,7 +412,6 @@ retriving | ||||
| retrun | ||||
| retuned | ||||
| returing | ||||
| re-use | ||||
| rever | ||||
| rocessor | ||||
| runing | ||||
|  | ||||
							
								
								
									
										6
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							| @ -30,17 +30,17 @@ jobs: | ||||
|   build: | ||||
|     runs-on: ubuntu-22.04 | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/checkout@v5 | ||||
| 
 | ||||
|       - name: Set up JDK 17 | ||||
|         uses: actions/setup-java@v4 | ||||
|         uses: actions/setup-java@v5 | ||||
|         with: | ||||
|           distribution: 'temurin' | ||||
|           java-version: '17' | ||||
|           cache: 'maven' | ||||
| 
 | ||||
|       - name: Set up Python | ||||
|         uses: actions/setup-python@v5 | ||||
|         uses: actions/setup-python@v6 | ||||
|         with: | ||||
|           python-version: '3.10' | ||||
|           architecture: 'x64' | ||||
|  | ||||
							
								
								
									
										12
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							| @ -29,7 +29,7 @@ permissions: | ||||
| jobs: | ||||
|   build: | ||||
|     if: github.repository == 'apache/cloudstack' | ||||
|     runs-on: ubuntu-22.04 | ||||
|     runs-on: ubuntu-24.04 | ||||
| 
 | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
| @ -216,19 +216,19 @@ jobs: | ||||
|                   smoke/test_list_volumes"] | ||||
| 
 | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/checkout@v5 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
| 
 | ||||
|       - name: Set up JDK 17 | ||||
|         uses: actions/setup-java@v4 | ||||
|         uses: actions/setup-java@v5 | ||||
|         with: | ||||
|           distribution: 'temurin' | ||||
|           java-version: '17' | ||||
|           cache: 'maven' | ||||
| 
 | ||||
|       - name: Set up Python | ||||
|         uses: actions/setup-python@v5 | ||||
|         uses: actions/setup-python@v6 | ||||
|         with: | ||||
|           python-version: '3.10' | ||||
|           architecture: 'x64' | ||||
| @ -236,7 +236,7 @@ jobs: | ||||
|       - name: Install Build Dependencies | ||||
|         run: | | ||||
|           sudo apt-get update | ||||
|           sudo apt-get install -y git uuid-runtime genisoimage netcat ipmitool build-essential libgcrypt20 libgpg-error-dev libgpg-error0 libopenipmi0 ipmitool libpython3-dev libssl-dev libffi-dev python3-openssl python3-dev python3-setuptools | ||||
|           sudo apt-get install -y git uuid-runtime genisoimage netcat-openbsd ipmitool build-essential libgcrypt20 libgpg-error-dev libgpg-error0 libopenipmi0 ipmitool libpython3-dev libssl-dev libffi-dev python3-openssl python3-dev python3-setuptools | ||||
| 
 | ||||
|       - name: Install Python dependencies | ||||
|         run: | | ||||
| @ -275,7 +275,7 @@ jobs: | ||||
|       - name: Setup Simulator Prerequisites | ||||
|         run: | | ||||
|           sudo python3 -m pip install --upgrade netaddr mysql-connector-python | ||||
|           python3 -m pip install --user --upgrade tools/marvin/dist/Marvin-*.tar.gz | ||||
|           python3 -m pip install --user --upgrade tools/marvin/dist/[mM]arvin-*.tar.gz | ||||
|           mvn -q -Pdeveloper -pl developer -Ddeploydb | ||||
|           mvn -q -Pdeveloper -pl developer -Ddeploydb-simulator | ||||
| 
 | ||||
|  | ||||
							
								
								
									
										4
									
								
								.github/workflows/codecov.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/codecov.yml
									
									
									
									
										vendored
									
									
								
							| @ -32,12 +32,12 @@ jobs: | ||||
|     name: codecov | ||||
|     runs-on: ubuntu-22.04 | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/checkout@v5 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
| 
 | ||||
|       - name: Set up JDK 17 | ||||
|         uses: actions/setup-java@v4 | ||||
|         uses: actions/setup-java@v5 | ||||
|         with: | ||||
|           distribution: 'temurin' | ||||
|           java-version: '17' | ||||
|  | ||||
							
								
								
									
										2
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
								
							| @ -35,7 +35,7 @@ jobs: | ||||
|         language: ["actions"] | ||||
|     steps: | ||||
|       - name: Checkout repository | ||||
|         uses: actions/checkout@v4 | ||||
|         uses: actions/checkout@v5 | ||||
|       - name: Initialize CodeQL | ||||
|         uses: github/codeql-action/init@v3 | ||||
|         with: | ||||
|  | ||||
| @ -47,7 +47,7 @@ jobs: | ||||
|       - name: Set Docker repository name | ||||
|         run: echo "DOCKER_REPOSITORY=apache" >> $GITHUB_ENV | ||||
| 
 | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/checkout@v5 | ||||
| 
 | ||||
|       - name: Set ACS version | ||||
|         run: echo "ACS_VERSION=$(grep '<version>' pom.xml | head -2 | tail -1 | cut -d'>' -f2 |cut -d'<' -f1)" >> $GITHUB_ENV | ||||
|  | ||||
							
								
								
									
										2
									
								
								.github/workflows/linter.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/linter.yml
									
									
									
									
										vendored
									
									
								
							| @ -32,7 +32,7 @@ jobs: | ||||
|     runs-on: ubuntu-22.04 | ||||
|     steps: | ||||
|       - name: Check Out | ||||
|         uses: actions/checkout@v4 | ||||
|         uses: actions/checkout@v5 | ||||
|       - name: Install | ||||
|         run: | | ||||
|           python -m pip install --upgrade pip | ||||
|  | ||||
							
								
								
									
										4
									
								
								.github/workflows/main-sonar-check.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/main-sonar-check.yml
									
									
									
									
										vendored
									
									
								
							| @ -32,12 +32,12 @@ jobs: | ||||
|     name: Main Sonar JaCoCo Build | ||||
|     runs-on: ubuntu-22.04 | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/checkout@v5 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
| 
 | ||||
|       - name: Set up JDK17 | ||||
|         uses: actions/setup-java@v4 | ||||
|         uses: actions/setup-java@v5 | ||||
|         with: | ||||
|           distribution: 'temurin' | ||||
|           java-version: '17' | ||||
|  | ||||
							
								
								
									
										4
									
								
								.github/workflows/rat.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/rat.yml
									
									
									
									
										vendored
									
									
								
							| @ -30,9 +30,9 @@ jobs: | ||||
|   build: | ||||
|     runs-on: ubuntu-22.04 | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/checkout@v5 | ||||
|       - name: Set up JDK 17 | ||||
|         uses: actions/setup-java@v4 | ||||
|         uses: actions/setup-java@v5 | ||||
|         with: | ||||
|           java-version: '17' | ||||
|           distribution: 'adopt' | ||||
|  | ||||
							
								
								
									
										4
									
								
								.github/workflows/sonar-check.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/sonar-check.yml
									
									
									
									
										vendored
									
									
								
							| @ -33,13 +33,13 @@ jobs: | ||||
|     name: Sonar JaCoCo Coverage | ||||
|     runs-on: ubuntu-22.04 | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/checkout@v5 | ||||
|         with: | ||||
|           ref: "refs/pull/${{ github.event.number }}/merge" | ||||
|           fetch-depth: 0 | ||||
| 
 | ||||
|       - name: Set up JDK17 | ||||
|         uses: actions/setup-java@v4 | ||||
|         uses: actions/setup-java@v5 | ||||
|         with: | ||||
|           distribution: 'temurin' | ||||
|           java-version: '17' | ||||
|  | ||||
							
								
								
									
										4
									
								
								.github/workflows/ui.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/ui.yml
									
									
									
									
										vendored
									
									
								
							| @ -31,10 +31,10 @@ jobs: | ||||
|     runs-on: ubuntu-22.04 | ||||
| 
 | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/checkout@v5 | ||||
| 
 | ||||
|       - name: Set up Node | ||||
|         uses: actions/setup-node@v3 | ||||
|         uses: actions/setup-node@v5 | ||||
|         with: | ||||
|           node-version: 16 | ||||
| 
 | ||||
|  | ||||
							
								
								
									
										1
									
								
								.markdownlintignore
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								.markdownlintignore
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1 @@ | ||||
| CHANGES.md | ||||
| @ -32,11 +32,12 @@ repos: | ||||
|         name: run gitleaks | ||||
|         description: detect hardcoded secrets | ||||
|   - repo: https://github.com/pre-commit/pre-commit-hooks | ||||
|     rev: v5.0.0 | ||||
|     rev: v6.0.0 | ||||
|     hooks: | ||||
|       #- id: check-added-large-files | ||||
|       - id: check-case-conflict | ||||
|       #- id: check-executables-have-shebangs | ||||
|       - id: check-illegal-windows-names | ||||
|       - id: check-merge-conflict | ||||
|       - id: check-shebang-scripts-are-executable | ||||
|         files: \.sh$ | ||||
| @ -50,7 +51,7 @@ repos: | ||||
|         exclude: > | ||||
|           (?x) | ||||
|           ^scripts/vm/systemvm/id_rsa\.cloud$| | ||||
|           ^server/src/test/java/org/apache/cloudstack/network/ssl/CertServiceTest.java$| | ||||
|           ^server/src/test/java/org/apache/cloudstack/network/ssl/CertServiceTest\.java$| | ||||
|           ^server/src/test/java/com/cloud/keystore/KeystoreTest\.java$| | ||||
|           ^server/src/test/resources/certs/dsa_self_signed\.key$| | ||||
|           ^server/src/test/resources/certs/non_root\.key$| | ||||
| @ -61,13 +62,15 @@ repos: | ||||
|           ^services/console-proxy/rdpconsole/src/test/doc/rdp-key\.pem$| | ||||
|           ^systemvm/agent/certs/localhost\.key$| | ||||
|           ^systemvm/agent/certs/realhostip\.key$| | ||||
|           ^test/integration/smoke/test_ssl_offloading.py$ | ||||
|           ^test/integration/smoke/test_ssl_offloading\.py$ | ||||
|       - id: end-of-file-fixer | ||||
|         exclude: \.vhd$ | ||||
|       - id: file-contents-sorter | ||||
|         args: [--unique] | ||||
|         files: ^\.github/linters/codespell\.txt$ | ||||
|       - id: fix-byte-order-marker | ||||
|       - id: forbid-submodules | ||||
|       - id: mixed-line-ending | ||||
|         exclude: \.cs$ | ||||
|       - id: trailing-whitespace | ||||
|         files: \.(bat|cfg|cs|css|gitignore|header|in|install|java|md|properties|py|rb|rc|sh|sql|te|template|txt|ucls|vue|xml|xsl|yaml|yml)$|^cloud-cli/bindir/cloud-tool$|^debian/changelog$ | ||||
|         args: [--markdown-linebreak-ext=md] | ||||
|  | ||||
| @ -1,31 +1,28 @@ | ||||
| Contributing to Apache CloudStack (ACS) | ||||
| ======================================= | ||||
| # Contributing to Apache CloudStack (ACS) | ||||
| 
 | ||||
| ## Summary | ||||
| 
 | ||||
| Summary | ||||
| ------- | ||||
| This document covers how to contribute to the ACS project. ACS uses GitHub PRs to manage code contributions. | ||||
| These instructions assume you have a GitHub.com account, so if you don't have one you will have to create one. Your proposed code changes will be published to your own fork of the ACS project, and you will submit a Pull Request for your changes to be added. | ||||
| 
 | ||||
| _Let's get started!!!_ | ||||
| 
 | ||||
| Bug fixes | ||||
| --------- | ||||
| ## Bug fixes | ||||
| 
 | ||||
| It's very important that we can easily track bug fix commits, so their hashes should remain the same in all branches. | ||||
| Therefore, a pull request (PR) that fixes a bug, should be sent against a release branch. | ||||
| This can be either the "current release" or the "previous release", depending on which ones are maintained. | ||||
| Since the goal is a stable main, bug fixes should be "merged forward" to the next branch in order: "previous release" -> "current release" -> main (in other words: old to new) | ||||
| 
 | ||||
| Developing new features | ||||
| ----------------------- | ||||
| ## Developing new features | ||||
| 
 | ||||
| Development should be done in a feature branch, branched off of main. | ||||
| Send a PR(steps below) to get it into main (2x LGTM applies). | ||||
| PR will only be merged when main is open, will be held otherwise until main is open again. | ||||
| No back porting / cherry-picking features to existing branches! | ||||
| 
 | ||||
| PendingReleaseNotes file | ||||
| ------------------------ | ||||
| ## PendingReleaseNotes file | ||||
| 
 | ||||
| When developing a new feature or making a (major) change to an existing feature you are encouraged to append this to the PendingReleaseNotes file so that the Release Manager can | ||||
| use this file as a source of information when compiling the Release Notes for a new release. | ||||
| 
 | ||||
| @ -33,8 +30,7 @@ When adding information to the PendingReleaseNotes file make sure that you write | ||||
| 
 | ||||
| Updating the PendingReleaseNotes file is preferably a part of the original Pull Request, but that is up to the developers' discretion. | ||||
| 
 | ||||
| Fork the code | ||||
| ------------- | ||||
| ## Fork the code | ||||
| 
 | ||||
| In your browser, navigate to: [https://github.com/apache/cloudstack](https://github.com/apache/cloudstack) | ||||
| 
 | ||||
| @ -51,8 +47,7 @@ $ git fetch upstream | ||||
| $ git rebase upstream/main | ||||
| ``` | ||||
| 
 | ||||
| Making changes | ||||
| -------------- | ||||
| ## Making changes | ||||
| 
 | ||||
| It is important that you create a new branch to make changes on and that you do not change the `main` branch (other than to rebase in changes from `upstream/main`).  In this example I will assume you will be making your changes to a branch called `feature_x`.  This `feature_x` branch will be created on your local repository and will be pushed to your forked repository on GitHub.  Once this branch is on your fork you will create a Pull Request for the changes to be added to the ACS project. | ||||
| 
 | ||||
| @ -68,8 +63,7 @@ $ git commit -a -m "descriptive commit message for your changes" | ||||
| 
 | ||||
| > The `-b` specifies that you want to create a new branch called `feature_x`.  You only specify `-b` the first time you checkout because you are creating a new branch.  Once the `feature_x` branch exists, you can later switch to it with only `git checkout feature_x`. | ||||
| 
 | ||||
| Rebase `feature_x` to include updates from `upstream/main` | ||||
| ------------------------------------------------------------ | ||||
| ## Rebase `feature_x` to include updates from `upstream/main` | ||||
| 
 | ||||
| It is important that you maintain an up-to-date `main` branch in your local repository.  This is done by rebasing in the code changes from `upstream/main` (the official ACS project repository) into your local repository.  You will want to do this before you start working on a feature as well as right before you submit your changes as a pull request.  I recommend you do this process periodically while you work to make sure you are working off the most recent project code. | ||||
| 
 | ||||
| @ -89,8 +83,7 @@ $ git rebase main | ||||
| 
 | ||||
| > Now your `feature_x` branch is up-to-date with all the code in `upstream/main`. | ||||
| 
 | ||||
| Make a GitHub Pull Request to contribute your changes | ||||
| ----------------------------------------------------- | ||||
| ## Make a GitHub Pull Request to contribute your changes | ||||
| 
 | ||||
| When you are happy with your changes, and you are ready to contribute them, you will create a Pull Request on GitHub to do so.  This is done by pushing your local changes to your forked repository (default remote name is `origin`) and then initiating a pull request on GitHub. | ||||
| 
 | ||||
| @ -114,8 +107,7 @@ To initiate the pull request, do the following: | ||||
| 
 | ||||
| If you are requested to make modifications to your proposed changes, make the changes locally on your `feature_x` branch, re-push the `feature_x` branch to your fork.  The existing pull request should automatically pick up the change and update accordingly. | ||||
| 
 | ||||
| Cleaning up after a successful pull request | ||||
| ------------------------------------------- | ||||
| ## Cleaning up after a successful pull request | ||||
| 
 | ||||
| Once the `feature_x` branch has been committed into the `upstream/main` branch, your local `feature_x` branch and the `origin/feature_x` branch are no longer needed.  If you want to make additional changes, restart the process with a new branch. | ||||
| 
 | ||||
| @ -129,6 +121,6 @@ $ git branch -D feature_x | ||||
| $ git push origin :feature_x | ||||
| ``` | ||||
| 
 | ||||
| Release Principles | ||||
| ------------------ | ||||
| ## Release Principles | ||||
| 
 | ||||
| Detailed information about ACS release principles is available at https://cwiki.apache.org/confluence/display/CLOUDSTACK/Release+principles+for+Apache+CloudStack+4.6+and+up | ||||
|  | ||||
							
								
								
									
										43
									
								
								PRE-COMMIT.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								PRE-COMMIT.md
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,43 @@ | ||||
| # pre-commit | ||||
| 
 | ||||
| We run [pre-commit](https://pre-commit.com/) with | ||||
| [GitHub Actions](https://github.com/apache/cloudstack/blob/main/.github/workflows/linter.yml) so installation on your | ||||
| local machine is currently optional. | ||||
| 
 | ||||
| The `pre-commit` [configuration file](https://github.com/apache/cloudstack/blob/main/.pre-commit-config.yaml) | ||||
| is in the repository root. Before you can run the hooks, you need to have `pre-commit` installed. `pre-commit` is a | ||||
| [Python package](https://pypi.org/project/pre-commit/). | ||||
| 
 | ||||
| From the repository root run: `pip install -r requirements-dev.txt` to install `pre-commit` and after you install | ||||
| `pre-commit` you will then need to install the pre-commit hooks by running `pre-commit install`. | ||||
| 
 | ||||
| The hooks run when running `git commit` and also from the command line with `pre-commit`. Some of the hooks will auto | ||||
| fix the code after the hooks fail whilst most will print error messages from the linters. If a hook fails the overall | ||||
| commit will fail, and you will need to fix the issues or problems and `git add` and `git commit` again. On `git commit` | ||||
| the hooks will run mostly only against modified files so if you want to test all hooks against all files and when you | ||||
| are adding a new hook you should always run: | ||||
| 
 | ||||
| `pre-commit run --all-files` | ||||
| 
 | ||||
| Sometimes you might need to skip a hook to commit because the hook is stopping you from committing or your computer | ||||
| might not have all the installation requirements for all the hooks. The `SKIP` variable is comma separated for two or | ||||
| more hooks: | ||||
| 
 | ||||
| `SKIP=codespell git commit -m "foo"` | ||||
| 
 | ||||
| The same applies when running pre-commit: | ||||
| 
 | ||||
| `SKIP=codespell pre-commit run --all-files` | ||||
| 
 | ||||
| Occasionally you can have more serious problems when using `pre-commit` with `git commit`. You can use `--no-verify` to | ||||
| commit and stop `pre-commit` from checking the hooks. For example: | ||||
| 
 | ||||
| `git commit --no-verify -m "foo"` | ||||
| 
 | ||||
| If you are having major problems using `pre-commit` you can always uninstall it. | ||||
| 
 | ||||
| To run a single hook use `pre-commit run --all-files <hook_id>` | ||||
| 
 | ||||
| For example just run the `codespell` hook: | ||||
| 
 | ||||
| `pre-commit run --all-files codespell` | ||||
| @ -451,3 +451,9 @@ iscsi.session.cleanup.enabled=false | ||||
| 
 | ||||
| # If set to true, creates VMs as full clones of their templates on KVM hypervisor. Creates as linked clones otherwise. | ||||
| # create.full.clone=false | ||||
| 
 | ||||
| # Instance conversion TMPDIR env var | ||||
| #convert.instance.env.tmpdir= | ||||
| 
 | ||||
| # Instance conversion VIRT_V2V_TMPDIR env var | ||||
| #convert.instance.env.virtv2v.tmpdir= | ||||
|  | ||||
| @ -613,7 +613,7 @@ public class Agent implements HandlerFactory, IAgentControl, AgentStatusUpdater | ||||
|     } | ||||
| 
 | ||||
|     protected String getAgentArch() { | ||||
|         String arch = Script.runSimpleBashScript(Script.getExecutableAbsolutePath("arch"), 1000); | ||||
|         String arch = Script.runSimpleBashScript(Script.getExecutableAbsolutePath("arch"), 2000); | ||||
|         logger.debug("Arch for agent: {} found: {}", _name, arch); | ||||
|         return arch; | ||||
|     } | ||||
|  | ||||
| @ -794,6 +794,20 @@ public class AgentProperties{ | ||||
|      */ | ||||
|     public static final Property<Boolean> VIRTV2V_VERBOSE_ENABLED = new Property<>("virtv2v.verbose.enabled", false); | ||||
| 
 | ||||
|     /** | ||||
|      * Set env TMPDIR var for virt-v2v Instance Conversion from VMware to KVM | ||||
|      * Data type: String.<br> | ||||
|      * Default value: <code>null</code> | ||||
|      */ | ||||
|     public static final Property<String> CONVERT_ENV_TMPDIR = new Property<>("convert.instance.env.tmpdir", null, String.class); | ||||
| 
 | ||||
|     /** | ||||
|      * Set env VIRT_V2V_TMPDIR var for virt-v2v Instance Conversion from VMware to KVM | ||||
|      * Data type: String.<br> | ||||
|      * Default value: <code>null</code> | ||||
|      */ | ||||
|     public static final Property<String> CONVERT_ENV_VIRTV2V_TMPDIR = new Property<>("convert.instance.env.virtv2v.tmpdir", null, String.class); | ||||
| 
 | ||||
|     /** | ||||
|      * BGP controll CIDR | ||||
|      * Data type: String.<br> | ||||
|  | ||||
| @ -172,4 +172,5 @@ public interface KubernetesCluster extends ControlledEntity, com.cloud.utils.fsm | ||||
|     Long getEtcdNodeCount(); | ||||
|     Long getCniConfigId(); | ||||
|     String getCniConfigDetails(); | ||||
|     boolean isCsiEnabled(); | ||||
| } | ||||
|  | ||||
| @ -70,6 +70,8 @@ public interface AutoScaleService { | ||||
| 
 | ||||
|     Counter createCounter(CreateCounterCmd cmd); | ||||
| 
 | ||||
|     Counter getCounter(long counterId); | ||||
| 
 | ||||
|     boolean deleteCounter(long counterId) throws ResourceInUseException; | ||||
| 
 | ||||
|     List<? extends Counter> listCounters(ListCountersCmd cmd); | ||||
|  | ||||
| @ -37,7 +37,7 @@ public interface DiskOffering extends InfrastructureEntity, Identity, InternalId | ||||
|     State getState(); | ||||
| 
 | ||||
|     enum DiskCacheMode { | ||||
|         NONE("none"), WRITEBACK("writeback"), WRITETHROUGH("writethrough"); | ||||
|         NONE("none"), WRITEBACK("writeback"), WRITETHROUGH("writethrough"), HYPERVISOR_DEFAULT("hypervisor_default"); | ||||
| 
 | ||||
|         private final String _diskCacheMode; | ||||
| 
 | ||||
| @ -69,6 +69,8 @@ public interface DiskOffering extends InfrastructureEntity, Identity, InternalId | ||||
| 
 | ||||
|     boolean isCustomized(); | ||||
| 
 | ||||
|     boolean isShared(); | ||||
| 
 | ||||
|     void setDiskSize(long diskSize); | ||||
| 
 | ||||
|     long getDiskSize(); | ||||
| @ -99,7 +101,6 @@ public interface DiskOffering extends InfrastructureEntity, Identity, InternalId | ||||
| 
 | ||||
|     Long getBytesReadRateMaxLength(); | ||||
| 
 | ||||
| 
 | ||||
|     void setBytesWriteRate(Long bytesWriteRate); | ||||
| 
 | ||||
|     Long getBytesWriteRate(); | ||||
| @ -112,7 +113,6 @@ public interface DiskOffering extends InfrastructureEntity, Identity, InternalId | ||||
| 
 | ||||
|     Long getBytesWriteRateMaxLength(); | ||||
| 
 | ||||
| 
 | ||||
|     void setIopsReadRate(Long iopsReadRate); | ||||
| 
 | ||||
|     Long getIopsReadRate(); | ||||
| @ -133,7 +133,6 @@ public interface DiskOffering extends InfrastructureEntity, Identity, InternalId | ||||
| 
 | ||||
|     Long getIopsWriteRateMax(); | ||||
| 
 | ||||
| 
 | ||||
|     void setIopsWriteRateMaxLength(Long iopsWriteRateMaxLength); | ||||
| 
 | ||||
|     Long getIopsWriteRateMaxLength(); | ||||
|  | ||||
| @ -20,7 +20,6 @@ import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
| 
 | ||||
| import com.cloud.user.UserData; | ||||
| import org.apache.cloudstack.api.command.admin.cluster.ListClustersCmd; | ||||
| import org.apache.cloudstack.api.command.admin.config.ListCfgGroupsByCmd; | ||||
| import org.apache.cloudstack.api.command.admin.config.ListCfgsByCmd; | ||||
| @ -72,6 +71,7 @@ import org.apache.cloudstack.api.command.user.vm.GetVMPasswordCmd; | ||||
| import org.apache.cloudstack.api.command.user.vmgroup.UpdateVMGroupCmd; | ||||
| import org.apache.cloudstack.config.Configuration; | ||||
| import org.apache.cloudstack.config.ConfigurationGroup; | ||||
| import org.apache.cloudstack.framework.config.ConfigKey; | ||||
| 
 | ||||
| import com.cloud.alert.Alert; | ||||
| import com.cloud.capacity.Capacity; | ||||
| @ -91,6 +91,7 @@ import com.cloud.storage.GuestOSHypervisor; | ||||
| import com.cloud.storage.GuestOsCategory; | ||||
| import com.cloud.storage.StoragePool; | ||||
| import com.cloud.user.SSHKeyPair; | ||||
| import com.cloud.user.UserData; | ||||
| import com.cloud.utils.Pair; | ||||
| import com.cloud.utils.Ternary; | ||||
| import com.cloud.vm.InstanceGroup; | ||||
| @ -104,6 +105,14 @@ import com.cloud.vm.VirtualMachine.Type; | ||||
| public interface ManagementService { | ||||
|     static final String Name = "management-server"; | ||||
| 
 | ||||
|     ConfigKey<Boolean> JsInterpretationEnabled = new ConfigKey<>("Hidden" | ||||
|             , Boolean.class | ||||
|             , "js.interpretation.enabled" | ||||
|             , "false" | ||||
|             , "Enable/Disable all JavaScript interpretation related functionalities to create or update Javascript rules." | ||||
|             , false | ||||
|             , ConfigKey.Scope.Global); | ||||
| 
 | ||||
|     /** | ||||
|      * returns the a map of the names/values in the configuration table | ||||
|      * | ||||
| @ -509,4 +518,6 @@ public interface ManagementService { | ||||
| 
 | ||||
|     boolean removeManagementServer(RemoveManagementServerCmd cmd); | ||||
| 
 | ||||
|     void checkJsInterpretationAllowedIfNeededForParameterValue(String paramName, boolean paramValue); | ||||
| 
 | ||||
| } | ||||
|  | ||||
| @ -18,6 +18,7 @@ package com.cloud.server; | ||||
| 
 | ||||
| public interface ResourceManagerUtil { | ||||
|     long getResourceId(String resourceId, ResourceTag.ResourceObjectType resourceType); | ||||
|     long getResourceId(String resourceId, ResourceTag.ResourceObjectType resourceType, boolean checkAccess); | ||||
|     String getUuid(String resourceId, ResourceTag.ResourceObjectType resourceType); | ||||
|     ResourceTag.ResourceObjectType getResourceType(String resourceTypeStr); | ||||
|     void checkResourceAccessible(Long accountId, Long domainId, String exceptionMessage); | ||||
|  | ||||
| @ -180,6 +180,8 @@ public interface VolumeApiService { | ||||
|      */ | ||||
|     boolean doesStoragePoolSupportDiskOfferingTags(StoragePool destPool, String diskOfferingTags); | ||||
| 
 | ||||
|     boolean validateConditionsToReplaceDiskOfferingOfVolume(Volume volume, DiskOffering newDiskOffering, StoragePool destPool); | ||||
| 
 | ||||
|     Volume destroyVolume(long volumeId, Account caller, boolean expunge, boolean forceExpunge); | ||||
| 
 | ||||
|     void destroyVolume(long volumeId); | ||||
|  | ||||
| @ -85,7 +85,7 @@ public interface SnapshotApiService { | ||||
|      *            the command that specifies the volume criteria | ||||
|      * @return list of snapshot policies | ||||
|      */ | ||||
|     Pair<List<? extends SnapshotPolicy>, Integer> listPoliciesforVolume(ListSnapshotPoliciesCmd cmd); | ||||
|     Pair<List<? extends SnapshotPolicy>, Integer> listSnapshotPolicies(ListSnapshotPoliciesCmd cmd); | ||||
| 
 | ||||
|     boolean deleteSnapshotPolicies(DeleteSnapshotPoliciesCmd cmd); | ||||
| 
 | ||||
|  | ||||
| @ -16,11 +16,12 @@ | ||||
| // under the License. | ||||
| package com.cloud.storage.snapshot; | ||||
| 
 | ||||
| import org.apache.cloudstack.acl.ControlledEntity; | ||||
| import org.apache.cloudstack.api.Displayable; | ||||
| import org.apache.cloudstack.api.Identity; | ||||
| import org.apache.cloudstack.api.InternalIdentity; | ||||
| 
 | ||||
| public interface SnapshotPolicy extends Identity, InternalIdentity, Displayable { | ||||
| public interface SnapshotPolicy extends ControlledEntity, Identity, InternalIdentity, Displayable { | ||||
| 
 | ||||
|     long getVolumeId(); | ||||
| 
 | ||||
|  | ||||
| @ -81,6 +81,34 @@ public abstract class AbstractGetUploadParamsCmd extends BaseCmd { | ||||
|         return projectId; | ||||
|     } | ||||
| 
 | ||||
|     public void setName(String name) { | ||||
|         this.name = name; | ||||
|     } | ||||
| 
 | ||||
|     public void setFormat(String format) { | ||||
|         this.format = format; | ||||
|     } | ||||
| 
 | ||||
|     public void setZoneId(Long zoneId) { | ||||
|         this.zoneId = zoneId; | ||||
|     } | ||||
| 
 | ||||
|     public void setChecksum(String checksum) { | ||||
|         this.checksum = checksum; | ||||
|     } | ||||
| 
 | ||||
|     public void setAccountName(String accountName) { | ||||
|         this.accountName = accountName; | ||||
|     } | ||||
| 
 | ||||
|     public void setDomainId(Long domainId) { | ||||
|         this.domainId = domainId; | ||||
|     } | ||||
| 
 | ||||
|     public void setProjectId(Long projectId) { | ||||
|         this.projectId = projectId; | ||||
|     } | ||||
| 
 | ||||
|     public GetUploadParamsResponse createGetUploadParamsResponse(UUID id, URL postURL, String metadata, String timeout, String signature) { | ||||
|         return new GetUploadParamsResponse(id, postURL, metadata, timeout, signature); | ||||
|     } | ||||
|  | ||||
| @ -64,6 +64,7 @@ public class ApiConstants { | ||||
|     public static final String BACKUP_STORAGE_LIMIT = "backupstoragelimit"; | ||||
|     public static final String BACKUP_STORAGE_TOTAL = "backupstoragetotal"; | ||||
|     public static final String BACKUP_VM_OFFERING_REMOVED = "vmbackupofferingremoved"; | ||||
|     public static final String IS_BACKUP_VM_EXPUNGED = "isbackupvmexpunged"; | ||||
|     public static final String BACKUP_TOTAL = "backuptotal"; | ||||
|     public static final String BASE64_IMAGE = "base64image"; | ||||
|     public static final String BGP_PEERS = "bgppeers"; | ||||
| @ -134,6 +135,7 @@ public class ApiConstants { | ||||
|     public static final String CNI_CONFIG_ID = "cniconfigurationid"; | ||||
|     public static final String CNI_CONFIG_DETAILS = "cniconfigdetails"; | ||||
|     public static final String CNI_CONFIG_NAME = "cniconfigname"; | ||||
|     public static final String CSI_ENABLED = "csienabled"; | ||||
|     public static final String COMPONENT = "component"; | ||||
|     public static final String CPU = "CPU"; | ||||
|     public static final String CPU_CORE_PER_SOCKET = "cpucorepersocket"; | ||||
| @ -214,6 +216,7 @@ public class ApiConstants { | ||||
|     public static final String DURATION = "duration"; | ||||
|     public static final String ELIGIBLE = "eligible"; | ||||
|     public static final String EMAIL = "email"; | ||||
|     public static final String ENABLE_CSI = "enablecsi"; | ||||
|     public static final String END_ASN = "endasn"; | ||||
|     public static final String END_DATE = "enddate"; | ||||
|     public static final String END_IP = "endip"; | ||||
| @ -225,6 +228,7 @@ public class ApiConstants { | ||||
|     public static final String EVENT_TYPE = "eventtype"; | ||||
|     public static final String EXPIRES = "expires"; | ||||
|     public static final String EXTRA_CONFIG = "extraconfig"; | ||||
|     public static final String EXTRA_PARAMS = "extraparams"; | ||||
|     public static final String EXTRA_DHCP_OPTION = "extradhcpoption"; | ||||
|     public static final String EXTRA_DHCP_OPTION_NAME = "extradhcpoptionname"; | ||||
|     public static final String EXTRA_DHCP_OPTION_CODE = "extradhcpoptioncode"; | ||||
| @ -243,6 +247,8 @@ public class ApiConstants { | ||||
|     public static final String FIRSTNAME = "firstname"; | ||||
|     public static final String FORCED = "forced"; | ||||
|     public static final String FORCED_DESTROY_LOCAL_STORAGE = "forcedestroylocalstorage"; | ||||
|     public static final String FORCE_CONVERT_TO_POOL = "forceconverttopool"; | ||||
| 
 | ||||
|     public static final String FORCE_DELETE_HOST = "forcedeletehost"; | ||||
|     public static final String FORCE_MS_TO_IMPORT_VM_FILES = "forcemstoimportvmfiles"; | ||||
|     public static final String FORCE_UPDATE_OS_TYPE = "forceupdateostype"; | ||||
| @ -578,6 +584,7 @@ public class ApiConstants { | ||||
|     public static final String SUITABLE_FOR_VM = "suitableforvirtualmachine"; | ||||
|     public static final String SUPPORTS_STORAGE_SNAPSHOT = "supportsstoragesnapshot"; | ||||
|     public static final String TARGET_IQN = "targetiqn"; | ||||
|     public static final String TASKS_FILTER = "tasksfilter"; | ||||
|     public static final String TEMPLATE_FILTER = "templatefilter"; | ||||
|     public static final String TEMPLATE_ID = "templateid"; | ||||
|     public static final String TEMPLATE_IDS = "templateids"; | ||||
| @ -1157,6 +1164,7 @@ public class ApiConstants { | ||||
|     public static final String OVM3_CLUSTER = "ovm3cluster"; | ||||
|     public static final String OVM3_VIP = "ovm3vip"; | ||||
|     public static final String CLEAN_UP_DETAILS = "cleanupdetails"; | ||||
|     public static final String CLEAN_UP_EXTERNAL_DETAILS = "cleanupexternaldetails"; | ||||
|     public static final String CLEAN_UP_PARAMETERS = "cleanupparameters"; | ||||
|     public static final String VIRTUAL_SIZE = "virtualsize"; | ||||
|     public static final String NETSCALER_CONTROLCENTER_ID = "netscalercontrolcenterid"; | ||||
|  | ||||
| @ -17,6 +17,7 @@ | ||||
| 
 | ||||
| package org.apache.cloudstack.api.command.admin.autoscale; | ||||
| 
 | ||||
| import org.apache.cloudstack.context.CallContext; | ||||
| 
 | ||||
| import org.apache.cloudstack.api.APICommand; | ||||
| import org.apache.cloudstack.api.ApiCommandResourceType; | ||||
| @ -89,9 +90,6 @@ public class CreateCounterCmd extends BaseAsyncCreateCmd { | ||||
|         if (ctr != null) { | ||||
|             this.setEntityId(ctr.getId()); | ||||
|             this.setEntityUuid(ctr.getUuid()); | ||||
|             CounterResponse response = _responseGenerator.createCounterResponse(ctr); | ||||
|             response.setResponseName(getCommandName()); | ||||
|             this.setResponseObject(response); | ||||
|         } else { | ||||
|             throw new ServerApiException(ApiErrorCode.INTERNAL_ERROR, "Failed to create Counter with name " + getName()); | ||||
|         } | ||||
| @ -99,6 +97,11 @@ public class CreateCounterCmd extends BaseAsyncCreateCmd { | ||||
| 
 | ||||
|     @Override | ||||
|     public void execute() { | ||||
|         CallContext.current().setEventDetails("Counter ID: " + getEntityId()); | ||||
|         Counter ctr = _autoScaleService.getCounter(getEntityId()); | ||||
|         CounterResponse response = _responseGenerator.createCounterResponse(ctr); | ||||
|         response.setResponseName(getCommandName()); | ||||
|         this.setResponseObject(response); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|  | ||||
| @ -72,6 +72,14 @@ public class UpdateHostCmd extends BaseCmd { | ||||
|     @Parameter(name = ApiConstants.EXTERNAL_DETAILS, type = CommandType.MAP, description = "Details in key/value pairs using format externaldetails[i].keyname=keyvalue. Example: externaldetails[0].endpoint.url=urlvalue", since = "4.21.0") | ||||
|     protected Map externalDetails; | ||||
| 
 | ||||
|     @Parameter(name = ApiConstants.CLEAN_UP_EXTERNAL_DETAILS, | ||||
|             type = CommandType.BOOLEAN, | ||||
|             description = "Optional boolean field, which indicates if external details should be cleaned up or not " + | ||||
|                     "(If set to true, external details removed for this host, externaldetails field ignored; " + | ||||
|                     "if false or not set, no action)", | ||||
|             since = "4.22.0") | ||||
|     protected Boolean cleanupExternalDetails; | ||||
| 
 | ||||
|     ///////////////////////////////////////////////////// | ||||
|     /////////////////// Accessors /////////////////////// | ||||
|     ///////////////////////////////////////////////////// | ||||
| @ -112,6 +120,10 @@ public class UpdateHostCmd extends BaseCmd { | ||||
|         return convertExternalDetailsToMap(externalDetails); | ||||
|     } | ||||
| 
 | ||||
|     public boolean isCleanupExternalDetails() { | ||||
|         return Boolean.TRUE.equals(cleanupExternalDetails); | ||||
|     } | ||||
| 
 | ||||
|     ///////////////////////////////////////////////////// | ||||
|     /////////////// API Implementation/////////////////// | ||||
|     ///////////////////////////////////////////////////// | ||||
|  | ||||
| @ -151,7 +151,7 @@ public class CreateDiskOfferingCmd extends BaseCmd { | ||||
|     @Parameter(name = ApiConstants.CACHE_MODE, | ||||
|             type = CommandType.STRING, | ||||
|             required = false, | ||||
|             description = "the cache mode to use for this disk offering. none, writeback or writethrough", | ||||
|             description = "the cache mode to use for this disk offering. none, writeback, writethrough or hypervisor default. If the hypervisor default cache mode is used on other hypervisors than KVM, it will fall back to none cache mode", | ||||
|             since = "4.14") | ||||
|     private String cacheMode; | ||||
| 
 | ||||
|  | ||||
| @ -190,7 +190,7 @@ public class CreateServiceOfferingCmd extends BaseCmd { | ||||
|     @Parameter(name = ApiConstants.CACHE_MODE, | ||||
|             type = CommandType.STRING, | ||||
|             required = false, | ||||
|             description = "the cache mode to use for this disk offering. none, writeback or writethrough", | ||||
|             description = "the cache mode to use for this disk offering. none, writeback, writethrough or hypervisor default. If the hypervisor default cache mode is used on other hypervisors than KVM, it will fall back to none cache mode", | ||||
|             since = "4.14") | ||||
|     private String cacheMode; | ||||
| 
 | ||||
|  | ||||
| @ -101,6 +101,14 @@ public class UpdateServiceOfferingCmd extends BaseCmd { | ||||
|             since = "4.21.0") | ||||
|     private Map externalDetails; | ||||
| 
 | ||||
|     @Parameter(name = ApiConstants.CLEAN_UP_EXTERNAL_DETAILS, | ||||
|             type = CommandType.BOOLEAN, | ||||
|             description = "Optional boolean field, which indicates if external details should be cleaned up or not " + | ||||
|                     "(If set to true, external details removed for this offering, externaldetails field ignored; " + | ||||
|                     "if false or not set, no action)", | ||||
|             since = "4.22.0") | ||||
|     protected Boolean cleanupExternalDetails; | ||||
| 
 | ||||
|     ///////////////////////////////////////////////////// | ||||
|     /////////////////// Accessors /////////////////////// | ||||
|     ///////////////////////////////////////////////////// | ||||
| @ -205,6 +213,10 @@ public class UpdateServiceOfferingCmd extends BaseCmd { | ||||
|         return convertExternalDetailsToMap(externalDetails); | ||||
|     } | ||||
| 
 | ||||
|     public boolean isCleanupExternalDetails() { | ||||
|         return Boolean.TRUE.equals(cleanupExternalDetails); | ||||
|     } | ||||
| 
 | ||||
|     ///////////////////////////////////////////////////// | ||||
|     /////////////// API Implementation/////////////////// | ||||
|     ///////////////////////////////////////////////////// | ||||
|  | ||||
| @ -159,6 +159,18 @@ public class ImportVmCmd extends ImportUnmanagedInstanceCmd { | ||||
|             description = "(only for importing VMs from VMware to KVM) optional - if true, forces MS to export OVF from VMware to temporary storage, else uses KVM Host if ovftool is available, falls back to MS if not.") | ||||
|     private Boolean forceMsToImportVmFiles; | ||||
| 
 | ||||
|     @Parameter(name = ApiConstants.EXTRA_PARAMS, | ||||
|             type = CommandType.STRING, | ||||
|             since = "4.22", | ||||
|             description = "(only for importing VMs from VMware to KVM) optional - extra parameters to be passed on the virt-v2v command, if allowed by the administrator") | ||||
|     private String extraParams; | ||||
| 
 | ||||
|     @Parameter(name = ApiConstants.FORCE_CONVERT_TO_POOL, | ||||
|             type = CommandType.BOOLEAN, | ||||
|             since = "4.22", | ||||
|             description = "(only for importing VMs from VMware to KVM) optional - if true, forces virt-v2v conversions to write directly on the provided storage pool (avoid using temporary conversion pool).") | ||||
|     private Boolean forceConvertToPool; | ||||
| 
 | ||||
|     ///////////////////////////////////////////////////// | ||||
|     /////////////////// Accessors /////////////////////// | ||||
|     ///////////////////////////////////////////////////// | ||||
| @ -248,6 +260,14 @@ public class ImportVmCmd extends ImportUnmanagedInstanceCmd { | ||||
|         return EventTypes.EVENT_VM_IMPORT; | ||||
|     } | ||||
| 
 | ||||
|     public String getExtraParams() { | ||||
|         return extraParams; | ||||
|     } | ||||
| 
 | ||||
|     public boolean getForceConvertToPool() { | ||||
|         return BooleanUtils.toBooleanDefaultIfNull(forceConvertToPool, false); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public String getEventDescription() { | ||||
|         String vmName = getName(); | ||||
|  | ||||
| @ -0,0 +1,116 @@ | ||||
| // Licensed to the Apache Software Foundation (ASF) under one | ||||
| // or more contributor license agreements.  See the NOTICE file | ||||
| // distributed with this work for additional information | ||||
| // regarding copyright ownership.  The ASF licenses this file | ||||
| // to you 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 | ||||
| // | ||||
| // 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. | ||||
| package org.apache.cloudstack.api.command.admin.vm; | ||||
| 
 | ||||
| import com.cloud.exception.ConcurrentOperationException; | ||||
| import com.cloud.exception.InsufficientCapacityException; | ||||
| import com.cloud.exception.NetworkRuleConflictException; | ||||
| import com.cloud.exception.ResourceAllocationException; | ||||
| import com.cloud.exception.ResourceUnavailableException; | ||||
| import com.cloud.user.Account; | ||||
| import org.apache.cloudstack.acl.RoleType; | ||||
| import org.apache.cloudstack.api.APICommand; | ||||
| import org.apache.cloudstack.api.ApiConstants; | ||||
| import org.apache.cloudstack.api.BaseListCmd; | ||||
| import org.apache.cloudstack.api.Parameter; | ||||
| import org.apache.cloudstack.api.ResponseObject; | ||||
| import org.apache.cloudstack.api.ServerApiException; | ||||
| import org.apache.cloudstack.api.response.AccountResponse; | ||||
| import org.apache.cloudstack.api.response.HostResponse; | ||||
| import org.apache.cloudstack.api.response.ImportVMTaskResponse; | ||||
| import org.apache.cloudstack.api.response.ListResponse; | ||||
| import org.apache.cloudstack.api.response.ZoneResponse; | ||||
| import org.apache.cloudstack.context.CallContext; | ||||
| import org.apache.cloudstack.vm.ImportVmTasksManager; | ||||
| 
 | ||||
| import javax.inject.Inject; | ||||
| 
 | ||||
| @APICommand(name = "listImportVmTasks", | ||||
|         description = "List running import virtual machine tasks from a unmanaged hosts into CloudStack", | ||||
|         responseObject = ImportVMTaskResponse.class, | ||||
|         responseView = ResponseObject.ResponseView.Full, | ||||
|         requestHasSensitiveInfo = false, | ||||
|         authorized = {RoleType.Admin}, | ||||
|         since = "4.22") | ||||
| public class ListImportVMTasksCmd extends BaseListCmd { | ||||
| 
 | ||||
|     @Inject | ||||
|     public ImportVmTasksManager importVmTasksManager; | ||||
| 
 | ||||
|     @Parameter(name = ApiConstants.ZONE_ID, | ||||
|             type = CommandType.UUID, | ||||
|             entityType = ZoneResponse.class, | ||||
|             required = true, | ||||
|             description = "the zone ID") | ||||
|     private Long zoneId; | ||||
| 
 | ||||
|     @Parameter(name = ApiConstants.ACCOUNT_ID, | ||||
|             type = CommandType.UUID, | ||||
|             entityType = AccountResponse.class, | ||||
|             description = "the ID of the Account") | ||||
|     private Long accountId; | ||||
| 
 | ||||
|     @Parameter(name = ApiConstants.VCENTER, | ||||
|             type = CommandType.STRING, | ||||
|             description = "The name/ip of vCenter. Make sure it is IP address or full qualified domain name for host running vCenter server.") | ||||
|     private String vcenter; | ||||
| 
 | ||||
|     @Parameter(name = ApiConstants.CONVERT_INSTANCE_HOST_ID, | ||||
|             type = CommandType.UUID, | ||||
|             entityType = HostResponse.class, | ||||
|             description = "Conversion host of the importing task") | ||||
|     private Long convertHostId; | ||||
| 
 | ||||
|     @Parameter(name = ApiConstants.TASKS_FILTER, type = CommandType.STRING, description = "Filter tasks by state, valid options are: All, Running, Completed, Failed") | ||||
|     private String tasksFilter; | ||||
| 
 | ||||
|     public Long getZoneId() { | ||||
|         return zoneId; | ||||
|     } | ||||
| 
 | ||||
|     public Long getAccountId() { | ||||
|         return accountId; | ||||
|     } | ||||
| 
 | ||||
|     public String getVcenter() { | ||||
|         return vcenter; | ||||
|     } | ||||
| 
 | ||||
|     public Long getConvertHostId() { | ||||
|         return convertHostId; | ||||
|     } | ||||
| 
 | ||||
|     public String getTasksFilter() { | ||||
|         return tasksFilter; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public void execute() throws ResourceUnavailableException, InsufficientCapacityException, ServerApiException, ConcurrentOperationException, ResourceAllocationException, NetworkRuleConflictException { | ||||
|         ListResponse<ImportVMTaskResponse> response = importVmTasksManager.listImportVMTasks(this); | ||||
|         response.setResponseName(getCommandName()); | ||||
|         setResponseObject(response); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public long getEntityOwnerId() { | ||||
|         Account account = CallContext.current().getCallingAccount(); | ||||
|         if (account != null) { | ||||
|             return account.getId(); | ||||
|         } | ||||
|         return Account.ACCOUNT_ID_SYSTEM; | ||||
|     } | ||||
| } | ||||
| @ -24,7 +24,7 @@ import org.apache.cloudstack.acl.RoleType; | ||||
| import org.apache.cloudstack.api.APICommand; | ||||
| import org.apache.cloudstack.api.ApiConstants; | ||||
| import org.apache.cloudstack.api.ApiErrorCode; | ||||
| import org.apache.cloudstack.api.BaseCmd; | ||||
| import org.apache.cloudstack.api.BaseListProjectAndAccountResourcesCmd; | ||||
| import org.apache.cloudstack.api.Parameter; | ||||
| import org.apache.cloudstack.api.ServerApiException; | ||||
| import org.apache.cloudstack.api.response.BackupScheduleResponse; | ||||
| @ -39,7 +39,6 @@ import com.cloud.exception.InsufficientCapacityException; | ||||
| import com.cloud.exception.NetworkRuleConflictException; | ||||
| import com.cloud.exception.ResourceAllocationException; | ||||
| import com.cloud.exception.ResourceUnavailableException; | ||||
| import com.cloud.utils.exception.CloudRuntimeException; | ||||
| 
 | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| @ -48,10 +47,10 @@ import java.util.List; | ||||
|         description = "List backup schedule of a VM", | ||||
|         responseObject = BackupScheduleResponse.class, since = "4.14.0", | ||||
|         authorized = {RoleType.Admin, RoleType.ResourceAdmin, RoleType.DomainAdmin, RoleType.User}) | ||||
| public class ListBackupScheduleCmd extends BaseCmd { | ||||
| public class ListBackupScheduleCmd extends BaseListProjectAndAccountResourcesCmd { | ||||
| 
 | ||||
|     @Inject | ||||
|     private BackupManager backupManager; | ||||
|     BackupManager backupManager; | ||||
| 
 | ||||
|     ///////////////////////////////////////////////////// | ||||
|     //////////////// API parameters ///////////////////// | ||||
| @ -60,10 +59,16 @@ public class ListBackupScheduleCmd extends BaseCmd { | ||||
|     @Parameter(name = ApiConstants.VIRTUAL_MACHINE_ID, | ||||
|             type = CommandType.UUID, | ||||
|             entityType = UserVmResponse.class, | ||||
|             required = true, | ||||
|             description = "ID of the VM") | ||||
|     private Long vmId; | ||||
| 
 | ||||
|     @Parameter(name = ApiConstants.ID, | ||||
|             type = CommandType.UUID, | ||||
|             entityType = BackupScheduleResponse.class, | ||||
|             description = "the ID of the backup schedule", | ||||
|             since = "4.22.0") | ||||
|     private Long id; | ||||
| 
 | ||||
|     ///////////////////////////////////////////////////// | ||||
|     /////////////////// Accessors /////////////////////// | ||||
|     ///////////////////////////////////////////////////// | ||||
| @ -72,6 +77,10 @@ public class ListBackupScheduleCmd extends BaseCmd { | ||||
|         return vmId; | ||||
|     } | ||||
| 
 | ||||
|     public Long getId() { | ||||
|         return id; | ||||
|     } | ||||
| 
 | ||||
|     ///////////////////////////////////////////////////// | ||||
|     /////////////// API Implementation/////////////////// | ||||
|     ///////////////////////////////////////////////////// | ||||
| @ -79,19 +88,18 @@ public class ListBackupScheduleCmd extends BaseCmd { | ||||
|     @Override | ||||
|     public void execute() throws ResourceUnavailableException, InsufficientCapacityException, ServerApiException, ConcurrentOperationException, ResourceAllocationException, NetworkRuleConflictException { | ||||
|         try{ | ||||
|             List<BackupSchedule> schedules = backupManager.listBackupSchedule(getVmId()); | ||||
|             List<BackupSchedule> schedules = backupManager.listBackupSchedules(this); | ||||
|             ListResponse<BackupScheduleResponse> response = new ListResponse<>(); | ||||
|             List<BackupScheduleResponse> scheduleResponses = new ArrayList<>(); | ||||
| 
 | ||||
|             if (!CollectionUtils.isNullOrEmpty(schedules)) { | ||||
|                 for (BackupSchedule schedule : schedules) { | ||||
|                     scheduleResponses.add(_responseGenerator.createBackupScheduleResponse(schedule)); | ||||
|                 } | ||||
|             } | ||||
|             response.setResponses(scheduleResponses, schedules.size()); | ||||
|             response.setResponseName(getCommandName()); | ||||
|             setResponseObject(response); | ||||
|             } else { | ||||
|                 throw new CloudRuntimeException("No backup schedule exists for the VM"); | ||||
|             } | ||||
|         } catch (Exception e) { | ||||
|             throw new ServerApiException(ApiErrorCode.INTERNAL_ERROR, e.getMessage()); | ||||
|         } | ||||
|  | ||||
| @ -104,6 +104,29 @@ public class GetUploadParamsForIsoCmd extends AbstractGetUploadParamsCmd { | ||||
|         return osTypeId; | ||||
|     } | ||||
| 
 | ||||
|     public void setBootable(Boolean bootable) { | ||||
|         this.bootable = bootable; | ||||
|     } | ||||
| 
 | ||||
|     public void setDisplayText(String displayText) { | ||||
|         this.displayText = displayText; | ||||
|     } | ||||
| 
 | ||||
|     public void setFeatured(Boolean featured) { | ||||
|         this.featured = featured; | ||||
|     } | ||||
| 
 | ||||
|     public void setPublicIso(Boolean publicIso) { | ||||
|         this.publicIso = publicIso; | ||||
|     } | ||||
| 
 | ||||
|     public void setExtractable(Boolean extractable) { | ||||
|         this.extractable = extractable; | ||||
|     } | ||||
| 
 | ||||
|     public void setOsTypeId(Long osTypeId) { | ||||
|         this.osTypeId = osTypeId; | ||||
|     } | ||||
| 
 | ||||
|     ///////////////////////////////////////////////////// | ||||
|     /////////////// API Implementation/////////////////// | ||||
|  | ||||
| @ -53,6 +53,9 @@ public class ListNetworksCmd extends BaseListRetrieveOnlyResourceCountCmd implem | ||||
|     @Parameter(name = ApiConstants.ID, type = CommandType.UUID, entityType = NetworkResponse.class, description = "list networks by ID") | ||||
|     private Long id; | ||||
| 
 | ||||
|     @Parameter(name = ApiConstants.NAME, type = CommandType.STRING, description = "list networks by name", since = "4.22.0") | ||||
|     private String name; | ||||
| 
 | ||||
|     @Parameter(name = ApiConstants.ZONE_ID, type = CommandType.UUID, entityType = ZoneResponse.class, description = "the zone ID of the network") | ||||
|     private Long zoneId; | ||||
| 
 | ||||
| @ -125,6 +128,10 @@ public class ListNetworksCmd extends BaseListRetrieveOnlyResourceCountCmd implem | ||||
|         return id; | ||||
|     } | ||||
| 
 | ||||
|     public String getName() { | ||||
|         return name; | ||||
|     } | ||||
| 
 | ||||
|     public Long getZoneId() { | ||||
|         return zoneId; | ||||
|     } | ||||
|  | ||||
| @ -23,7 +23,7 @@ import org.apache.cloudstack.acl.RoleType; | ||||
| 
 | ||||
| import org.apache.cloudstack.api.APICommand; | ||||
| import org.apache.cloudstack.api.ApiConstants; | ||||
| import org.apache.cloudstack.api.BaseListCmd; | ||||
| import org.apache.cloudstack.api.BaseListProjectAndAccountResourcesCmd; | ||||
| import org.apache.cloudstack.api.Parameter; | ||||
| import org.apache.cloudstack.api.response.ListResponse; | ||||
| import org.apache.cloudstack.api.response.SnapshotPolicyResponse; | ||||
| @ -34,7 +34,7 @@ import com.cloud.utils.Pair; | ||||
| 
 | ||||
| @APICommand(name = "listSnapshotPolicies", description = "Lists snapshot policies.", responseObject = SnapshotPolicyResponse.class, | ||||
|         requestHasSensitiveInfo = false, responseHasSensitiveInfo = false) | ||||
| public class ListSnapshotPoliciesCmd extends BaseListCmd { | ||||
| public class ListSnapshotPoliciesCmd extends BaseListProjectAndAccountResourcesCmd { | ||||
| 
 | ||||
| 
 | ||||
|     ///////////////////////////////////////////////////// | ||||
| @ -69,13 +69,14 @@ public class ListSnapshotPoliciesCmd extends BaseListCmd { | ||||
|     public Long getId() { | ||||
|         return id; | ||||
|     } | ||||
| 
 | ||||
|     ///////////////////////////////////////////////////// | ||||
|     /////////////// API Implementation/////////////////// | ||||
|     ///////////////////////////////////////////////////// | ||||
| 
 | ||||
|     @Override | ||||
|     public void execute() { | ||||
|         Pair<List<? extends SnapshotPolicy>, Integer> result = _snapshotService.listPoliciesforVolume(this); | ||||
|         Pair<List<? extends SnapshotPolicy>, Integer> result = _snapshotService.listSnapshotPolicies(this); | ||||
|         ListResponse<SnapshotPolicyResponse> response = new ListResponse<SnapshotPolicyResponse>(); | ||||
|         List<SnapshotPolicyResponse> policyResponses = new ArrayList<SnapshotPolicyResponse>(); | ||||
|         for (SnapshotPolicy policy : result.first()) { | ||||
|  | ||||
| @ -28,7 +28,6 @@ import org.apache.cloudstack.api.response.ProjectResponse; | ||||
| import org.apache.cloudstack.api.response.VpnUsersResponse; | ||||
| import org.apache.cloudstack.context.CallContext; | ||||
| 
 | ||||
| import com.cloud.domain.Domain; | ||||
| import com.cloud.event.EventTypes; | ||||
| import com.cloud.network.VpnUser; | ||||
| import com.cloud.user.Account; | ||||
| @ -110,7 +109,6 @@ public class AddVpnUserCmd extends BaseAsyncCreateCmd { | ||||
|     @Override | ||||
|     public void execute() { | ||||
|         VpnUser vpnUser = _entityMgr.findById(VpnUser.class, getEntityId()); | ||||
|         Account account = _entityMgr.findById(Account.class, vpnUser.getAccountId()); | ||||
|         try { | ||||
|             if (!_ravService.applyVpnUsers(vpnUser.getAccountId(), userName)) { | ||||
|                 throw new ServerApiException(ApiErrorCode.INTERNAL_ERROR, "Failed to add vpn user"); | ||||
| @ -118,24 +116,10 @@ public class AddVpnUserCmd extends BaseAsyncCreateCmd { | ||||
|         } catch (Exception ex) { | ||||
|             throw new ServerApiException(ApiErrorCode.INTERNAL_ERROR, ex.getMessage()); | ||||
|         } | ||||
| 
 | ||||
|         VpnUsersResponse vpnResponse = new VpnUsersResponse(); | ||||
|         vpnResponse.setId(vpnUser.getUuid()); | ||||
|         vpnResponse.setUserName(vpnUser.getUsername()); | ||||
|         vpnResponse.setAccountName(account.getAccountName()); | ||||
|         // re-retrieve the vpnuser, as the call to `applyVpnUsers` might have changed the state | ||||
|         vpnUser = _entityMgr.findById(VpnUser.class, getEntityId()); | ||||
|         vpnResponse.setState(vpnUser.getState().toString()); | ||||
| 
 | ||||
|         Domain domain = _entityMgr.findById(Domain.class, account.getDomainId()); | ||||
|         if (domain != null) { | ||||
|             vpnResponse.setDomainId(domain.getUuid()); | ||||
|             vpnResponse.setDomainName(domain.getName()); | ||||
|             vpnResponse.setDomainPath(domain.getPath()); | ||||
|         } | ||||
| 
 | ||||
|         VpnUsersResponse vpnResponse = _responseGenerator.createVpnUserResponse(vpnUser); | ||||
|         vpnResponse.setResponseName(getCommandName()); | ||||
|         vpnResponse.setObjectName("vpnuser"); | ||||
|         setResponseObject(vpnResponse); | ||||
|     } | ||||
| 
 | ||||
|  | ||||
| @ -61,6 +61,10 @@ public class BackupOfferingResponse extends BaseResponse { | ||||
|     @Param(description = "zone name") | ||||
|     private String zoneName; | ||||
| 
 | ||||
|     @SerializedName(ApiConstants.CROSS_ZONE_INSTANCE_CREATION) | ||||
|     @Param(description = "the backups with this offering can be used to create Instances on all Zones", since = "4.22.0") | ||||
|     private Boolean crossZoneInstanceCreation; | ||||
| 
 | ||||
|     @SerializedName(ApiConstants.CREATED) | ||||
|     @Param(description = "the date this backup offering was created") | ||||
|     private Date created; | ||||
| @ -97,6 +101,10 @@ public class BackupOfferingResponse extends BaseResponse { | ||||
|         this.zoneName = zoneName; | ||||
|     } | ||||
| 
 | ||||
|     public void setCrossZoneInstanceCreation(Boolean crossZoneInstanceCreation) { | ||||
|         this.crossZoneInstanceCreation = crossZoneInstanceCreation; | ||||
|     } | ||||
| 
 | ||||
|     public void setCreated(Date created) { | ||||
|         this.created = created; | ||||
|     } | ||||
|  | ||||
| @ -123,6 +123,10 @@ public class BackupResponse extends BaseResponse { | ||||
|     @Param(description = "The backup offering corresponding to this backup was removed from the VM", since = "4.21.0") | ||||
|     private Boolean vmOfferingRemoved; | ||||
| 
 | ||||
|     @SerializedName(ApiConstants.IS_BACKUP_VM_EXPUNGED) | ||||
|     @Param(description = "Indicates whether the VM from which the backup was taken is expunged or not", since = "4.22.0") | ||||
|     private Boolean isVmExpunged; | ||||
| 
 | ||||
|     public String getId() { | ||||
|         return id; | ||||
|     } | ||||
| @ -306,4 +310,8 @@ public class BackupResponse extends BaseResponse { | ||||
|     public void setVmOfferingRemoved(Boolean vmOfferingRemoved) { | ||||
|         this.vmOfferingRemoved = vmOfferingRemoved; | ||||
|     } | ||||
| 
 | ||||
|     public void setVmExpunged(Boolean isVmExpunged) { | ||||
|         this.isVmExpunged = isVmExpunged; | ||||
|     } | ||||
| } | ||||
|  | ||||
| @ -62,6 +62,10 @@ public class GetUploadParamsResponse extends BaseResponse { | ||||
|         setObjectName("getuploadparams"); | ||||
|     } | ||||
| 
 | ||||
|     public UUID getId() { | ||||
|         return id; | ||||
|     } | ||||
| 
 | ||||
|     public void setId(UUID id) { | ||||
|         this.id = id; | ||||
|     } | ||||
|  | ||||
| @ -0,0 +1,257 @@ | ||||
| // | ||||
| // Licensed to the Apache Software Foundation (ASF) under one | ||||
| // or more contributor license agreements.  See the NOTICE file | ||||
| // distributed with this work for additional information | ||||
| // regarding copyright ownership.  The ASF licenses this file | ||||
| // to you 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 | ||||
| // | ||||
| // 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. | ||||
| // | ||||
| package org.apache.cloudstack.api.response; | ||||
| 
 | ||||
| import com.cloud.serializer.Param; | ||||
| import com.google.gson.annotations.SerializedName; | ||||
| import org.apache.cloudstack.api.ApiConstants; | ||||
| import org.apache.cloudstack.api.BaseResponse; | ||||
| 
 | ||||
| import java.util.Date; | ||||
| 
 | ||||
| public class ImportVMTaskResponse extends BaseResponse { | ||||
| 
 | ||||
|     @SerializedName(ApiConstants.ID) | ||||
|     @Param(description = "the ID of importing task") | ||||
|     private String id; | ||||
| 
 | ||||
|     @SerializedName(ApiConstants.ZONE_ID) | ||||
|     @Param(description = "the Zone ID") | ||||
|     private String zoneId; | ||||
| 
 | ||||
|     @SerializedName(ApiConstants.ZONE_NAME) | ||||
|     @Param(description = "the Zone name") | ||||
|     private String zoneName; | ||||
| 
 | ||||
|     @SerializedName(ApiConstants.ACCOUNT) | ||||
|     @Param(description = "the account name") | ||||
|     private String accountName; | ||||
| 
 | ||||
|     @SerializedName(ApiConstants.ACCOUNT_ID) | ||||
|     @Param(description = "the ID of account") | ||||
|     private String accountId; | ||||
| 
 | ||||
|     @SerializedName(ApiConstants.VIRTUAL_MACHINE_ID) | ||||
|     @Param(description = "the ID of the imported VM (after task is completed)") | ||||
|     private String virtualMachineId; | ||||
| 
 | ||||
|     @SerializedName(ApiConstants.DISPLAY_NAME) | ||||
|     @Param(description = "the display name of the importing VM") | ||||
|     private String displayName; | ||||
| 
 | ||||
|     @SerializedName(ApiConstants.STATE) | ||||
|     @Param(description = "the state of the importing VM task") | ||||
|     private String state; | ||||
| 
 | ||||
|     @SerializedName(ApiConstants.VCENTER) | ||||
|     @Param(description = "the vcenter name of the importing VM task") | ||||
|     private String vcenter; | ||||
| 
 | ||||
|     @SerializedName(ApiConstants.DATACENTER_NAME) | ||||
|     @Param(description = "the datacenter name of the importing VM task") | ||||
|     private String datacenterName; | ||||
| 
 | ||||
|     @SerializedName("sourcevmname") | ||||
|     @Param(description = "the source VM name") | ||||
|     private String sourceVMName; | ||||
| 
 | ||||
|     @SerializedName("step") | ||||
|     @Param(description = "the current step on the importing VM task") | ||||
|     private String step; | ||||
| 
 | ||||
|     @SerializedName("stepduration") | ||||
|     @Param(description = "the duration of the current step") | ||||
|     private String stepDuration; | ||||
| 
 | ||||
|     @SerializedName(ApiConstants.DURATION) | ||||
|     @Param(description = "the total task duration") | ||||
|     private String duration; | ||||
| 
 | ||||
|     @SerializedName(ApiConstants.DESCRIPTION) | ||||
|     @Param(description = "the current step description on the importing VM task") | ||||
|     private String description; | ||||
| 
 | ||||
|     @SerializedName(ApiConstants.CONVERT_INSTANCE_HOST_ID) | ||||
|     @Param(description = "the ID of the host on which the instance is being converted") | ||||
|     private String convertInstanceHostId; | ||||
| 
 | ||||
|     @SerializedName("convertinstancehostname") | ||||
|     @Param(description = "the name of the host on which the instance is being converted") | ||||
|     private String convertInstanceHostName; | ||||
| 
 | ||||
|     @SerializedName(ApiConstants.CREATED) | ||||
|     @Param(description = "the create date of the importing task") | ||||
|     private Date created; | ||||
| 
 | ||||
|     @SerializedName(ApiConstants.LAST_UPDATED) | ||||
|     @Param(description = "the last updated date of the importing task") | ||||
|     private Date lastUpdated; | ||||
| 
 | ||||
|     public String getId() { | ||||
|         return id; | ||||
|     } | ||||
| 
 | ||||
|     public void setId(String id) { | ||||
|         this.id = id; | ||||
|     } | ||||
| 
 | ||||
|     public String getZoneId() { | ||||
|         return zoneId; | ||||
|     } | ||||
| 
 | ||||
|     public void setZoneId(String zoneId) { | ||||
|         this.zoneId = zoneId; | ||||
|     } | ||||
| 
 | ||||
|     public String getZoneName() { | ||||
|         return zoneName; | ||||
|     } | ||||
| 
 | ||||
|     public void setZoneName(String zoneName) { | ||||
|         this.zoneName = zoneName; | ||||
|     } | ||||
| 
 | ||||
|     public String getAccountName() { | ||||
|         return accountName; | ||||
|     } | ||||
| 
 | ||||
|     public void setAccountName(String accountName) { | ||||
|         this.accountName = accountName; | ||||
|     } | ||||
| 
 | ||||
|     public String getAccountId() { | ||||
|         return accountId; | ||||
|     } | ||||
| 
 | ||||
|     public void setAccountId(String accountId) { | ||||
|         this.accountId = accountId; | ||||
|     } | ||||
| 
 | ||||
|     public String getVirtualMachineId() { | ||||
|         return virtualMachineId; | ||||
|     } | ||||
| 
 | ||||
|     public void setVirtualMachineId(String virtualMachineId) { | ||||
|         this.virtualMachineId = virtualMachineId; | ||||
|     } | ||||
| 
 | ||||
|     public String getDisplayName() { | ||||
|         return displayName; | ||||
|     } | ||||
| 
 | ||||
|     public void setDisplayName(String displayName) { | ||||
|         this.displayName = displayName; | ||||
|     } | ||||
| 
 | ||||
|     public String getVcenter() { | ||||
|         return vcenter; | ||||
|     } | ||||
| 
 | ||||
|     public void setVcenter(String vcenter) { | ||||
|         this.vcenter = vcenter; | ||||
|     } | ||||
| 
 | ||||
|     public String getDatacenterName() { | ||||
|         return datacenterName; | ||||
|     } | ||||
| 
 | ||||
|     public void setDatacenterName(String datacenterName) { | ||||
|         this.datacenterName = datacenterName; | ||||
|     } | ||||
| 
 | ||||
|     public String getSourceVMName() { | ||||
|         return sourceVMName; | ||||
|     } | ||||
| 
 | ||||
|     public void setSourceVMName(String sourceVMName) { | ||||
|         this.sourceVMName = sourceVMName; | ||||
|     } | ||||
| 
 | ||||
|     public String getStep() { | ||||
|         return step; | ||||
|     } | ||||
| 
 | ||||
|     public void setStep(String step) { | ||||
|         this.step = step; | ||||
|     } | ||||
| 
 | ||||
|     public String getStepDuration() { | ||||
|         return stepDuration; | ||||
|     } | ||||
| 
 | ||||
|     public void setStepDuration(String stepDuration) { | ||||
|         this.stepDuration = stepDuration; | ||||
|     } | ||||
| 
 | ||||
|     public String getDuration() { | ||||
|         return duration; | ||||
|     } | ||||
| 
 | ||||
|     public void setDuration(String duration) { | ||||
|         this.duration = duration; | ||||
|     } | ||||
| 
 | ||||
|     public String getDescription() { | ||||
|         return description; | ||||
|     } | ||||
| 
 | ||||
|     public void setDescription(String description) { | ||||
|         this.description = description; | ||||
|     } | ||||
| 
 | ||||
|     public String getConvertInstanceHostId() { | ||||
|         return convertInstanceHostId; | ||||
|     } | ||||
| 
 | ||||
|     public void setConvertInstanceHostId(String convertInstanceHostId) { | ||||
|         this.convertInstanceHostId = convertInstanceHostId; | ||||
|     } | ||||
| 
 | ||||
|     public String getConvertInstanceHostName() { | ||||
|         return convertInstanceHostName; | ||||
|     } | ||||
| 
 | ||||
|     public void setConvertInstanceHostName(String convertInstanceHostName) { | ||||
|         this.convertInstanceHostName = convertInstanceHostName; | ||||
|     } | ||||
| 
 | ||||
|     public Date getCreated() { | ||||
|         return created; | ||||
|     } | ||||
| 
 | ||||
|     public void setCreated(Date created) { | ||||
|         this.created = created; | ||||
|     } | ||||
| 
 | ||||
|     public Date getLastUpdated() { | ||||
|         return lastUpdated; | ||||
|     } | ||||
| 
 | ||||
|     public void setLastUpdated(Date lastUpdated) { | ||||
|         this.lastUpdated = lastUpdated; | ||||
|     } | ||||
| 
 | ||||
|     public String getState() { | ||||
|         return state; | ||||
|     } | ||||
| 
 | ||||
|     public void setState(String state) { | ||||
|         this.state = state; | ||||
|     } | ||||
| } | ||||
| @ -197,7 +197,7 @@ public class ServiceOfferingResponse extends BaseResponseWithAnnotations { | ||||
|     private Boolean isCustomized; | ||||
| 
 | ||||
|     @SerializedName("cacheMode") | ||||
|     @Param(description = "the cache mode to use for this disk offering. none, writeback or writethrough", since = "4.14") | ||||
|     @Param(description = "the cache mode to use for this disk offering. none, writeback, writethrough or hypervisor default", since = "4.14") | ||||
|     private String cacheMode; | ||||
| 
 | ||||
|     @SerializedName("vspherestoragepolicy") | ||||
|  | ||||
| @ -37,6 +37,10 @@ public class SnapshotPolicyResponse extends BaseResponseWithTagInformation { | ||||
|     @Param(description = "the ID of the disk volume") | ||||
|     private String volumeId; | ||||
| 
 | ||||
|     @SerializedName("volumename") | ||||
|     @Param(description = "the name of the disk volume") | ||||
|     private String volumeName; | ||||
| 
 | ||||
|     @SerializedName("schedule") | ||||
|     @Param(description = "time the snapshot is scheduled to be taken.") | ||||
|     private String schedule; | ||||
| @ -87,6 +91,10 @@ public class SnapshotPolicyResponse extends BaseResponseWithTagInformation { | ||||
|         this.volumeId = volumeId; | ||||
|     } | ||||
| 
 | ||||
|     public void setVolumeName(String volumeName) { | ||||
|         this.volumeName = volumeName; | ||||
|     } | ||||
| 
 | ||||
|     public String getSchedule() { | ||||
|         return schedule; | ||||
|     } | ||||
|  | ||||
| @ -41,7 +41,7 @@ import com.google.gson.annotations.SerializedName; | ||||
| 
 | ||||
| @SuppressWarnings("unused") | ||||
| @EntityReference(value = {VirtualMachine.class, UserVm.class, VirtualRouter.class}) | ||||
| public class UserVmResponse extends BaseResponseWithTagInformation implements ControlledEntityResponse, SetResourceIconResponse { | ||||
| public class UserVmResponse extends BaseResponseWithTagInformation implements ControlledViewEntityResponse, SetResourceIconResponse { | ||||
|     @SerializedName(ApiConstants.ID) | ||||
|     @Param(description = "the ID of the virtual machine") | ||||
|     private String id; | ||||
|  | ||||
| @ -28,6 +28,7 @@ import org.apache.cloudstack.api.command.user.backup.CreateBackupCmd; | ||||
| import org.apache.cloudstack.api.command.user.backup.CreateBackupScheduleCmd; | ||||
| import org.apache.cloudstack.api.command.user.backup.DeleteBackupScheduleCmd; | ||||
| import org.apache.cloudstack.api.command.user.backup.ListBackupOfferingsCmd; | ||||
| import org.apache.cloudstack.api.command.user.backup.ListBackupScheduleCmd; | ||||
| import org.apache.cloudstack.api.command.user.backup.ListBackupsCmd; | ||||
| import org.apache.cloudstack.api.response.BackupResponse; | ||||
| import org.apache.cloudstack.framework.config.ConfigKey; | ||||
| @ -174,7 +175,7 @@ public interface BackupManager extends BackupService, Configurable, PluggableSer | ||||
|      * @param vmId | ||||
|      * @return | ||||
|      */ | ||||
|     List<BackupSchedule> listBackupSchedule(Long vmId); | ||||
|     List<BackupSchedule> listBackupSchedules(ListBackupScheduleCmd cmd); | ||||
| 
 | ||||
|     /** | ||||
|      * Deletes VM backup schedule for a VM | ||||
|  | ||||
| @ -19,11 +19,12 @@ package org.apache.cloudstack.backup; | ||||
| 
 | ||||
| import java.util.Date; | ||||
| 
 | ||||
| import org.apache.cloudstack.acl.ControlledEntity; | ||||
| import org.apache.cloudstack.api.InternalIdentity; | ||||
| 
 | ||||
| import com.cloud.utils.DateUtil; | ||||
| 
 | ||||
| public interface BackupSchedule extends InternalIdentity { | ||||
| public interface BackupSchedule extends ControlledEntity, InternalIdentity { | ||||
|     Long getVmId(); | ||||
|     DateUtil.IntervalType getScheduleType(); | ||||
|     String getSchedule(); | ||||
|  | ||||
							
								
								
									
										41
									
								
								api/src/main/java/org/apache/cloudstack/vm/ImportVmTask.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								api/src/main/java/org/apache/cloudstack/vm/ImportVmTask.java
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,41 @@ | ||||
| /* | ||||
|  * Licensed to the Apache Software Foundation (ASF) under one | ||||
|  * or more contributor license agreements.  See the NOTICE file | ||||
|  * distributed with this work for additional information | ||||
|  * regarding copyright ownership.  The ASF licenses this file | ||||
|  * to you 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 | ||||
|  * | ||||
|  * 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. | ||||
|  */ | ||||
| package org.apache.cloudstack.vm; | ||||
| 
 | ||||
| import org.apache.cloudstack.api.Identity; | ||||
| import org.apache.cloudstack.api.InternalIdentity; | ||||
| 
 | ||||
| public interface ImportVmTask extends Identity, InternalIdentity { | ||||
|     enum Step { | ||||
|         Prepare, CloningInstance, ConvertingInstance, Importing, Completed | ||||
|     } | ||||
| 
 | ||||
|     enum TaskState { | ||||
|         Running, Completed, Failed; | ||||
| 
 | ||||
|         public static TaskState getValue(String state) { | ||||
|             for (TaskState s : TaskState.values()) { | ||||
|                 if (s.name().equalsIgnoreCase(state)) { | ||||
|                     return s; | ||||
|                 } | ||||
|             } | ||||
|             throw new IllegalArgumentException("Invalid task state: " + state); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,38 @@ | ||||
| // Licensed to the Apache Software Foundation (ASF) under one | ||||
| // or more contributor license agreements.  See the NOTICE file | ||||
| // distributed with this work for additional information | ||||
| // regarding copyright ownership.  The ASF licenses this file | ||||
| // to you 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 | ||||
| // | ||||
| // 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. | ||||
| package org.apache.cloudstack.vm; | ||||
| 
 | ||||
| import com.cloud.dc.DataCenter; | ||||
| import com.cloud.host.Host; | ||||
| import com.cloud.user.Account; | ||||
| import org.apache.cloudstack.api.command.admin.vm.ListImportVMTasksCmd; | ||||
| import org.apache.cloudstack.api.response.ImportVMTaskResponse; | ||||
| import org.apache.cloudstack.api.response.ListResponse; | ||||
| 
 | ||||
| public interface ImportVmTasksManager { | ||||
| 
 | ||||
|     ListResponse<ImportVMTaskResponse> listImportVMTasks(ListImportVMTasksCmd cmd); | ||||
| 
 | ||||
|     ImportVmTask createImportVMTaskRecord(DataCenter zone, Account owner, long userId, String displayName, | ||||
|                                             String vcenter, String datacenterName, String sourceVMName, | ||||
|                                             Host convertHost, Host importHost); | ||||
| 
 | ||||
|     void updateImportVMTaskStep(ImportVmTask importVMTaskVO, DataCenter zone, Account owner, Host convertHost, | ||||
|                                 Host importHost, Long vmId, ImportVmTask.Step step); | ||||
| 
 | ||||
|     void updateImportVMTaskErrorState(ImportVmTask importVMTaskVO, ImportVmTask.TaskState state, String errorMsg); | ||||
| } | ||||
| @ -0,0 +1,98 @@ | ||||
| // Licensed to the Apache Software Foundation (ASF) under one | ||||
| // or more contributor license agreements.  See the NOTICE file | ||||
| // distributed with this work for additional information | ||||
| // regarding copyright ownership.  The ASF licenses this file | ||||
| // to you 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 | ||||
| // | ||||
| // 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. | ||||
| package org.apache.cloudstack.api.command.user.backup; | ||||
| 
 | ||||
| import com.cloud.exception.InsufficientCapacityException; | ||||
| import com.cloud.exception.NetworkRuleConflictException; | ||||
| import com.cloud.exception.ResourceAllocationException; | ||||
| import com.cloud.exception.ResourceUnavailableException; | ||||
| import com.cloud.user.Account; | ||||
| import org.apache.cloudstack.api.ResponseGenerator; | ||||
| import org.apache.cloudstack.api.response.BackupScheduleResponse; | ||||
| import org.apache.cloudstack.api.response.ListResponse; | ||||
| import org.apache.cloudstack.backup.BackupManager; | ||||
| import org.apache.cloudstack.backup.BackupSchedule; | ||||
| import org.apache.cloudstack.context.CallContext; | ||||
| import org.junit.Assert; | ||||
| import org.junit.Before; | ||||
| import org.junit.Test; | ||||
| import org.junit.runner.RunWith; | ||||
| import org.mockito.Mock; | ||||
| import org.mockito.Mockito; | ||||
| import org.mockito.junit.MockitoJUnitRunner; | ||||
| 
 | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| 
 | ||||
| @RunWith(MockitoJUnitRunner.class) | ||||
| public class ListBackupScheduleCmdTest { | ||||
| 
 | ||||
|     @Mock | ||||
|     private BackupManager backupManager; | ||||
| 
 | ||||
|     @Mock | ||||
|     private ResponseGenerator responseGenerator; | ||||
| 
 | ||||
|     private ListBackupScheduleCmd cmd; | ||||
| 
 | ||||
|     @Before | ||||
|     public void setUp() { | ||||
|         cmd = new ListBackupScheduleCmd(); | ||||
|         cmd.backupManager = backupManager; | ||||
|         cmd._responseGenerator = responseGenerator; | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void testExecuteWithSchedules() throws ResourceUnavailableException, InsufficientCapacityException, ResourceAllocationException, NetworkRuleConflictException { | ||||
|         BackupSchedule schedule = Mockito.mock(BackupSchedule.class); | ||||
|         BackupScheduleResponse scheduleResponse = Mockito.mock(BackupScheduleResponse.class); | ||||
|         List<BackupSchedule> schedules = new ArrayList<>(); | ||||
|         schedules.add(schedule); | ||||
| 
 | ||||
|         Mockito.when(backupManager.listBackupSchedules(cmd)).thenReturn(schedules); | ||||
|         Mockito.when(responseGenerator.createBackupScheduleResponse(schedule)).thenReturn(scheduleResponse); | ||||
| 
 | ||||
|         Account mockAccount = Mockito.mock(Account.class); | ||||
|         CallContext callContext = Mockito.mock(CallContext.class); | ||||
|         try (org.mockito.MockedStatic<CallContext> mocked = Mockito.mockStatic(CallContext.class)) { | ||||
|             cmd.execute(); | ||||
|         } | ||||
| 
 | ||||
|         ListResponse<?> response = (ListResponse<?>) cmd.getResponseObject(); | ||||
|         Assert.assertNotNull(response); | ||||
|         Assert.assertEquals(1, response.getResponses().size()); | ||||
|         Assert.assertEquals(scheduleResponse, response.getResponses().get(0)); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void testExecuteWithNoSchedules() { | ||||
|         Mockito.when(backupManager.listBackupSchedules(cmd)).thenReturn(new ArrayList<>()); | ||||
|         CallContext callContext = Mockito.mock(CallContext.class); | ||||
| 
 | ||||
|         try (org.mockito.MockedStatic<CallContext> mocked = Mockito.mockStatic(CallContext.class)) { | ||||
|             mocked.when(CallContext::current).thenReturn(callContext); | ||||
|             cmd.execute(); | ||||
|         } catch (ResourceUnavailableException | InsufficientCapacityException | ResourceAllocationException | | ||||
|                  NetworkRuleConflictException e) { | ||||
|             throw new RuntimeException(e); | ||||
|         } | ||||
| 
 | ||||
|         ListResponse<?> response = (ListResponse<?>) cmd.getResponseObject(); | ||||
|         Assert.assertNotNull(response); | ||||
|         Assert.assertEquals(0, response.getResponses().size()); | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,79 @@ | ||||
| // Licensed to the Apache Software Foundation (ASF) under one | ||||
| // or more contributor license agreements.  See the NOTICE file | ||||
| // distributed with this work for additional information | ||||
| // regarding copyright ownership.  The ASF licenses this file | ||||
| // to you 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 | ||||
| // | ||||
| // 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. | ||||
| package org.apache.cloudstack.api.command.user.snapshot; | ||||
| 
 | ||||
| import com.cloud.storage.snapshot.SnapshotApiService; | ||||
| import com.cloud.storage.snapshot.SnapshotPolicy; | ||||
| import com.cloud.utils.Pair; | ||||
| import org.apache.cloudstack.api.ResponseGenerator; | ||||
| import org.apache.cloudstack.api.response.ListResponse; | ||||
| import org.apache.cloudstack.api.response.SnapshotPolicyResponse; | ||||
| import org.junit.Assert; | ||||
| import org.junit.Before; | ||||
| import org.junit.Test; | ||||
| import org.mockito.Mockito; | ||||
| 
 | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| 
 | ||||
| public class ListSnapshotPoliciesCmdTest { | ||||
|     private ListSnapshotPoliciesCmd cmd; | ||||
|     private SnapshotApiService snapshotService; | ||||
|     private ResponseGenerator responseGenerator; | ||||
| 
 | ||||
|     @Before | ||||
|     public void setUp() { | ||||
|         cmd = new ListSnapshotPoliciesCmd(); | ||||
|         snapshotService = Mockito.mock(SnapshotApiService.class); | ||||
|         responseGenerator = Mockito.mock(ResponseGenerator.class); | ||||
| 
 | ||||
|         cmd._snapshotService = snapshotService; | ||||
|         cmd._responseGenerator = responseGenerator; | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void testExecuteWithPolicies() { | ||||
|         SnapshotPolicy policy = Mockito.mock(SnapshotPolicy.class); | ||||
|         SnapshotPolicyResponse policyResponse = Mockito.mock(SnapshotPolicyResponse.class); | ||||
|         List<SnapshotPolicy> policies = new ArrayList<>(); | ||||
|         policies.add(policy); | ||||
| 
 | ||||
|         Mockito.when(snapshotService.listSnapshotPolicies(cmd)) | ||||
|                 .thenReturn(new Pair<>(policies, 1)); | ||||
|         Mockito.when(responseGenerator.createSnapshotPolicyResponse(policy)) | ||||
|                 .thenReturn(policyResponse); | ||||
| 
 | ||||
|         cmd.execute(); | ||||
| 
 | ||||
|         ListResponse<?> response = (ListResponse<?>) cmd.getResponseObject(); | ||||
|         Assert.assertNotNull(response); | ||||
|         Assert.assertEquals(1, response.getResponses().size()); | ||||
|         Assert.assertEquals(policyResponse, response.getResponses().get(0)); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void testExecuteWithNoPolicies() { | ||||
|         Mockito.when(snapshotService.listSnapshotPolicies(cmd)) | ||||
|                 .thenReturn(new Pair<>(new ArrayList<>(), 0)); | ||||
| 
 | ||||
|         cmd.execute(); | ||||
| 
 | ||||
|         ListResponse<?> response = (ListResponse<?>) cmd.getResponseObject(); | ||||
|         Assert.assertNotNull(response); | ||||
|         Assert.assertTrue(response.getResponses().isEmpty()); | ||||
|     } | ||||
| } | ||||
| @ -23,30 +23,37 @@ import com.cloud.hypervisor.Hypervisor; | ||||
| public class ConvertInstanceCommand extends Command { | ||||
| 
 | ||||
|     private RemoteInstanceTO sourceInstance; | ||||
|     private String originalVMName; | ||||
|     private Hypervisor.HypervisorType destinationHypervisorType; | ||||
|     private DataStoreTO conversionTemporaryLocation; | ||||
|     private String templateDirOnConversionLocation; | ||||
|     private boolean checkConversionSupport; | ||||
|     private boolean exportOvfToConversionLocation; | ||||
|     private int threadsCountToExportOvf = 0; | ||||
|     private String extraParams; | ||||
| 
 | ||||
|     public ConvertInstanceCommand() { | ||||
|     } | ||||
| 
 | ||||
|     public ConvertInstanceCommand(RemoteInstanceTO sourceInstance, Hypervisor.HypervisorType destinationHypervisorType, DataStoreTO conversionTemporaryLocation, | ||||
|                                   String templateDirOnConversionLocation, boolean checkConversionSupport, boolean exportOvfToConversionLocation) { | ||||
|                                   String templateDirOnConversionLocation, boolean checkConversionSupport, boolean exportOvfToConversionLocation, String sourceVMName) { | ||||
|         this.sourceInstance = sourceInstance; | ||||
|         this.destinationHypervisorType = destinationHypervisorType; | ||||
|         this.conversionTemporaryLocation = conversionTemporaryLocation; | ||||
|         this.templateDirOnConversionLocation = templateDirOnConversionLocation; | ||||
|         this.checkConversionSupport = checkConversionSupport; | ||||
|         this.exportOvfToConversionLocation = exportOvfToConversionLocation; | ||||
|         this.originalVMName = sourceVMName; | ||||
|     } | ||||
| 
 | ||||
|     public RemoteInstanceTO getSourceInstance() { | ||||
|         return sourceInstance; | ||||
|     } | ||||
| 
 | ||||
|     public String getOriginalVMName() { | ||||
|         return originalVMName; | ||||
|     } | ||||
| 
 | ||||
|     public Hypervisor.HypervisorType getDestinationHypervisorType() { | ||||
|         return destinationHypervisorType; | ||||
|     } | ||||
| @ -75,6 +82,14 @@ public class ConvertInstanceCommand extends Command { | ||||
|         this.threadsCountToExportOvf = threadsCountToExportOvf; | ||||
|     } | ||||
| 
 | ||||
|     public String getExtraParams() { | ||||
|         return extraParams; | ||||
|     } | ||||
| 
 | ||||
|     public void setExtraParams(String extraParams) { | ||||
|         this.extraParams = extraParams; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public boolean executeInSequence() { | ||||
|         return false; | ||||
|  | ||||
| @ -27,17 +27,20 @@ public class ImportConvertedInstanceCommand extends Command { | ||||
|     private List<String> destinationStoragePools; | ||||
|     private DataStoreTO conversionTemporaryLocation; | ||||
|     private String temporaryConvertUuid; | ||||
|     private boolean forceConvertToPool; | ||||
| 
 | ||||
|     public ImportConvertedInstanceCommand() { | ||||
|     } | ||||
| 
 | ||||
|     public ImportConvertedInstanceCommand(RemoteInstanceTO sourceInstance, | ||||
|                                           List<String> destinationStoragePools, | ||||
|                                           DataStoreTO conversionTemporaryLocation, String temporaryConvertUuid) { | ||||
|                                           DataStoreTO conversionTemporaryLocation, String temporaryConvertUuid, | ||||
|                                           boolean forceConvertToPool) { | ||||
|         this.sourceInstance = sourceInstance; | ||||
|         this.destinationStoragePools = destinationStoragePools; | ||||
|         this.conversionTemporaryLocation = conversionTemporaryLocation; | ||||
|         this.temporaryConvertUuid = temporaryConvertUuid; | ||||
|         this.forceConvertToPool = forceConvertToPool; | ||||
|     } | ||||
| 
 | ||||
|     public RemoteInstanceTO getSourceInstance() { | ||||
| @ -56,6 +59,10 @@ public class ImportConvertedInstanceCommand extends Command { | ||||
|         return temporaryConvertUuid; | ||||
|     } | ||||
| 
 | ||||
|     public boolean isForceConvertToPool() { | ||||
|         return forceConvertToPool; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public boolean executeInSequence() { | ||||
|         return false; | ||||
|  | ||||
| @ -116,8 +116,8 @@ public class VolumeObjectTO extends DownloadableObjectTO implements DataTO { | ||||
|         iopsWriteRate = volume.getIopsWriteRate(); | ||||
|         iopsWriteRateMax = volume.getIopsWriteRateMax(); | ||||
|         iopsWriteRateMaxLength = volume.getIopsWriteRateMaxLength(); | ||||
|         cacheMode = volume.getCacheMode(); | ||||
|         hypervisorType = volume.getHypervisorType(); | ||||
|         setCacheMode(volume.getCacheMode()); | ||||
|         setDeviceId(volume.getDeviceId()); | ||||
|         this.migrationOptions = volume.getMigrationOptions(); | ||||
|         this.directDownload = volume.isDirectDownload(); | ||||
| @ -343,6 +343,10 @@ public class VolumeObjectTO extends DownloadableObjectTO implements DataTO { | ||||
|     } | ||||
| 
 | ||||
|     public void setCacheMode(DiskCacheMode cacheMode) { | ||||
|         if (DiskCacheMode.HYPERVISOR_DEFAULT.equals(cacheMode) && !Hypervisor.HypervisorType.KVM.equals(hypervisorType)) { | ||||
|             this.cacheMode = DiskCacheMode.NONE; | ||||
|             return; | ||||
|         } | ||||
|         this.cacheMode = cacheMode; | ||||
|     } | ||||
| 
 | ||||
|  | ||||
							
								
								
									
										1
									
								
								debian/cloudstack-management.install
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								debian/cloudstack-management.install
									
									
									
									
										vendored
									
									
								
							| @ -24,6 +24,7 @@ | ||||
| /etc/cloudstack/management/config.json | ||||
| /etc/cloudstack/extensions/Proxmox/proxmox.sh | ||||
| /etc/cloudstack/extensions/HyperV/hyperv.py | ||||
| /etc/cloudstack/extensions/MaaS/maas.py | ||||
| /etc/default/cloudstack-management | ||||
| /etc/security/limits.d/cloudstack-limits.conf | ||||
| /etc/sudoers.d/cloudstack | ||||
|  | ||||
| @ -23,6 +23,7 @@ import java.util.Map; | ||||
| import java.util.Set; | ||||
| 
 | ||||
| import com.cloud.exception.ResourceAllocationException; | ||||
| import com.cloud.storage.Storage; | ||||
| import com.cloud.utils.Pair; | ||||
| import org.apache.cloudstack.engine.subsystem.api.storage.DataObject; | ||||
| import org.apache.cloudstack.engine.subsystem.api.storage.DataStore; | ||||
| @ -182,10 +183,10 @@ public interface VolumeOrchestrationService { | ||||
|      */ | ||||
|     DiskProfile importVolume(Type type, String name, DiskOffering offering, Long sizeInBytes, Long minIops, Long maxIops, | ||||
|                              Long zoneId, HypervisorType hypervisorType, VirtualMachine vm, VirtualMachineTemplate template, | ||||
|                              Account owner, Long deviceId, Long poolId, String path, String chainInfo); | ||||
|                              Account owner, Long deviceId, Long poolId, Storage.StoragePoolType poolType, String path, String chainInfo); | ||||
| 
 | ||||
|     DiskProfile updateImportedVolume(Type type, DiskOffering offering, VirtualMachine vm, VirtualMachineTemplate template, | ||||
|                                      Long deviceId, Long poolId, String path, String chainInfo, DiskProfile diskProfile); | ||||
|                                      Long deviceId, Long poolId, Storage.StoragePoolType poolType, String path, String chainInfo, DiskProfile diskProfile); | ||||
| 
 | ||||
|     /** | ||||
|      * Unmanage VM volumes | ||||
|  | ||||
| @ -19,6 +19,7 @@ | ||||
| package org.apache.cloudstack.engine.subsystem.api.storage; | ||||
| 
 | ||||
| import com.cloud.storage.ScopeType; | ||||
| import org.apache.cloudstack.utils.reflectiontostringbuilderutils.ReflectionToStringBuilderUtils; | ||||
| 
 | ||||
| public class ClusterScope extends AbstractScope { | ||||
|     private ScopeType type = ScopeType.CLUSTER; | ||||
| @ -51,4 +52,9 @@ public class ClusterScope extends AbstractScope { | ||||
|         return this.zoneId; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public String toString() { | ||||
|         return String.format("ClusterScope %s", ReflectionToStringBuilderUtils.reflectOnlySelectedFields( | ||||
|                 this, "zoneId", "clusterId", "podId")); | ||||
|     } | ||||
| } | ||||
|  | ||||
| @ -19,8 +19,10 @@ | ||||
| package org.apache.cloudstack.engine.subsystem.api.storage; | ||||
| 
 | ||||
| import com.cloud.storage.ScopeType; | ||||
| import org.apache.cloudstack.utils.reflectiontostringbuilderutils.ReflectionToStringBuilderUtils; | ||||
| 
 | ||||
| public class HostScope extends AbstractScope { | ||||
|     private ScopeType type = ScopeType.HOST; | ||||
|     private Long hostId; | ||||
|     private Long clusterId; | ||||
|     private Long zoneId; | ||||
| @ -34,7 +36,7 @@ public class HostScope extends AbstractScope { | ||||
| 
 | ||||
|     @Override | ||||
|     public ScopeType getScopeType() { | ||||
|         return ScopeType.HOST; | ||||
|         return this.type; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
| @ -49,4 +51,10 @@ public class HostScope extends AbstractScope { | ||||
|     public Long getZoneId() { | ||||
|         return zoneId; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public String toString() { | ||||
|         return String.format("HostScope %s", ReflectionToStringBuilderUtils.reflectOnlySelectedFields( | ||||
|                 this, "zoneId", "clusterId", "hostId")); | ||||
|     } | ||||
| } | ||||
|  | ||||
| @ -19,6 +19,7 @@ | ||||
| package org.apache.cloudstack.engine.subsystem.api.storage; | ||||
| 
 | ||||
| import com.cloud.storage.ScopeType; | ||||
| import org.apache.cloudstack.utils.reflectiontostringbuilderutils.ReflectionToStringBuilderUtils; | ||||
| 
 | ||||
| public class ZoneScope extends AbstractScope { | ||||
|     private ScopeType type = ScopeType.ZONE; | ||||
| @ -39,4 +40,9 @@ public class ZoneScope extends AbstractScope { | ||||
|         return this.zoneId; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public String toString() { | ||||
|         return String.format("ZoneScope %s", ReflectionToStringBuilderUtils.reflectOnlySelectedFields( | ||||
|                 this, "zoneId")); | ||||
|     } | ||||
| } | ||||
|  | ||||
| @ -302,6 +302,8 @@ public interface StorageManager extends StorageService { | ||||
| 
 | ||||
|     Answer sendToPool(StoragePool pool, long[] hostIdsToTryFirst, Command cmd) throws StorageUnavailableException; | ||||
| 
 | ||||
|     void updateStoragePoolHostVOAndBytes(StoragePool pool, long hostId, ModifyStoragePoolAnswer mspAnswer); | ||||
| 
 | ||||
|     CapacityVO getSecondaryStorageUsedStats(Long hostId, Long zoneId); | ||||
| 
 | ||||
|     CapacityVO getStoragePoolUsedStats(Long poolId, Long clusterId, Long podId, Long zoneId); | ||||
|  | ||||
| @ -57,6 +57,13 @@ public interface TemplateManager { | ||||
|             + "will validate if the provided URL is resolvable during the register of templates/ISOs before persisting them in the database.", | ||||
|             true); | ||||
| 
 | ||||
|     ConfigKey<Boolean> TemplateDeleteFromPrimaryStorage = new ConfigKey<Boolean>("Advanced", | ||||
|             Boolean.class, | ||||
|             "template.delete.from.primary.storage", "true", | ||||
|             "Template when deleted will be instantly deleted from the Primary Storage", | ||||
|             true, | ||||
|             ConfigKey.Scope.Global); | ||||
| 
 | ||||
|     static final String VMWARE_TOOLS_ISO = "vmware-tools.iso"; | ||||
|     static final String XS_TOOLS_ISO = "xs-tools.iso"; | ||||
| 
 | ||||
| @ -104,6 +111,8 @@ public interface TemplateManager { | ||||
|      */ | ||||
|     List<VMTemplateStoragePoolVO> getUnusedTemplatesInPool(StoragePoolVO pool); | ||||
| 
 | ||||
|     void evictTemplateFromStoragePoolsForZones(Long templateId, List<Long> zoneId); | ||||
| 
 | ||||
|     /** | ||||
|      * Deletes a template in the specified storage pool. | ||||
|      * | ||||
|  | ||||
| @ -2859,6 +2859,7 @@ public class VirtualMachineManagerImpl extends ManagerBase implements VirtualMac | ||||
|             } | ||||
|             volume.setPath(result.getPath()); | ||||
|             volume.setPoolId(pool.getId()); | ||||
|             volume.setPoolType(pool.getPoolType()); | ||||
|             if (result.getChainInfo() != null) { | ||||
|                 volume.setChainInfo(result.getChainInfo()); | ||||
|             } | ||||
|  | ||||
| @ -4799,6 +4799,18 @@ public class NetworkOrchestrator extends ManagerBase implements NetworkOrchestra | ||||
|             } | ||||
|         }); | ||||
| 
 | ||||
|         if (selectedIp != null && GuestType.Shared.equals(network.getGuestType())) { | ||||
|             IPAddressVO ipAddressVO = _ipAddressDao.findByIpAndSourceNetworkId(network.getId(), selectedIp); | ||||
|             if (ipAddressVO != null && IpAddress.State.Free.equals(ipAddressVO.getState())) { | ||||
|                 ipAddressVO.setState(IPAddressVO.State.Allocated); | ||||
|                 ipAddressVO.setAllocatedTime(new Date()); | ||||
|                 Account account = _accountDao.findById(vm.getAccountId()); | ||||
|                 ipAddressVO.setAllocatedInDomainId(account.getDomainId()); | ||||
|                 ipAddressVO.setAllocatedToAccountId(account.getId()); | ||||
|                 _ipAddressDao.update(ipAddressVO.getId(), ipAddressVO); | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         final Integer networkRate = _networkModel.getNetworkRate(network.getId(), vm.getId()); | ||||
|         final NicProfile vmNic = new NicProfile(vo, network, vo.getBroadcastUri(), vo.getIsolationUri(), networkRate, _networkModel.isSecurityGroupSupportedInNetwork(network), | ||||
|                 _networkModel.getNetworkTag(vm.getHypervisorType(), network)); | ||||
| @ -4810,15 +4822,15 @@ public class NetworkOrchestrator extends ManagerBase implements NetworkOrchestra | ||||
|         if (network.getGuestType() == GuestType.L2) { | ||||
|             return null; | ||||
|         } | ||||
|         return dataCenter.getNetworkType() == NetworkType.Basic ? | ||||
|                 getSelectedIpForNicImportOnBasicZone(ipAddresses.getIp4Address(), network, dataCenter): | ||||
|         return GuestType.Shared.equals(network.getGuestType()) ? | ||||
|                 getSelectedIpForNicImportOnSharedNetwork(ipAddresses.getIp4Address(), network, dataCenter): | ||||
|                 _ipAddrMgr.acquireGuestIpAddress(network, ipAddresses.getIp4Address()); | ||||
|     } | ||||
| 
 | ||||
|     protected String getSelectedIpForNicImportOnBasicZone(String requestedIp, Network network, DataCenter dataCenter) { | ||||
|     protected String getSelectedIpForNicImportOnSharedNetwork(String requestedIp, Network network, DataCenter dataCenter) { | ||||
|         IPAddressVO ipAddressVO = StringUtils.isBlank(requestedIp) ? | ||||
|                 _ipAddressDao.findBySourceNetworkIdAndDatacenterIdAndState(network.getId(), dataCenter.getId(), IpAddress.State.Free): | ||||
|                 _ipAddressDao.findByIp(requestedIp); | ||||
|                 _ipAddressDao.findByIpAndSourceNetworkId(network.getId(), requestedIp); | ||||
|         if (ipAddressVO == null || ipAddressVO.getState() != IpAddress.State.Free) { | ||||
|             String msg = String.format("Cannot find a free IP to assign to VM NIC on network %s", network.getName()); | ||||
|             logger.error(msg); | ||||
|  | ||||
| @ -2308,6 +2308,7 @@ public class VolumeOrchestrator extends ManagerBase implements VolumeOrchestrati | ||||
|                 StoragePoolVO pool = _storagePoolDao.findByUuid(updatedDataStoreUUID); | ||||
|                 if (pool != null) { | ||||
|                     vol.setPoolId(pool.getId()); | ||||
|                     vol.setPoolType(pool.getPoolType()); | ||||
|                 } | ||||
|             } | ||||
|             _volsDao.update(volumeId, vol); | ||||
| @ -2317,7 +2318,7 @@ public class VolumeOrchestrator extends ManagerBase implements VolumeOrchestrati | ||||
|     @Override | ||||
|     public DiskProfile importVolume(Type type, String name, DiskOffering offering, Long sizeInBytes, Long minIops, Long maxIops, | ||||
|                                     Long zoneId, HypervisorType hypervisorType, VirtualMachine vm, VirtualMachineTemplate template, Account owner, | ||||
|                                     Long deviceId, Long poolId, String path, String chainInfo) { | ||||
|                                     Long deviceId, Long poolId, Storage.StoragePoolType poolType, String path, String chainInfo) { | ||||
|         if (sizeInBytes == null) { | ||||
|             sizeInBytes = offering.getDiskSize(); | ||||
|         } | ||||
| @ -2358,6 +2359,7 @@ public class VolumeOrchestrator extends ManagerBase implements VolumeOrchestrati | ||||
| 
 | ||||
|         vol.setFormat(getSupportedImageFormatForCluster(hypervisorType)); | ||||
|         vol.setPoolId(poolId); | ||||
|         vol.setPoolType(poolType); | ||||
|         vol.setPath(path); | ||||
|         vol.setChainInfo(chainInfo); | ||||
|         vol.setState(Volume.State.Ready); | ||||
| @ -2367,7 +2369,7 @@ public class VolumeOrchestrator extends ManagerBase implements VolumeOrchestrati | ||||
| 
 | ||||
|     @Override | ||||
|     public DiskProfile updateImportedVolume(Type type, DiskOffering offering, VirtualMachine vm, VirtualMachineTemplate template, | ||||
|                                     Long deviceId, Long poolId, String path, String chainInfo, DiskProfile diskProfile) { | ||||
|                                     Long deviceId, Long poolId, Storage.StoragePoolType poolType, String path, String chainInfo, DiskProfile diskProfile) { | ||||
| 
 | ||||
|         VolumeVO vol = _volsDao.findById(diskProfile.getVolumeId()); | ||||
|         if (vm != null) { | ||||
| @ -2401,6 +2403,7 @@ public class VolumeOrchestrator extends ManagerBase implements VolumeOrchestrati | ||||
| 
 | ||||
|         vol.setFormat(getSupportedImageFormatForCluster(vm.getHypervisorType())); | ||||
|         vol.setPoolId(poolId); | ||||
|         vol.setPoolType(poolType); | ||||
|         vol.setPath(path); | ||||
|         vol.setChainInfo(chainInfo); | ||||
|         vol.setSize(diskProfile.getSize()); | ||||
|  | ||||
| @ -822,7 +822,7 @@ public class NetworkOrchestratorTest extends TestCase { | ||||
|         Mockito.when(network.getId()).thenReturn(networkId); | ||||
|         Mockito.when(dataCenter.getId()).thenReturn(dataCenterId); | ||||
|         Mockito.when(ipAddresses.getIp4Address()).thenReturn(requestedIp); | ||||
|         Mockito.when(testOrchestrator._ipAddressDao.findByIp(requestedIp)).thenReturn(ipAddressVO); | ||||
|         Mockito.when(testOrchestrator._ipAddressDao.findByIpAndSourceNetworkId(networkId, requestedIp)).thenReturn(ipAddressVO); | ||||
|         String ipAddress = testOrchestrator.getSelectedIpForNicImport(network, dataCenter, ipAddresses); | ||||
|         Assert.assertEquals(requestedIp, ipAddress); | ||||
|     } | ||||
|  | ||||
| @ -241,7 +241,7 @@ public class VolumeOrchestratorTest { | ||||
| 
 | ||||
|         volumeOrchestrator.importVolume(volumeType, name, diskOffering, sizeInBytes, null, null, | ||||
|                 zoneId, hypervisorType, null, null, owner, | ||||
|                 deviceId, poolId, path, chainInfo); | ||||
|                 deviceId, poolId, Storage.StoragePoolType.NetworkFilesystem, path, chainInfo); | ||||
| 
 | ||||
|         VolumeVO volume = volumeVOMockedConstructionConstruction.constructed().get(0); | ||||
|         Mockito.verify(volume, Mockito.never()).setInstanceId(Mockito.anyLong()); | ||||
|  | ||||
| @ -33,6 +33,8 @@ public interface HostDetailsDao extends GenericDao<DetailVO, Long> { | ||||
| 
 | ||||
|     List<DetailVO> findByName(String name); | ||||
| 
 | ||||
|     void removeExternalDetails(long hostId); | ||||
| 
 | ||||
|     void replaceExternalDetails(long hostId, Map<String, String> details); | ||||
| 
 | ||||
| } | ||||
|  | ||||
| @ -39,6 +39,7 @@ public class HostDetailsDaoImpl extends GenericDaoBase<DetailVO, Long> implement | ||||
|     protected final SearchBuilder<DetailVO> HostSearch; | ||||
|     protected final SearchBuilder<DetailVO> DetailSearch; | ||||
|     protected final SearchBuilder<DetailVO> DetailNameSearch; | ||||
|     protected final SearchBuilder<DetailVO> ExternalDetailSearch; | ||||
| 
 | ||||
|     public HostDetailsDaoImpl() { | ||||
|         HostSearch = createSearchBuilder(); | ||||
| @ -53,6 +54,11 @@ public class HostDetailsDaoImpl extends GenericDaoBase<DetailVO, Long> implement | ||||
|         DetailNameSearch = createSearchBuilder(); | ||||
|         DetailNameSearch.and("name", DetailNameSearch.entity().getName(), SearchCriteria.Op.EQ); | ||||
|         DetailNameSearch.done(); | ||||
| 
 | ||||
|         ExternalDetailSearch = createSearchBuilder(); | ||||
|         ExternalDetailSearch.and("hostId", ExternalDetailSearch.entity().getHostId(), SearchCriteria.Op.EQ); | ||||
|         ExternalDetailSearch.and("name", ExternalDetailSearch.entity().getName(), SearchCriteria.Op.LIKE); | ||||
|         ExternalDetailSearch.done(); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
| @ -133,6 +139,17 @@ public class HostDetailsDaoImpl extends GenericDaoBase<DetailVO, Long> implement | ||||
|         return listBy(sc); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public void removeExternalDetails(long hostId) { | ||||
|         TransactionLegacy txn = TransactionLegacy.currentTxn(); | ||||
|         txn.start(); | ||||
|         SearchCriteria<DetailVO> sc = ExternalDetailSearch.create(); | ||||
|         sc.setParameters("hostId", hostId); | ||||
|         sc.setParameters("name", VmDetailConstants.EXTERNAL_DETAIL_PREFIX + "%"); | ||||
|         remove(sc); | ||||
|         txn.commit(); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public void replaceExternalDetails(long hostId, Map<String, String> details) { | ||||
|         if (details.isEmpty()) { | ||||
| @ -149,11 +166,7 @@ public class HostDetailsDaoImpl extends GenericDaoBase<DetailVO, Long> implement | ||||
|             } | ||||
|             detailVOs.add(new DetailVO(hostId, name, value)); | ||||
|         } | ||||
|         SearchBuilder<DetailVO> sb = createSearchBuilder(); | ||||
|         sb.and("hostId", sb.entity().getHostId(), SearchCriteria.Op.EQ); | ||||
|         sb.and("name", sb.entity().getName(), SearchCriteria.Op.LIKE); | ||||
|         sb.done(); | ||||
|         SearchCriteria<DetailVO> sc = sb.create(); | ||||
|         SearchCriteria<DetailVO> sc = ExternalDetailSearch.create(); | ||||
|         sc.setParameters("hostId", hostId); | ||||
|         sc.setParameters("name", VmDetailConstants.EXTERNAL_DETAIL_PREFIX + "%"); | ||||
|         remove(sc); | ||||
|  | ||||
| @ -24,6 +24,7 @@ import com.cloud.utils.db.Filter; | ||||
| import com.cloud.utils.db.GenericDao; | ||||
| 
 | ||||
| public interface CounterDao extends GenericDao<CounterVO, Long> { | ||||
|     CounterVO findByNameProviderValue(String name, String value, String provider); | ||||
|     public List<CounterVO> listCounters(Long id, String name, String source, String provider, String keyword, Filter filter); | ||||
| 
 | ||||
| } | ||||
|  | ||||
| @ -32,6 +32,7 @@ import com.cloud.utils.db.SearchCriteria.Op; | ||||
| @Component | ||||
| public class CounterDaoImpl extends GenericDaoBase<CounterVO, Long> implements CounterDao { | ||||
|     final SearchBuilder<CounterVO> AllFieldsSearch; | ||||
|     final SearchBuilder<CounterVO> CounterValueSearch; | ||||
| 
 | ||||
|     protected CounterDaoImpl() { | ||||
|         AllFieldsSearch = createSearchBuilder(); | ||||
| @ -40,6 +41,21 @@ public class CounterDaoImpl extends GenericDaoBase<CounterVO, Long> implements C | ||||
|         AllFieldsSearch.and("source", AllFieldsSearch.entity().getSource(), Op.EQ); | ||||
|         AllFieldsSearch.and("provider", AllFieldsSearch.entity().getProvider(), Op.EQ); | ||||
|         AllFieldsSearch.done(); | ||||
| 
 | ||||
|         CounterValueSearch = createSearchBuilder(); | ||||
|         CounterValueSearch.and("name", CounterValueSearch.entity().getName(), Op.EQ); | ||||
|         CounterValueSearch.and("value", CounterValueSearch.entity().getValue(), Op.EQ); | ||||
|         CounterValueSearch.and("provider", CounterValueSearch.entity().getProvider(), Op.EQ); | ||||
|         CounterValueSearch.done(); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public CounterVO findByNameProviderValue(String name, String value, String provider) { | ||||
|         SearchCriteria<CounterVO> sc = CounterValueSearch.create(); | ||||
|         sc.setParameters("name", name); | ||||
|         sc.setParameters("value", value); | ||||
|         sc.setParameters("provider", provider); | ||||
|         return findOneBy(sc); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|  | ||||
| @ -18,6 +18,7 @@ package com.cloud.network.dao; | ||||
| 
 | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
| import java.util.Set; | ||||
| 
 | ||||
| import com.cloud.network.Network; | ||||
| import com.cloud.network.Network.GuestType; | ||||
| @ -47,6 +48,12 @@ public interface NetworkDao extends GenericDao<NetworkVO, Long>, StateDao<State, | ||||
| 
 | ||||
|     int getOtherPersistentNetworksCount(long id, String broadcastURI, boolean isPersistent); | ||||
| 
 | ||||
|     List<NetworkVO> listByNetworkDomains(Set<String> uniqueNtwkDomains); | ||||
| 
 | ||||
|     List<NetworkVO> listByNetworkDomainsAndAccountIds(Set<String> uniqueNtwkDomains, Set<Long> accountIds); | ||||
| 
 | ||||
|     List<NetworkVO> listByNetworkDomainsAndDomainIds(Set<String> uniqueNtwkDomains, Set<Long> domainIds); | ||||
| 
 | ||||
|     /** | ||||
|      * Retrieves the next available mac address in this network configuration. | ||||
|      * | ||||
|  | ||||
| @ -86,6 +86,7 @@ public class NetworkDaoImpl extends GenericDaoBase<NetworkVO, Long>implements Ne | ||||
| 
 | ||||
|     GenericSearchBuilder<NetworkVO, Long> GarbageCollectedSearch; | ||||
|     SearchBuilder<NetworkVO> PrivateNetworkSearch; | ||||
|     SearchBuilder<NetworkVO> NetworkDomainSearch; | ||||
| 
 | ||||
|     @Inject | ||||
|     ResourceTagDao _tagsDao; | ||||
| @ -198,6 +199,12 @@ public class NetworkDaoImpl extends GenericDaoBase<NetworkVO, Long>implements Ne | ||||
|         PersistentNetworkSearch.join("persistent", persistentNtwkOffJoin, PersistentNetworkSearch.entity().getNetworkOfferingId(), persistentNtwkOffJoin.entity().getId(), JoinType.INNER); | ||||
|         PersistentNetworkSearch.done(); | ||||
| 
 | ||||
|         NetworkDomainSearch = createSearchBuilder(); | ||||
|         NetworkDomainSearch.and("networkDomains", NetworkDomainSearch.entity().getNetworkDomain(), Op.IN); | ||||
|         NetworkDomainSearch.and("accounts", NetworkDomainSearch.entity().getAccountId(), Op.IN); | ||||
|         NetworkDomainSearch.and("domains", NetworkDomainSearch.entity().getDomainId(), Op.IN); | ||||
|         NetworkDomainSearch.done(); | ||||
| 
 | ||||
|         PhysicalNetworkSearch = createSearchBuilder(); | ||||
|         PhysicalNetworkSearch.and("physicalNetworkId", PhysicalNetworkSearch.entity().getPhysicalNetworkId(), Op.EQ); | ||||
|         PhysicalNetworkSearch.done(); | ||||
| @ -428,6 +435,29 @@ public class NetworkDaoImpl extends GenericDaoBase<NetworkVO, Long>implements Ne | ||||
|         return search(sc, null); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public List<NetworkVO> listByNetworkDomains(Set<String> uniqueNtwkDomains) { | ||||
|         SearchCriteria<NetworkVO> sc = NetworkDomainSearch.create(); | ||||
|         sc.setParameters("networkDomains", uniqueNtwkDomains.toArray()); | ||||
|         return search(sc, null); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public List<NetworkVO> listByNetworkDomainsAndAccountIds(Set<String> uniqueNtwkDomains, Set<Long> accountIds) { | ||||
|         SearchCriteria<NetworkVO> sc = NetworkDomainSearch.create(); | ||||
|         sc.setParameters("networkDomains", uniqueNtwkDomains.toArray()); | ||||
|         sc.setParameters("accounts", accountIds.toArray()); | ||||
|         return search(sc, null); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public List<NetworkVO> listByNetworkDomainsAndDomainIds(Set<String> uniqueNtwkDomains, Set<Long> domainIds) { | ||||
|         SearchCriteria<NetworkVO> sc = NetworkDomainSearch.create(); | ||||
|         sc.setParameters("networkDomains", uniqueNtwkDomains.toArray()); | ||||
|         sc.setParameters("domains", domainIds.toArray()); | ||||
|         return search(sc, null); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public String getNextAvailableMacAddress(final long networkConfigId, Integer zoneMacIdentifier) { | ||||
|         final SequenceFetcher fetch = SequenceFetcher.getInstance(); | ||||
|  | ||||
| @ -577,11 +577,11 @@ public class DiskOfferingVO implements DiskOffering { | ||||
|     @Override | ||||
|     public void setEncrypt(boolean encrypt) { this.encrypt = encrypt; } | ||||
| 
 | ||||
|     @Override | ||||
|     public boolean isShared() { | ||||
|         return !useLocalStorage; | ||||
|     } | ||||
| 
 | ||||
| 
 | ||||
|     public boolean getDiskSizeStrictness() { | ||||
|         return diskSizeStrictness; | ||||
|     } | ||||
|  | ||||
| @ -59,6 +59,12 @@ public class SnapshotPolicyVO implements SnapshotPolicy { | ||||
|     @Column(name = "uuid") | ||||
|     String uuid; | ||||
| 
 | ||||
|     @Column(name = "account_id") | ||||
|     long accountId; | ||||
| 
 | ||||
|     @Column(name = "domain_id") | ||||
|     long domainId; | ||||
| 
 | ||||
|     @Column(name = "display", updatable = true, nullable = false) | ||||
|     protected boolean display = true; | ||||
| 
 | ||||
| @ -66,7 +72,7 @@ public class SnapshotPolicyVO implements SnapshotPolicy { | ||||
|         this.uuid = UUID.randomUUID().toString(); | ||||
|     } | ||||
| 
 | ||||
|     public SnapshotPolicyVO(long volumeId, String schedule, String timezone, IntervalType intvType, int maxSnaps, boolean display) { | ||||
|     public SnapshotPolicyVO(long volumeId, String schedule, String timezone, IntervalType intvType, int maxSnaps, long accountId, long domainId, boolean display) { | ||||
|         this.volumeId = volumeId; | ||||
|         this.schedule = schedule; | ||||
|         this.timezone = timezone; | ||||
| @ -75,6 +81,8 @@ public class SnapshotPolicyVO implements SnapshotPolicy { | ||||
|         this.active = true; | ||||
|         this.display = display; | ||||
|         this.uuid = UUID.randomUUID().toString(); | ||||
|         this.accountId = accountId; | ||||
|         this.domainId = domainId; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
| @ -160,4 +168,32 @@ public class SnapshotPolicyVO implements SnapshotPolicy { | ||||
|     public void setDisplay(boolean display) { | ||||
|         this.display = display; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public long getAccountId() { | ||||
|         return accountId; | ||||
|     } | ||||
| 
 | ||||
|     public void setAccountId(long accountId) { | ||||
|         this.accountId = accountId; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public long getDomainId() { | ||||
|         return domainId; | ||||
|     } | ||||
| 
 | ||||
|     public void setDomainId(long domainId) { | ||||
|         this.domainId = domainId; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public Class<?> getEntityType() { | ||||
|         return SnapshotPolicy.class; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public String getName() { | ||||
|         return null; | ||||
|     } | ||||
| } | ||||
|  | ||||
| @ -28,6 +28,7 @@ import javax.persistence.Temporal; | ||||
| import javax.persistence.TemporalType; | ||||
| 
 | ||||
| import com.cloud.utils.db.GenericDaoBase; | ||||
| import org.apache.cloudstack.utils.reflectiontostringbuilderutils.ReflectionToStringBuilderUtils; | ||||
| 
 | ||||
| /** | ||||
|  * Join table for storage pools and hosts | ||||
| @ -100,4 +101,9 @@ public class StoragePoolHostVO implements StoragePoolHostAssoc { | ||||
|         this.localPath = localPath; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public String toString() { | ||||
|         return ReflectionToStringBuilderUtils.reflectOnlySelectedFields(this, "hostId", "poolId"); | ||||
|     } | ||||
| 
 | ||||
| } | ||||
|  | ||||
| @ -31,6 +31,8 @@ public interface DiskOfferingDao extends GenericDao<DiskOfferingVO, Long> { | ||||
|     List<DiskOfferingVO> listAllBySizeAndProvisioningType(long size, Storage.ProvisioningType provisioningType); | ||||
| 
 | ||||
|     List<DiskOfferingVO> findCustomDiskOfferings(); | ||||
| 
 | ||||
|     List<DiskOfferingVO> listByStorageTag(String tag); | ||||
| 
 | ||||
|     List<DiskOfferingVO> listAllActiveAndNonComputeDiskOfferings(); | ||||
| } | ||||
|  | ||||
| @ -26,6 +26,7 @@ import java.util.List; | ||||
| import javax.inject.Inject; | ||||
| import javax.persistence.EntityExistsException; | ||||
| 
 | ||||
| import com.cloud.offering.DiskOffering; | ||||
| import org.apache.cloudstack.resourcedetail.dao.DiskOfferingDetailsDao; | ||||
| import org.springframework.stereotype.Component; | ||||
| 
 | ||||
| @ -45,6 +46,8 @@ public class DiskOfferingDaoImpl extends GenericDaoBase<DiskOfferingVO, Long> im | ||||
|     protected DiskOfferingDetailsDao detailsDao; | ||||
| 
 | ||||
|     protected final SearchBuilder<DiskOfferingVO> UniqueNameSearch; | ||||
|     protected final SearchBuilder<DiskOfferingVO> ActiveAndNonComputeSearch; | ||||
| 
 | ||||
|     private final String SizeDiskOfferingSearch = "SELECT * FROM disk_offering WHERE " + | ||||
|             "disk_size = ? AND provisioning_type = ? AND removed IS NULL"; | ||||
| 
 | ||||
| @ -56,6 +59,11 @@ public class DiskOfferingDaoImpl extends GenericDaoBase<DiskOfferingVO, Long> im | ||||
|         UniqueNameSearch.and("name", UniqueNameSearch.entity().getUniqueName(), SearchCriteria.Op.EQ); | ||||
|         UniqueNameSearch.done(); | ||||
| 
 | ||||
|         ActiveAndNonComputeSearch = createSearchBuilder(); | ||||
|         ActiveAndNonComputeSearch.and("state", ActiveAndNonComputeSearch.entity().getState(), SearchCriteria.Op.EQ); | ||||
|         ActiveAndNonComputeSearch.and("computeOnly", ActiveAndNonComputeSearch.entity().isComputeOnly(), SearchCriteria.Op.EQ); | ||||
|         ActiveAndNonComputeSearch.done(); | ||||
| 
 | ||||
|         _computeOnlyAttr = _allAttributes.get("computeOnly"); | ||||
|     } | ||||
| 
 | ||||
| @ -164,4 +172,12 @@ public class DiskOfferingDaoImpl extends GenericDaoBase<DiskOfferingVO, Long> im | ||||
|         sc.setParameters("tagEndLike",   "%," + tag); | ||||
|         return listBy(sc); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public List<DiskOfferingVO> listAllActiveAndNonComputeDiskOfferings() { | ||||
|         SearchCriteria<DiskOfferingVO> sc = ActiveAndNonComputeSearch.create(); | ||||
|         sc.setParameters("state", DiskOffering.State.Active); | ||||
|         sc.setParameters("computeOnly", false); | ||||
|         return listBy(sc); | ||||
|     } | ||||
| } | ||||
|  | ||||
| @ -35,6 +35,8 @@ public interface VMTemplatePoolDao extends GenericDao<VMTemplateStoragePoolVO, L | ||||
| 
 | ||||
|     List<VMTemplateStoragePoolVO> listByPoolIdAndState(long poolId, ObjectInDataStoreStateMachine.State state); | ||||
| 
 | ||||
|     List<VMTemplateStoragePoolVO> listByPoolIdsAndTemplate(List<Long> poolIds, Long templateId); | ||||
| 
 | ||||
|     List<VMTemplateStoragePoolVO> listByTemplateStatus(long templateId, VMTemplateStoragePoolVO.Status downloadState); | ||||
| 
 | ||||
|     List<VMTemplateStoragePoolVO> listByTemplateStatus(long templateId, VMTemplateStoragePoolVO.Status downloadState, long poolId); | ||||
|  | ||||
| @ -150,6 +150,16 @@ public class VMTemplatePoolDaoImpl extends GenericDaoBase<VMTemplateStoragePoolV | ||||
|         return findOneIncludingRemovedBy(sc); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public List<VMTemplateStoragePoolVO> listByPoolIdsAndTemplate(List<Long> poolIds, Long templateId) { | ||||
|         SearchCriteria<VMTemplateStoragePoolVO> sc = PoolTemplateSearch.create(); | ||||
|         if (CollectionUtils.isNotEmpty(poolIds)) { | ||||
|             sc.setParameters("pool_id", poolIds.toArray()); | ||||
|         } | ||||
|         sc.setParameters("template_id", templateId); | ||||
|         return listBy(sc); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public List<VMTemplateStoragePoolVO> listByTemplateStatus(long templateId, VMTemplateStoragePoolVO.Status downloadState) { | ||||
|         SearchCriteria<VMTemplateStoragePoolVO> sc = TemplateStatusSearch.create(); | ||||
|  | ||||
| @ -822,6 +822,7 @@ public class VolumeDaoImpl extends GenericDaoBase<VolumeVO, Long> implements Vol | ||||
|         if (volume.getState() != Volume.State.Destroy) { | ||||
|             volume.setState(Volume.State.Destroy); | ||||
|             volume.setPoolId(null); | ||||
|             volume.setPoolType(null); | ||||
|             volume.setInstanceId(null); | ||||
|             update(volume.getId(), volume); | ||||
|             remove(volume.getId()); | ||||
|  | ||||
| @ -16,6 +16,14 @@ | ||||
| // under the License. | ||||
| package com.cloud.upgrade.dao; | ||||
| 
 | ||||
| import com.cloud.utils.exception.CloudRuntimeException; | ||||
| 
 | ||||
| import java.io.InputStream; | ||||
| import java.sql.Connection; | ||||
| import java.sql.PreparedStatement; | ||||
| import java.sql.ResultSet; | ||||
| import java.sql.SQLException; | ||||
| 
 | ||||
| public class Upgrade42100to42200 extends DbUpgradeAbstractImpl implements DbUpgrade, DbUpgradeSystemVmTemplate { | ||||
| 
 | ||||
|     @Override | ||||
| @ -27,4 +35,69 @@ public class Upgrade42100to42200 extends DbUpgradeAbstractImpl implements DbUpgr | ||||
|     public String getUpgradedVersion() { | ||||
|         return "4.22.0.0"; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public InputStream[] getPrepareScripts() { | ||||
|         final String scriptFile = "META-INF/db/schema-42100to42200.sql"; | ||||
|         final InputStream script = Thread.currentThread().getContextClassLoader().getResourceAsStream(scriptFile); | ||||
|         if (script == null) { | ||||
|             throw new CloudRuntimeException("Unable to find " + scriptFile); | ||||
|         } | ||||
| 
 | ||||
|         return new InputStream[] {script}; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public void performDataMigration(Connection conn) { | ||||
|         updateSnapshotPolicyOwnership(conn); | ||||
|         updateBackupScheduleOwnership(conn); | ||||
|     } | ||||
| 
 | ||||
|     protected void updateSnapshotPolicyOwnership(Connection conn) { | ||||
|         // set account_id and domain_id in snapshot_policy table from volume table | ||||
|         String selectSql = "SELECT sp.id, v.account_id, v.domain_id FROM snapshot_policy sp, volumes v WHERE sp.volume_id = v.id AND (sp.account_id IS NULL AND sp.domain_id IS NULL)"; | ||||
|         String updateSql = "UPDATE snapshot_policy SET account_id = ?, domain_id = ? WHERE id = ?"; | ||||
| 
 | ||||
|         try (PreparedStatement selectPstmt = conn.prepareStatement(selectSql); | ||||
|              ResultSet rs = selectPstmt.executeQuery(); | ||||
|              PreparedStatement updatePstmt = conn.prepareStatement(updateSql)) { | ||||
| 
 | ||||
|             while (rs.next()) { | ||||
|                 long policyId = rs.getLong(1); | ||||
|                 long accountId = rs.getLong(2); | ||||
|                 long domainId = rs.getLong(3); | ||||
| 
 | ||||
|                 updatePstmt.setLong(1, accountId); | ||||
|                 updatePstmt.setLong(2, domainId); | ||||
|                 updatePstmt.setLong(3, policyId); | ||||
|                 updatePstmt.executeUpdate(); | ||||
|             } | ||||
|         } catch (SQLException e) { | ||||
|             throw new CloudRuntimeException("Unable to update snapshot_policy table with account_id and domain_id", e); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     protected void updateBackupScheduleOwnership(Connection conn) { | ||||
|         // Set account_id and domain_id in backup_schedule table from vm_instance table | ||||
|         String selectSql = "SELECT bs.id, vm.account_id, vm.domain_id FROM backup_schedule bs, vm_instance vm WHERE bs.vm_id = vm.id AND (bs.account_id IS NULL AND bs.domain_id IS NULL)"; | ||||
|         String updateSql = "UPDATE backup_schedule SET account_id = ?, domain_id = ? WHERE id = ?"; | ||||
| 
 | ||||
|         try (PreparedStatement selectPstmt = conn.prepareStatement(selectSql); | ||||
|              ResultSet rs = selectPstmt.executeQuery(); | ||||
|              PreparedStatement updatePstmt = conn.prepareStatement(updateSql)) { | ||||
| 
 | ||||
|             while (rs.next()) { | ||||
|                 long scheduleId = rs.getLong(1); | ||||
|                 long accountId = rs.getLong(2); | ||||
|                 long domainId = rs.getLong(3); | ||||
| 
 | ||||
|                 updatePstmt.setLong(1, accountId); | ||||
|                 updatePstmt.setLong(2, domainId); | ||||
|                 updatePstmt.setLong(3, scheduleId); | ||||
|                 updatePstmt.executeUpdate(); | ||||
|             } | ||||
|         } catch (SQLException e) { | ||||
|             throw new CloudRuntimeException("Unable to update backup_schedule table with account_id and domain_id", e); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
							
								
								
									
										270
									
								
								engine/schema/src/main/java/com/cloud/vm/ImportVMTaskVO.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										270
									
								
								engine/schema/src/main/java/com/cloud/vm/ImportVMTaskVO.java
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,270 @@ | ||||
| // | ||||
| // Licensed to the Apache Software Foundation (ASF) under one | ||||
| // or more contributor license agreements.  See the NOTICE file | ||||
| // distributed with this work for additional information | ||||
| // regarding copyright ownership.  The ASF licenses this file | ||||
| // to you 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 | ||||
| // | ||||
| // 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. | ||||
| // | ||||
| package com.cloud.vm; | ||||
| 
 | ||||
| import org.apache.cloudstack.vm.ImportVmTask; | ||||
| 
 | ||||
| import javax.persistence.Column; | ||||
| import javax.persistence.Entity; | ||||
| import javax.persistence.GeneratedValue; | ||||
| import javax.persistence.GenerationType; | ||||
| import javax.persistence.Id; | ||||
| import javax.persistence.Table; | ||||
| import javax.persistence.Temporal; | ||||
| import javax.persistence.TemporalType; | ||||
| import java.util.Date; | ||||
| import java.util.UUID; | ||||
| 
 | ||||
| @Entity | ||||
| @Table(name = "import_vm_task") | ||||
| public class ImportVMTaskVO implements ImportVmTask { | ||||
| 
 | ||||
|     public ImportVMTaskVO(long zoneId, long accountId, long userId, String displayName, | ||||
|                           String vcenter, String datacenter, String sourceVMName, long convertHostId, long importHostId) { | ||||
|         this.zoneId = zoneId; | ||||
|         this.accountId = accountId; | ||||
|         this.userId = userId; | ||||
|         this.displayName = displayName; | ||||
|         this.vcenter = vcenter; | ||||
|         this.datacenter = datacenter; | ||||
|         this.sourceVMName = sourceVMName; | ||||
|         this.step = Step.Prepare; | ||||
|         this.uuid = UUID.randomUUID().toString(); | ||||
|         this.convertHostId = convertHostId; | ||||
|         this.importHostId = importHostId; | ||||
|     } | ||||
| 
 | ||||
|     public ImportVMTaskVO() { | ||||
|     } | ||||
| 
 | ||||
|     @Id | ||||
|     @GeneratedValue(strategy = GenerationType.IDENTITY) | ||||
|     @Column(name = "id") | ||||
|     private long id; | ||||
| 
 | ||||
|     @Column(name = "uuid") | ||||
|     private String uuid; | ||||
| 
 | ||||
|     @Column(name = "zone_id") | ||||
|     private long zoneId; | ||||
| 
 | ||||
|     @Column(name = "account_id") | ||||
|     private long accountId; | ||||
| 
 | ||||
|     @Column(name = "user_id") | ||||
|     private long userId; | ||||
| 
 | ||||
|     @Column(name = "vm_id") | ||||
|     private Long vmId; | ||||
|     @Column(name = "display_name") | ||||
|     private String displayName; | ||||
| 
 | ||||
|     @Column(name = "vcenter") | ||||
|     private String vcenter; | ||||
| 
 | ||||
|     @Column(name = "datacenter") | ||||
|     private String datacenter; | ||||
| 
 | ||||
|     @Column(name = "source_vm_name") | ||||
|     private String sourceVMName; | ||||
| 
 | ||||
|     @Column(name = "convert_host_id") | ||||
|     private long convertHostId; | ||||
| 
 | ||||
|     @Column(name = "import_host_id") | ||||
|     private long importHostId; | ||||
| 
 | ||||
|     @Column(name = "step") | ||||
|     private Step step; | ||||
| 
 | ||||
|     @Column(name = "state") | ||||
|     private TaskState state; | ||||
| 
 | ||||
|     @Column(name = "description") | ||||
|     private String description; | ||||
| 
 | ||||
|     @Column(name = "duration") | ||||
|     private Long duration; | ||||
| 
 | ||||
|     @Column(name = "created") | ||||
|     @Temporal(value = TemporalType.TIMESTAMP) | ||||
|     private Date created; | ||||
| 
 | ||||
|     @Column(name = "updated") | ||||
|     @Temporal(value = TemporalType.TIMESTAMP) | ||||
|     private Date updated; | ||||
| 
 | ||||
|     @Column(name = "removed") | ||||
|     @Temporal(value = TemporalType.TIMESTAMP) | ||||
|     private Date removed; | ||||
| 
 | ||||
|     @Override | ||||
|     public long getId() { | ||||
|         return id; | ||||
|     } | ||||
| 
 | ||||
|     public void setId(long id) { | ||||
|         this.id = id; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public String getUuid() { | ||||
|         return uuid; | ||||
|     } | ||||
| 
 | ||||
|     public void setUuid(String uuid) { | ||||
|         this.uuid = uuid; | ||||
|     } | ||||
| 
 | ||||
|     public long getZoneId() { | ||||
|         return zoneId; | ||||
|     } | ||||
| 
 | ||||
|     public void setZoneId(long zoneId) { | ||||
|         this.zoneId = zoneId; | ||||
|     } | ||||
| 
 | ||||
|     public long getAccountId() { | ||||
|         return accountId; | ||||
|     } | ||||
| 
 | ||||
|     public void setAccountId(long accountId) { | ||||
|         this.accountId = accountId; | ||||
|     } | ||||
| 
 | ||||
|     public long getUserId() { | ||||
|         return userId; | ||||
|     } | ||||
| 
 | ||||
|     public void setUserId(long userId) { | ||||
|         this.userId = userId; | ||||
|     } | ||||
| 
 | ||||
|     public Long getVmId() { | ||||
|         return vmId; | ||||
|     } | ||||
| 
 | ||||
|     public void setVmId(Long vmId) { | ||||
|         this.vmId = vmId; | ||||
|     } | ||||
| 
 | ||||
|     public String getDisplayName() { | ||||
|         return displayName; | ||||
|     } | ||||
| 
 | ||||
|     public void setDisplayName(String displayName) { | ||||
|         this.displayName = displayName; | ||||
|     } | ||||
| 
 | ||||
|     public String getVcenter() { | ||||
|         return vcenter; | ||||
|     } | ||||
| 
 | ||||
|     public void setVcenter(String vcenter) { | ||||
|         this.vcenter = vcenter; | ||||
|     } | ||||
| 
 | ||||
|     public String getDatacenter() { | ||||
|         return datacenter; | ||||
|     } | ||||
| 
 | ||||
|     public void setDatacenter(String datacenter) { | ||||
|         this.datacenter = datacenter; | ||||
|     } | ||||
| 
 | ||||
|     public String getSourceVMName() { | ||||
|         return sourceVMName; | ||||
|     } | ||||
| 
 | ||||
|     public void setSourceVMName(String sourceVMName) { | ||||
|         this.sourceVMName = sourceVMName; | ||||
|     } | ||||
| 
 | ||||
|     public long getConvertHostId() { | ||||
|         return convertHostId; | ||||
|     } | ||||
| 
 | ||||
|     public void setConvertHostId(long convertHostId) { | ||||
|         this.convertHostId = convertHostId; | ||||
|     } | ||||
| 
 | ||||
|     public long getImportHostId() { | ||||
|         return importHostId; | ||||
|     } | ||||
| 
 | ||||
|     public void setImportHostId(long importHostId) { | ||||
|         this.importHostId = importHostId; | ||||
|     } | ||||
| 
 | ||||
|     public Step getStep() { | ||||
|         return step; | ||||
|     } | ||||
| 
 | ||||
|     public void setStep(Step step) { | ||||
|         this.step = step; | ||||
|     } | ||||
| 
 | ||||
|     public TaskState getState() { | ||||
|         return state; | ||||
|     } | ||||
| 
 | ||||
|     public void setState(TaskState state) { | ||||
|         this.state = state; | ||||
|     } | ||||
| 
 | ||||
|     public String getDescription() { | ||||
|         return description; | ||||
|     } | ||||
| 
 | ||||
|     public void setDescription(String description) { | ||||
|         this.description = description; | ||||
|     } | ||||
| 
 | ||||
|     public Long getDuration() { | ||||
|         return duration; | ||||
|     } | ||||
| 
 | ||||
|     public void setDuration(Long duration) { | ||||
|         this.duration = duration; | ||||
|     } | ||||
| 
 | ||||
|     public Date getCreated() { | ||||
|         return created; | ||||
|     } | ||||
| 
 | ||||
|     public void setCreated(Date created) { | ||||
|         this.created = created; | ||||
|     } | ||||
| 
 | ||||
|     public Date getUpdated() { | ||||
|         return updated; | ||||
|     } | ||||
| 
 | ||||
|     public void setUpdated(Date updated) { | ||||
|         this.updated = updated; | ||||
|     } | ||||
| 
 | ||||
|     public Date getRemoved() { | ||||
|         return removed; | ||||
|     } | ||||
| 
 | ||||
|     public void setRemoved(Date removed) { | ||||
|         this.removed = removed; | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,31 @@ | ||||
| // | ||||
| // Licensed to the Apache Software Foundation (ASF) under one | ||||
| // or more contributor license agreements.  See the NOTICE file | ||||
| // distributed with this work for additional information | ||||
| // regarding copyright ownership.  The ASF licenses this file | ||||
| // to you 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 | ||||
| // | ||||
| // 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. | ||||
| // | ||||
| package com.cloud.vm.dao; | ||||
| 
 | ||||
| import com.cloud.utils.Pair; | ||||
| import com.cloud.utils.db.GenericDao; | ||||
| import com.cloud.vm.ImportVMTaskVO; | ||||
| import org.apache.cloudstack.vm.ImportVmTask; | ||||
| import java.util.List; | ||||
| 
 | ||||
| public interface ImportVMTaskDao extends GenericDao<ImportVMTaskVO, Long> { | ||||
| 
 | ||||
|     Pair<List<ImportVMTaskVO>, Integer> listImportVMTasks(Long zoneId, Long accountId, String vcenter, Long convertHostId, | ||||
|                                                           ImportVmTask.TaskState state, Long startIndex, Long pageSizeVal); | ||||
| } | ||||
| @ -0,0 +1,74 @@ | ||||
| // Licensed to the Apache Software Foundation (ASF) under one | ||||
| // or more contributor license agreements.  See the NOTICE file | ||||
| // distributed with this work for additional information | ||||
| // regarding copyright ownership.  The ASF licenses this file | ||||
| // to you 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 | ||||
| // | ||||
| // 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. | ||||
| package com.cloud.vm.dao; | ||||
| 
 | ||||
| import com.cloud.utils.Pair; | ||||
| import com.cloud.utils.db.Filter; | ||||
| import com.cloud.utils.db.GenericDaoBase; | ||||
| import com.cloud.utils.db.SearchBuilder; | ||||
| import com.cloud.utils.db.SearchCriteria; | ||||
| import com.cloud.vm.ImportVMTaskVO; | ||||
| import org.apache.cloudstack.vm.ImportVmTask; | ||||
| import org.apache.commons.lang3.StringUtils; | ||||
| import org.springframework.stereotype.Component; | ||||
| 
 | ||||
| import javax.annotation.PostConstruct; | ||||
| import java.util.List; | ||||
| 
 | ||||
| @Component | ||||
| public class ImportVMTaskDaoImpl extends GenericDaoBase<ImportVMTaskVO, Long> implements ImportVMTaskDao { | ||||
| 
 | ||||
|     private SearchBuilder<ImportVMTaskVO> AllFieldsSearch; | ||||
| 
 | ||||
|     public ImportVMTaskDaoImpl() { | ||||
|     } | ||||
| 
 | ||||
|     @PostConstruct | ||||
|     void init() { | ||||
|         AllFieldsSearch = createSearchBuilder(); | ||||
|         AllFieldsSearch.and("zoneId", AllFieldsSearch.entity().getZoneId(), SearchCriteria.Op.EQ); | ||||
|         AllFieldsSearch.and("accountId", AllFieldsSearch.entity().getAccountId(), SearchCriteria.Op.EQ); | ||||
|         AllFieldsSearch.and("vcenter", AllFieldsSearch.entity().getVcenter(), SearchCriteria.Op.EQ); | ||||
|         AllFieldsSearch.and("convertHostId", AllFieldsSearch.entity().getConvertHostId(), SearchCriteria.Op.EQ); | ||||
|         AllFieldsSearch.and("state", AllFieldsSearch.entity().getState(), SearchCriteria.Op.EQ); | ||||
|         AllFieldsSearch.done(); | ||||
|     } | ||||
| 
 | ||||
| 
 | ||||
|     @Override | ||||
|     public Pair<List<ImportVMTaskVO>, Integer> listImportVMTasks(Long zoneId, Long accountId, String vcenter, Long convertHostId, | ||||
|                                                                  ImportVmTask.TaskState state, Long startIndex, Long pageSizeVal) { | ||||
|         SearchCriteria<ImportVMTaskVO> sc = AllFieldsSearch.create(); | ||||
|         if (zoneId != null) { | ||||
|             sc.setParameters("zoneId", zoneId); | ||||
|         } | ||||
|         if (accountId != null) { | ||||
|             sc.setParameters("accountId", accountId); | ||||
|         } | ||||
|         if (StringUtils.isNotBlank(vcenter)) { | ||||
|             sc.setParameters("vcenter", vcenter); | ||||
|         } | ||||
|         if (convertHostId != null) { | ||||
|             sc.setParameters("convertHostId", convertHostId); | ||||
|         } | ||||
|         if (state != null) { | ||||
|             sc.setParameters("state", state); | ||||
|         } | ||||
|         Filter filter = new Filter(ImportVMTaskVO.class, "created", false, startIndex, pageSizeVal); | ||||
|         return searchAndCount(sc, filter); | ||||
|     } | ||||
| } | ||||
| @ -68,10 +68,16 @@ public class BackupScheduleVO implements BackupSchedule { | ||||
|     @Column(name = "quiescevm") | ||||
|     Boolean quiesceVM = false; | ||||
| 
 | ||||
|     @Column(name = "account_id") | ||||
|     Long accountId; | ||||
| 
 | ||||
|     @Column(name = "domain_id") | ||||
|     Long domainId; | ||||
| 
 | ||||
|     public BackupScheduleVO() { | ||||
|     } | ||||
| 
 | ||||
|     public BackupScheduleVO(Long vmId, DateUtil.IntervalType scheduleType, String schedule, String timezone, Date scheduledTimestamp, int maxBackups, Boolean quiesceVM) { | ||||
|     public BackupScheduleVO(Long vmId, DateUtil.IntervalType scheduleType, String schedule, String timezone, Date scheduledTimestamp, int maxBackups, Boolean quiesceVM, Long accountId, Long domainId) { | ||||
|         this.vmId = vmId; | ||||
|         this.scheduleType = (short) scheduleType.ordinal(); | ||||
|         this.schedule = schedule; | ||||
| @ -79,6 +85,8 @@ public class BackupScheduleVO implements BackupSchedule { | ||||
|         this.scheduledTimestamp = scheduledTimestamp; | ||||
|         this.maxBackups = maxBackups; | ||||
|         this.quiesceVM = quiesceVM; | ||||
|         this.accountId = accountId; | ||||
|         this.domainId = domainId; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
| @ -161,4 +169,32 @@ public class BackupScheduleVO implements BackupSchedule { | ||||
|     public Boolean getQuiesceVM() { | ||||
|         return quiesceVM; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public Class<?> getEntityType() { | ||||
|         return BackupSchedule.class; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public String getName() { | ||||
|         return null; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public long getDomainId() { | ||||
|         return domainId; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public long getAccountId() { | ||||
|         return accountId; | ||||
|     } | ||||
| 
 | ||||
|     public void setAccountId(Long accountId) { | ||||
|         this.accountId = accountId; | ||||
|     } | ||||
| 
 | ||||
|     public void setDomainId(Long domainId) { | ||||
|         this.domainId = domainId; | ||||
|     } | ||||
| } | ||||
|  | ||||
| @ -24,7 +24,7 @@ import org.apache.cloudstack.backup.BackupOfferingVO; | ||||
| import com.cloud.utils.db.GenericDao; | ||||
| 
 | ||||
| public interface BackupOfferingDao extends GenericDao<BackupOfferingVO, Long> { | ||||
|     BackupOfferingResponse newBackupOfferingResponse(BackupOffering policy); | ||||
|     BackupOfferingResponse newBackupOfferingResponse(BackupOffering policy, Boolean crossZoneInstanceCreation); | ||||
|     BackupOffering findByExternalId(String externalId, Long zoneId); | ||||
|     BackupOffering findByName(String name, Long zoneId); | ||||
| } | ||||
|  | ||||
| @ -50,7 +50,7 @@ public class BackupOfferingDaoImpl extends GenericDaoBase<BackupOfferingVO, Long | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public BackupOfferingResponse newBackupOfferingResponse(BackupOffering offering) { | ||||
|     public BackupOfferingResponse newBackupOfferingResponse(BackupOffering offering, Boolean crossZoneInstanceCreation) { | ||||
|         DataCenterVO zone = dataCenterDao.findById(offering.getZoneId()); | ||||
| 
 | ||||
|         BackupOfferingResponse response = new BackupOfferingResponse(); | ||||
| @ -64,6 +64,9 @@ public class BackupOfferingDaoImpl extends GenericDaoBase<BackupOfferingVO, Long | ||||
|             response.setZoneId(zone.getUuid()); | ||||
|             response.setZoneName(zone.getName()); | ||||
|         } | ||||
|         if (crossZoneInstanceCreation) { | ||||
|             response.setCrossZoneInstanceCreation(true); | ||||
|         } | ||||
|         response.setCreated(offering.getCreated()); | ||||
|         response.setObjectName("backupoffering"); | ||||
|         return response; | ||||
|  | ||||
| @ -171,4 +171,5 @@ public interface PrimaryDataStoreDao extends GenericDao<StoragePoolVO, Long> { | ||||
| 
 | ||||
|     List<StoragePoolVO> findPoolsByStorageTypeAndZone(Storage.StoragePoolType storageType, Long zoneId); | ||||
| 
 | ||||
|     List<StoragePoolVO> listByDataCenterIds(List<Long> dataCenterIds); | ||||
| } | ||||
|  | ||||
| @ -65,6 +65,7 @@ public class PrimaryDataStoreDaoImpl extends GenericDaoBase<StoragePoolVO, Long> | ||||
|     private final GenericSearchBuilder<StoragePoolVO, Long> StatusCountSearch; | ||||
|     private final SearchBuilder<StoragePoolVO> ClustersSearch; | ||||
|     private final SearchBuilder<StoragePoolVO> IdsSearch; | ||||
|     private final SearchBuilder<StoragePoolVO> DcsSearch; | ||||
| 
 | ||||
|     @Inject | ||||
|     private StoragePoolDetailsDao _detailsDao; | ||||
| @ -167,6 +168,9 @@ public class PrimaryDataStoreDaoImpl extends GenericDaoBase<StoragePoolVO, Long> | ||||
|         IdsSearch.and("ids", IdsSearch.entity().getId(), SearchCriteria.Op.IN); | ||||
|         IdsSearch.done(); | ||||
| 
 | ||||
|         DcsSearch = createSearchBuilder(); | ||||
|         DcsSearch.and("dataCenterId", DcsSearch.entity().getDataCenterId(), SearchCriteria.Op.IN); | ||||
|         DcsSearch.done(); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
| @ -927,6 +931,16 @@ public class PrimaryDataStoreDaoImpl extends GenericDaoBase<StoragePoolVO, Long> | ||||
|         return listBy(sc); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public List<StoragePoolVO> listByDataCenterIds(List<Long> dataCenterIds) { | ||||
|         if (CollectionUtils.isEmpty(dataCenterIds)) { | ||||
|             return Collections.emptyList(); | ||||
|         } | ||||
|         SearchCriteria<StoragePoolVO> sc = DcsSearch.create(); | ||||
|         sc.setParameters("dataCenterId", dataCenterIds.toArray()); | ||||
|         return listBy(sc); | ||||
|     } | ||||
| 
 | ||||
|     private SearchCriteria<StoragePoolVO> createStoragePoolSearchCriteria(Long storagePoolId, String storagePoolName, | ||||
|                                                                           Long zoneId, String path, Long podId, Long clusterId, Long hostId, String address, ScopeType scopeType, | ||||
|                                                                           StoragePoolStatus status, String keyword, String storageAccessGroup) { | ||||
|  | ||||
| @ -309,4 +309,5 @@ | ||||
|   <bean id="gpuCardDaoImpl" class="com.cloud.gpu.dao.GpuCardDaoImpl" /> | ||||
|   <bean id="gpuDeviceDaoImpl" class="com.cloud.gpu.dao.GpuDeviceDaoImpl" /> | ||||
|   <bean id="vgpuProfileDaoImpl" class="com.cloud.gpu.dao.VgpuProfileDaoImpl" /> | ||||
|   <bean id="importVMTaskDaoImpl" class="com.cloud.vm.dao.ImportVMTaskDaoImpl" /> | ||||
| </beans> | ||||
|  | ||||
| @ -0,0 +1,26 @@ | ||||
| -- Licensed to the Apache Software Foundation (ASF) under one | ||||
| -- or more contributor license agreements.  See the NOTICE file | ||||
| -- distributed with this work for additional information | ||||
| -- regarding copyright ownership.  The ASF licenses this file | ||||
| -- to you 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 | ||||
| -- | ||||
| -- 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. | ||||
| 
 | ||||
| -- in cloud | ||||
| DROP PROCEDURE IF EXISTS `cloud`.`IDEMPOTENT_DROP_UNIQUE_KEY`; | ||||
| 
 | ||||
| CREATE PROCEDURE `cloud`.`IDEMPOTENT_DROP_UNIQUE_KEY` ( | ||||
|     IN in_table_name VARCHAR(200), | ||||
|     IN in_index_name VARCHAR(200) | ||||
| ) | ||||
| BEGIN | ||||
|     DECLARE CONTINUE HANDLER FOR 1091, 1025 BEGIN END; SET @ddl = CONCAT('ALTER TABLE ', in_table_name, ' DROP KEY ', in_index_name); PREPARE stmt FROM @ddl; EXECUTE stmt; DEALLOCATE PREPARE stmt; END; | ||||
| @ -2305,7 +2305,7 @@ CREATE TABLE `cloud_usage`.`usage_vmsnapshot` ( | ||||
| ) ENGINE=InnoDB CHARSET=utf8; | ||||
| 
 | ||||
| INSERT IGNORE INTO `cloud`.`configuration` VALUES ('Advanced', 'DEFAULT', 'management-server', 'healthcheck.update.interval', '600', 'Time Interval to fetch the LB health check states (in sec)'); | ||||
| INSERT IGNORE INTO `cloud`.`configuration` VALUES ('Snapshots', 'DEFAULT', 'SnapshotManager', 'kvm.snapshot.enabled', 'false', 'whether snapshot is enabled for KVM hosts'); | ||||
| INSERT IGNORE INTO `cloud`.`configuration` VALUES ('Snapshots', 'DEFAULT', 'SnapshotManager', 'kvm.snapshot.enabled', 'true', 'whether snapshot is enabled for KVM hosts'); | ||||
| INSERT IGNORE INTO `cloud`.`configuration` VALUES ('Advanced', 'DEFAULT', 'management-server', 'eip.use.multiple.netscalers', 'false', 'Should be set to true, if there will be multiple NetScaler devices providing EIP service in a zone'); | ||||
| INSERT IGNORE INTO `cloud`.`configuration` VALUES ('Snapshots', 'DEFAULT', 'SnapshotManager', 'snapshot.backup.rightafter', 'true', 'backup snapshot right after snapshot is taken'); | ||||
| 
 | ||||
|  | ||||
| @ -26,6 +26,15 @@ CALL `cloud`.`IDEMPOTENT_CHANGE_COLUMN`('router_health_check', 'check_result', ' | ||||
| -- Increase length of scripts_version column to 128 due to md5sum to sha512sum change | ||||
| CALL `cloud`.`IDEMPOTENT_CHANGE_COLUMN`('cloud.domain_router', 'scripts_version', 'scripts_version', 'VARCHAR(128)'); | ||||
| 
 | ||||
| CALL `cloud`.`IDEMPOTENT_ADD_COLUMN`('cloud.snapshot_policy','domain_id', 'BIGINT(20) DEFAULT NULL'); | ||||
| CALL `cloud`.`IDEMPOTENT_ADD_COLUMN`('cloud.snapshot_policy','account_id', 'BIGINT(20) DEFAULT NULL'); | ||||
| 
 | ||||
| CALL `cloud`.`IDEMPOTENT_ADD_COLUMN`('cloud.backup_schedule','domain_id', 'BIGINT(20) DEFAULT NULL'); | ||||
| CALL `cloud`.`IDEMPOTENT_ADD_COLUMN`('cloud.backup_schedule','account_id', 'BIGINT(20) DEFAULT NULL'); | ||||
| 
 | ||||
| -- Increase the cache_mode column size from cloud.disk_offering table | ||||
| CALL `cloud`.`IDEMPOTENT_CHANGE_COLUMN`('cloud.disk_offering', 'cache_mode', 'cache_mode', 'varchar(18) DEFAULT "none" COMMENT "The disk cache mode to use for disks created with this offering"'); | ||||
| 
 | ||||
| -- Add uuid column to ldap_configuration table | ||||
| CALL `cloud`.`IDEMPOTENT_ADD_COLUMN`('cloud.ldap_configuration', 'uuid', 'VARCHAR(40) NOT NULL'); | ||||
| 
 | ||||
| @ -38,3 +47,43 @@ CALL `cloud`.`IDEMPOTENT_ADD_COLUMN`('cloud.backup_repository', 'cross_zone_inst | ||||
| -- Updated display to false for password/token detail of the storage pool details | ||||
| UPDATE `cloud`.`storage_pool_details` SET display = 0 WHERE name LIKE '%password%'; | ||||
| UPDATE `cloud`.`storage_pool_details` SET display = 0 WHERE name LIKE '%token%'; | ||||
| 
 | ||||
| -- Add csi_enabled column to kubernetes_cluster table to indicate if the cluster is using csi or not | ||||
| CALL `cloud`.`IDEMPOTENT_ADD_COLUMN`('cloud.kubernetes_cluster', 'csi_enabled', 'TINYINT(1) unsigned NOT NULL DEFAULT 0 COMMENT "true if kubernetes cluster is using csi, false otherwise" '); | ||||
| 
 | ||||
| -- VMware to KVM migration improvements | ||||
| CREATE TABLE IF NOT EXISTS `cloud`.`import_vm_task`( | ||||
|     `id` bigint unsigned NOT NULL auto_increment COMMENT 'id', | ||||
|     `uuid` varchar(40), | ||||
|     `zone_id` bigint unsigned NOT NULL COMMENT 'Zone ID', | ||||
|     `account_id` bigint unsigned NOT NULL COMMENT 'Account ID', | ||||
|     `user_id` bigint unsigned NOT NULL COMMENT 'User ID', | ||||
|     `vm_id` bigint unsigned COMMENT 'VM ID', | ||||
|     `display_name` varchar(255) COMMENT 'Display VM Name', | ||||
|     `vcenter` varchar(255) COMMENT 'VCenter', | ||||
|     `datacenter` varchar(255) COMMENT 'VCenter Datacenter name', | ||||
|     `source_vm_name` varchar(255) COMMENT 'Source VM name on vCenter', | ||||
|     `convert_host_id` bigint unsigned COMMENT 'Convert Host ID', | ||||
|     `import_host_id` bigint unsigned COMMENT 'Import Host ID', | ||||
|     `step` varchar(20) COMMENT 'Importing VM Task Step', | ||||
|     `state` varchar(20) COMMENT 'Importing VM Task State', | ||||
|     `description` varchar(255) COMMENT 'Importing VM Task Description', | ||||
|     `duration` bigint unsigned COMMENT 'Duration in milliseconds for the completed tasks', | ||||
|     `created` datetime NOT NULL COMMENT 'date created', | ||||
|     `updated` datetime COMMENT 'date updated if not null', | ||||
|     `removed` datetime COMMENT 'date removed if not null', | ||||
|     PRIMARY KEY (`id`), | ||||
|     CONSTRAINT `fk_import_vm_task__zone_id` FOREIGN KEY `fk_import_vm_task__zone_id` (`zone_id`) REFERENCES `data_center`(`id`) ON DELETE CASCADE, | ||||
|     CONSTRAINT `fk_import_vm_task__account_id` FOREIGN KEY `fk_import_vm_task__account_id` (`account_id`) REFERENCES `account`(`id`) ON DELETE CASCADE, | ||||
|     CONSTRAINT `fk_import_vm_task__user_id` FOREIGN KEY `fk_import_vm_task__user_id` (`user_id`) REFERENCES `user`(`id`) ON DELETE CASCADE, | ||||
|     CONSTRAINT `fk_import_vm_task__vm_id` FOREIGN KEY `fk_import_vm_task__vm_id` (`vm_id`) REFERENCES `vm_instance`(`id`) ON DELETE CASCADE, | ||||
|     CONSTRAINT `fk_import_vm_task__convert_host_id` FOREIGN KEY `fk_import_vm_task__convert_host_id` (`convert_host_id`) REFERENCES `host`(`id`) ON DELETE CASCADE, | ||||
|     CONSTRAINT `fk_import_vm_task__import_host_id` FOREIGN KEY `fk_import_vm_task__import_host_id` (`import_host_id`) REFERENCES `host`(`id`) ON DELETE CASCADE, | ||||
|     INDEX `i_import_vm_task__zone_id`(`zone_id`) | ||||
| ) ENGINE=InnoDB DEFAULT CHARSET=utf8; | ||||
| 
 | ||||
| CALL `cloud`.`INSERT_EXTENSION_IF_NOT_EXISTS`('MaaS', 'Baremetal Extension for Canonical MaaS written in Python', 'MaaS/maas.py'); | ||||
| CALL `cloud`.`INSERT_EXTENSION_DETAIL_IF_NOT_EXISTS`('MaaS', 'orchestratorrequirespreparevm', 'true', 0); | ||||
| 
 | ||||
| CALL `cloud`.`IDEMPOTENT_DROP_UNIQUE_KEY`('counter', 'uc_counter__provider__source__value'); | ||||
| CALL `cloud`.`IDEMPOTENT_ADD_UNIQUE_KEY`('cloud.counter', 'uc_counter__provider__source__value__removed', '(provider, source, value, removed)'); | ||||
|  | ||||
| @ -0,0 +1,242 @@ | ||||
| // Licensed to the Apache Software Foundation (ASF) under one | ||||
| // or more contributor license agreements.  See the NOTICE file | ||||
| // distributed with this work for additional information | ||||
| // regarding copyright ownership.  The ASF licenses this file | ||||
| // to you 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 | ||||
| // | ||||
| // 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. | ||||
| package com.cloud.upgrade.dao; | ||||
| 
 | ||||
| import static org.mockito.ArgumentMatchers.anyString; | ||||
| import static org.mockito.Mockito.verify; | ||||
| import static org.mockito.Mockito.when; | ||||
| import static org.mockito.Mockito.times; | ||||
| import static org.mockito.Mockito.inOrder; | ||||
| 
 | ||||
| import java.sql.Connection; | ||||
| import java.sql.PreparedStatement; | ||||
| import java.sql.ResultSet; | ||||
| import java.sql.SQLException; | ||||
| 
 | ||||
| import org.junit.Test; | ||||
| import org.junit.runner.RunWith; | ||||
| import org.mockito.InOrder; | ||||
| import org.mockito.Mock; | ||||
| import org.mockito.Spy; | ||||
| import org.mockito.junit.MockitoJUnitRunner; | ||||
| 
 | ||||
| @RunWith(MockitoJUnitRunner.class) | ||||
| public class Upgrade42100to42200Test { | ||||
| 
 | ||||
|     @Spy | ||||
|     Upgrade42100to42200 upgrade; | ||||
| 
 | ||||
|     @Mock | ||||
|     private Connection conn; | ||||
| 
 | ||||
|     @Mock | ||||
|     private PreparedStatement selectStmt; | ||||
| 
 | ||||
|     @Mock | ||||
|     private PreparedStatement updateStmt; | ||||
| 
 | ||||
|     @Mock | ||||
|     private ResultSet resultSet; | ||||
| 
 | ||||
|     @Test | ||||
|     public void testUpdateSnapshotPolicyOwnership() throws SQLException { | ||||
|         // Setup mock data for snapshot policies without ownership | ||||
|         when(conn.prepareStatement("SELECT sp.id, v.account_id, v.domain_id FROM snapshot_policy sp, volumes v WHERE sp.volume_id = v.id AND (sp.account_id IS NULL AND sp.domain_id IS NULL)")) | ||||
|                 .thenReturn(selectStmt); | ||||
|         when(conn.prepareStatement("UPDATE snapshot_policy SET account_id = ?, domain_id = ? WHERE id = ?")) | ||||
|                 .thenReturn(updateStmt); | ||||
|         when(selectStmt.executeQuery()).thenReturn(resultSet); | ||||
| 
 | ||||
|         when(resultSet.next()) | ||||
|                 .thenReturn(true) | ||||
|                 .thenReturn(true) | ||||
|                 .thenReturn(false); | ||||
| 
 | ||||
|         when(resultSet.getLong(1)) | ||||
|                 .thenReturn(1L) | ||||
|                 .thenReturn(2L); | ||||
|         when(resultSet.getLong(2)) | ||||
|                 .thenReturn(100L) | ||||
|                 .thenReturn(200L); | ||||
|         when(resultSet.getLong(3)) | ||||
|                 .thenReturn(1L) | ||||
|                 .thenReturn(2L); | ||||
| 
 | ||||
|         upgrade.updateSnapshotPolicyOwnership(conn); | ||||
| 
 | ||||
|         verify(conn).prepareStatement("SELECT sp.id, v.account_id, v.domain_id FROM snapshot_policy sp, volumes v WHERE sp.volume_id = v.id AND (sp.account_id IS NULL AND sp.domain_id IS NULL)"); | ||||
|         verify(conn).prepareStatement("UPDATE snapshot_policy SET account_id = ?, domain_id = ? WHERE id = ?"); | ||||
| 
 | ||||
|         InOrder inOrder = inOrder(updateStmt); | ||||
| 
 | ||||
|         inOrder.verify(updateStmt).setLong(1, 100L); // account_id | ||||
|         inOrder.verify(updateStmt).setLong(2, 1L);   // domain_id | ||||
|         inOrder.verify(updateStmt).setLong(3, 1L);   // policy_id | ||||
|         inOrder.verify(updateStmt).executeUpdate(); | ||||
| 
 | ||||
|         inOrder.verify(updateStmt).setLong(1, 200L); // account_id | ||||
|         inOrder.verify(updateStmt).setLong(2, 2L);   // domain_id | ||||
|         inOrder.verify(updateStmt).setLong(3, 2L);   // policy_id | ||||
|         inOrder.verify(updateStmt).executeUpdate(); | ||||
| 
 | ||||
|         verify(updateStmt, times(2)).executeUpdate(); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void testUpdateBackupScheduleOwnership() throws SQLException { | ||||
|         when(conn.prepareStatement("SELECT bs.id, vm.account_id, vm.domain_id FROM backup_schedule bs, vm_instance vm WHERE bs.vm_id = vm.id AND (bs.account_id IS NULL AND bs.domain_id IS NULL)")) | ||||
|                 .thenReturn(selectStmt); | ||||
|         when(conn.prepareStatement("UPDATE backup_schedule SET account_id = ?, domain_id = ? WHERE id = ?")) | ||||
|                 .thenReturn(updateStmt); | ||||
|         when(selectStmt.executeQuery()).thenReturn(resultSet); | ||||
| 
 | ||||
|         when(resultSet.next()) | ||||
|                 .thenReturn(true) | ||||
|                 .thenReturn(true) | ||||
|                 .thenReturn(true) | ||||
|                 .thenReturn(false); | ||||
| 
 | ||||
|         when(resultSet.getLong(1)) | ||||
|                 .thenReturn(10L) | ||||
|                 .thenReturn(20L) | ||||
|                 .thenReturn(30L); | ||||
|         when(resultSet.getLong(2)) | ||||
|                 .thenReturn(500L) | ||||
|                 .thenReturn(600L) | ||||
|                 .thenReturn(700L); | ||||
|         when(resultSet.getLong(3)) | ||||
|                 .thenReturn(5L) | ||||
|                 .thenReturn(6L) | ||||
|                 .thenReturn(7L); | ||||
| 
 | ||||
|         upgrade.updateBackupScheduleOwnership(conn); | ||||
| 
 | ||||
|         verify(conn).prepareStatement("SELECT bs.id, vm.account_id, vm.domain_id FROM backup_schedule bs, vm_instance vm WHERE bs.vm_id = vm.id AND (bs.account_id IS NULL AND bs.domain_id IS NULL)"); | ||||
|         verify(conn).prepareStatement("UPDATE backup_schedule SET account_id = ?, domain_id = ? WHERE id = ?"); | ||||
| 
 | ||||
|         InOrder inOrder = inOrder(updateStmt); | ||||
| 
 | ||||
|         inOrder.verify(updateStmt).setLong(1, 500L); | ||||
|         inOrder.verify(updateStmt).setLong(2, 5L); | ||||
|         inOrder.verify(updateStmt).setLong(3, 10L); | ||||
|         inOrder.verify(updateStmt).executeUpdate(); | ||||
| 
 | ||||
|         inOrder.verify(updateStmt).setLong(1, 600L); | ||||
|         inOrder.verify(updateStmt).setLong(2, 6L); | ||||
|         inOrder.verify(updateStmt).setLong(3, 20L); | ||||
|         inOrder.verify(updateStmt).executeUpdate(); | ||||
| 
 | ||||
|         inOrder.verify(updateStmt).setLong(1, 700L); | ||||
|         inOrder.verify(updateStmt).setLong(2, 7L); | ||||
|         inOrder.verify(updateStmt).setLong(3, 30L); | ||||
|         inOrder.verify(updateStmt).executeUpdate(); | ||||
| 
 | ||||
|         verify(updateStmt, times(3)).executeUpdate(); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void testUpdateSnapshotPolicyOwnershipNoResults() throws SQLException { | ||||
|         when(conn.prepareStatement("SELECT sp.id, v.account_id, v.domain_id FROM snapshot_policy sp, volumes v WHERE sp.volume_id = v.id AND (sp.account_id IS NULL AND sp.domain_id IS NULL)")) | ||||
|                 .thenReturn(selectStmt); | ||||
|         when(conn.prepareStatement("UPDATE snapshot_policy SET account_id = ?, domain_id = ? WHERE id = ?")) | ||||
|                 .thenReturn(updateStmt); | ||||
|         when(selectStmt.executeQuery()).thenReturn(resultSet); | ||||
| 
 | ||||
|         when(resultSet.next()).thenReturn(false); | ||||
| 
 | ||||
|         upgrade.updateSnapshotPolicyOwnership(conn); | ||||
| 
 | ||||
|         verify(selectStmt).executeQuery(); | ||||
|         verify(updateStmt, times(0)).executeUpdate(); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void testUpdateBackupScheduleOwnershipNoResults() throws SQLException { | ||||
|         when(conn.prepareStatement("SELECT bs.id, vm.account_id, vm.domain_id FROM backup_schedule bs, vm_instance vm WHERE bs.vm_id = vm.id AND (bs.account_id IS NULL AND bs.domain_id IS NULL)")) | ||||
|                 .thenReturn(selectStmt); | ||||
|         when(conn.prepareStatement("UPDATE backup_schedule SET account_id = ?, domain_id = ? WHERE id = ?")) | ||||
|                 .thenReturn(updateStmt); | ||||
|         when(selectStmt.executeQuery()).thenReturn(resultSet); | ||||
| 
 | ||||
|         when(resultSet.next()).thenReturn(false); | ||||
| 
 | ||||
|         upgrade.updateBackupScheduleOwnership(conn); | ||||
| 
 | ||||
|         verify(selectStmt).executeQuery(); | ||||
|         verify(updateStmt, times(0)).executeUpdate(); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void testPerformDataMigration() throws SQLException { | ||||
|         when(conn.prepareStatement(anyString())).thenReturn(selectStmt); | ||||
|         when(selectStmt.executeQuery()).thenReturn(resultSet); | ||||
|         when(resultSet.next()).thenReturn(false); | ||||
| 
 | ||||
|         upgrade.performDataMigration(conn); | ||||
| 
 | ||||
|         verify(conn).prepareStatement("SELECT sp.id, v.account_id, v.domain_id FROM snapshot_policy sp, volumes v WHERE sp.volume_id = v.id AND (sp.account_id IS NULL AND sp.domain_id IS NULL)"); | ||||
|         verify(conn).prepareStatement("SELECT bs.id, vm.account_id, vm.domain_id FROM backup_schedule bs, vm_instance vm WHERE bs.vm_id = vm.id AND (bs.account_id IS NULL AND bs.domain_id IS NULL)"); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void testUpdateSnapshotPolicyOwnershipSingleRecord() throws SQLException { | ||||
|         when(conn.prepareStatement("SELECT sp.id, v.account_id, v.domain_id FROM snapshot_policy sp, volumes v WHERE sp.volume_id = v.id AND (sp.account_id IS NULL AND sp.domain_id IS NULL)")) | ||||
|                 .thenReturn(selectStmt); | ||||
|         when(conn.prepareStatement("UPDATE snapshot_policy SET account_id = ?, domain_id = ? WHERE id = ?")) | ||||
|                 .thenReturn(updateStmt); | ||||
|         when(selectStmt.executeQuery()).thenReturn(resultSet); | ||||
| 
 | ||||
|         when(resultSet.next()) | ||||
|                 .thenReturn(true) | ||||
|                 .thenReturn(false); | ||||
| 
 | ||||
|         when(resultSet.getLong(1)).thenReturn(42L); | ||||
|         when(resultSet.getLong(2)).thenReturn(999L); | ||||
|         when(resultSet.getLong(3)).thenReturn(10L); | ||||
| 
 | ||||
|         upgrade.updateSnapshotPolicyOwnership(conn); | ||||
| 
 | ||||
|         verify(updateStmt).setLong(1, 999L); | ||||
|         verify(updateStmt).setLong(2, 10L); | ||||
|         verify(updateStmt).setLong(3, 42L); | ||||
|         verify(updateStmt, times(1)).executeUpdate(); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void testUpdateBackupScheduleOwnershipSingleRecord() throws SQLException { | ||||
|         when(conn.prepareStatement("SELECT bs.id, vm.account_id, vm.domain_id FROM backup_schedule bs, vm_instance vm WHERE bs.vm_id = vm.id AND (bs.account_id IS NULL AND bs.domain_id IS NULL)")) | ||||
|                 .thenReturn(selectStmt); | ||||
|         when(conn.prepareStatement("UPDATE backup_schedule SET account_id = ?, domain_id = ? WHERE id = ?")) | ||||
|                 .thenReturn(updateStmt); | ||||
|         when(selectStmt.executeQuery()).thenReturn(resultSet); | ||||
| 
 | ||||
|         when(resultSet.next()) | ||||
|                 .thenReturn(true) | ||||
|                 .thenReturn(false); | ||||
| 
 | ||||
|         when(resultSet.getLong(1)).thenReturn(55L); | ||||
|         when(resultSet.getLong(2)).thenReturn(888L); | ||||
|         when(resultSet.getLong(3)).thenReturn(15L); | ||||
| 
 | ||||
|         upgrade.updateBackupScheduleOwnership(conn); | ||||
| 
 | ||||
|         verify(updateStmt).setLong(1, 888L); | ||||
|         verify(updateStmt).setLong(2, 15L); | ||||
|         verify(updateStmt).setLong(3, 55L); | ||||
|         verify(updateStmt, times(1)).executeUpdate(); | ||||
|     } | ||||
| } | ||||
| @ -18,9 +18,11 @@ | ||||
|  */ | ||||
| package org.apache.cloudstack.storage.motion; | ||||
| 
 | ||||
| import java.util.Arrays; | ||||
| import java.util.HashMap; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
| import java.util.Objects; | ||||
| 
 | ||||
| import javax.inject.Inject; | ||||
| 
 | ||||
| @ -67,6 +69,7 @@ import com.cloud.configuration.Config; | ||||
| import com.cloud.host.Host; | ||||
| import com.cloud.hypervisor.Hypervisor; | ||||
| import com.cloud.storage.DataStoreRole; | ||||
| import com.cloud.storage.ScopeType; | ||||
| import com.cloud.storage.Snapshot.Type; | ||||
| import com.cloud.storage.SnapshotVO; | ||||
| import com.cloud.storage.StorageManager; | ||||
| @ -85,6 +88,11 @@ public class AncientDataMotionStrategy implements DataMotionStrategy { | ||||
|     protected Logger logger = LogManager.getLogger(getClass()); | ||||
|     private static final String NO_REMOTE_ENDPOINT_SSVM = "No remote endpoint to send command, check if host or ssvm is down?"; | ||||
|     private static final String NO_REMOTE_ENDPOINT_WITH_ENCRYPTION = "No remote endpoint to send command, unable to find a valid endpoint. Requires encryption support: %s"; | ||||
|     private static final List<StoragePoolType> SUPPORTED_POOL_TYPES_TO_BYPASS_SECONDARY_STORE = Arrays.asList( | ||||
|             StoragePoolType.NetworkFilesystem, | ||||
|             StoragePoolType.Filesystem, | ||||
|             StoragePoolType.RBD | ||||
|     ); | ||||
| 
 | ||||
|     @Inject | ||||
|     EndPointSelector selector; | ||||
| @ -240,7 +248,6 @@ public class AncientDataMotionStrategy implements DataMotionStrategy { | ||||
|         return dataTO; | ||||
|     } | ||||
| 
 | ||||
| 
 | ||||
|     protected Answer copyObject(DataObject srcData, DataObject destData) { | ||||
|         return copyObject(srcData, destData, null); | ||||
|     } | ||||
| @ -352,14 +359,12 @@ public class AncientDataMotionStrategy implements DataMotionStrategy { | ||||
| 
 | ||||
|         Scope destScope = getZoneScope(destData.getDataStore().getScope()); | ||||
|         DataStore cacheStore = cacheMgr.getCacheStorage(destScope); | ||||
|         boolean bypassSecondaryStorage = false; | ||||
|         if (srcData instanceof VolumeInfo && ((VolumeInfo)srcData).isDirectDownload()) { | ||||
|             bypassSecondaryStorage = true; | ||||
|         } | ||||
|         boolean bypassSecondaryStorage = canBypassSecondaryStorage(srcData, destData); | ||||
|         boolean encryptionRequired = anyVolumeRequiresEncryption(srcData, destData); | ||||
| 
 | ||||
|         if (cacheStore == null) { | ||||
|             if (bypassSecondaryStorage) { | ||||
|                 logger.debug("Secondary storage is bypassed, copy volume between pools directly"); | ||||
|                 CopyCommand cmd = new CopyCommand(srcData.getTO(), destData.getTO(), _copyvolumewait, VirtualMachineManager.ExecuteInSequence.value()); | ||||
|                 EndPoint ep = selector.select(srcData, destData, encryptionRequired); | ||||
|                 Answer answer = null; | ||||
| @ -388,8 +393,8 @@ public class AncientDataMotionStrategy implements DataMotionStrategy { | ||||
|                 answer = copyObject(srcData, objOnImageStore); | ||||
| 
 | ||||
|                 if (answer == null || !answer.getResult()) { | ||||
|                     if (answer != null) { | ||||
|                         if (logger.isDebugEnabled()) logger.debug("copy to image store failed: " + answer.getDetails()); | ||||
|                     if (answer != null && logger.isDebugEnabled()) { | ||||
|                         logger.debug("copy to image store failed: {}", answer.getDetails()); | ||||
|                     } | ||||
|                     objOnImageStore.processEvent(Event.OperationFailed); | ||||
|                     imageStore.delete(objOnImageStore); | ||||
| @ -411,8 +416,8 @@ public class AncientDataMotionStrategy implements DataMotionStrategy { | ||||
|                 } | ||||
| 
 | ||||
|                 if (answer == null || !answer.getResult()) { | ||||
|                     if (answer != null) { | ||||
|                         if (logger.isDebugEnabled()) logger.debug("copy to primary store failed: " + answer.getDetails()); | ||||
|                     if (answer != null && logger.isDebugEnabled()) { | ||||
|                         logger.debug("copy to primary store failed: {}", answer.getDetails()); | ||||
|                     } | ||||
|                     objOnImageStore.processEvent(Event.OperationFailed); | ||||
|                     imageStore.delete(objOnImageStore); | ||||
| @ -423,7 +428,7 @@ public class AncientDataMotionStrategy implements DataMotionStrategy { | ||||
|                     objOnImageStore.processEvent(Event.OperationFailed); | ||||
|                     imageStore.delete(objOnImageStore); | ||||
|                 } | ||||
|                 logger.error("Failed to perform operation: "+ e.getLocalizedMessage()); | ||||
|                 logger.error("Failed to perform operation: {}", e.getLocalizedMessage()); | ||||
|                 throw e; | ||||
|             } | ||||
| 
 | ||||
| @ -448,7 +453,78 @@ public class AncientDataMotionStrategy implements DataMotionStrategy { | ||||
|             } | ||||
|             return answer; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     private boolean canBypassSecondaryStorage(DataObject srcData, DataObject destData) { | ||||
|         if (srcData instanceof VolumeInfo) { | ||||
|             if (((VolumeInfo)srcData).isDirectDownload()) { | ||||
|                 return true; | ||||
|             } | ||||
| 
 | ||||
|             if (destData instanceof VolumeInfo) { | ||||
|                 Scope srcDataStoreScope = srcData.getDataStore().getScope(); | ||||
|                 Scope destDataStoreScope = destData.getDataStore().getScope(); | ||||
|                 logger.info("srcDataStoreScope: {}, srcData pool type: {}; destDataStoreScope: {}, destData pool type: {}", | ||||
|                         srcDataStoreScope, ((VolumeInfo)srcData).getStoragePoolType(), destDataStoreScope, ((VolumeInfo)destData).getStoragePoolType()); | ||||
| 
 | ||||
|                 if (srcDataStoreScope != null && destDataStoreScope != null && | ||||
|                         SUPPORTED_POOL_TYPES_TO_BYPASS_SECONDARY_STORE.contains(((VolumeInfo)srcData).getStoragePoolType()) && | ||||
|                         SUPPORTED_POOL_TYPES_TO_BYPASS_SECONDARY_STORE.contains(((VolumeInfo)destData).getStoragePoolType())) { | ||||
| 
 | ||||
|                     return canDirectlyCopyBetweenDataStoreScopes(srcDataStoreScope, destDataStoreScope); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         return false; | ||||
|     } | ||||
| 
 | ||||
|     private boolean canDirectlyCopyBetweenDataStoreScopes(Scope srcDataStoreScope, Scope destDataStoreScope) { | ||||
|         if (srcDataStoreScope == null || destDataStoreScope == null) { | ||||
|             return false; | ||||
|         } | ||||
| 
 | ||||
|         if (srcDataStoreScope.isSameScope(destDataStoreScope)) { | ||||
|             return true; | ||||
|         } | ||||
| 
 | ||||
|         if (srcDataStoreScope.getScopeType() == ScopeType.HOST) { | ||||
|             if (destDataStoreScope.getScopeType() == ScopeType.CLUSTER && | ||||
|                     (Objects.equals(((HostScope) srcDataStoreScope).getClusterId(), ((ClusterScope) destDataStoreScope).getScopeId()))) { | ||||
|                 return true; | ||||
|             } | ||||
|             if (destDataStoreScope.getScopeType() == ScopeType.ZONE && | ||||
|                     (Objects.equals(((HostScope) srcDataStoreScope).getZoneId(), ((ZoneScope) destDataStoreScope).getScopeId()))) { | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         if (destDataStoreScope.getScopeType() == ScopeType.HOST) { | ||||
|             if (srcDataStoreScope.getScopeType() == ScopeType.CLUSTER && | ||||
|                     (Objects.equals(((ClusterScope) srcDataStoreScope).getScopeId(), ((HostScope) destDataStoreScope).getClusterId()))) { | ||||
|                 return true; | ||||
|             } | ||||
|             if (srcDataStoreScope.getScopeType() == ScopeType.ZONE && | ||||
|                     (Objects.equals(((ZoneScope) srcDataStoreScope).getScopeId(), ((HostScope) destDataStoreScope).getZoneId()))) { | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         if (srcDataStoreScope.getScopeType() == ScopeType.CLUSTER) { | ||||
|             if (destDataStoreScope.getScopeType() == ScopeType.ZONE && | ||||
|                     (Objects.equals(((ClusterScope) srcDataStoreScope).getZoneId(), ((ZoneScope) destDataStoreScope).getScopeId()))) { | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         if (destDataStoreScope.getScopeType() == ScopeType.CLUSTER) { | ||||
|             if (srcDataStoreScope.getScopeType() == ScopeType.ZONE && | ||||
|                     (Objects.equals(((ZoneScope) srcDataStoreScope).getScopeId(), ((ClusterScope) destDataStoreScope).getZoneId()))) { | ||||
|                 return true; | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         return false; | ||||
|     } | ||||
| 
 | ||||
|     protected Answer migrateVolumeToPool(DataObject srcData, DataObject destData) { | ||||
| @ -492,6 +568,7 @@ public class AncientDataMotionStrategy implements DataMotionStrategy { | ||||
|             } | ||||
|             volumeVo.setPodId(destPool.getPodId()); | ||||
|             volumeVo.setPoolId(destPool.getId()); | ||||
|             volumeVo.setPoolType(destPool.getPoolType()); | ||||
|             volumeVo.setLastPoolId(oldPoolId); | ||||
|             // For SMB, pool credentials are also stored in the uri query string.  We trim the query string | ||||
|             // part  here to make sure the credentials do not get stored in the db unencrypted. | ||||
|  | ||||
| @ -122,6 +122,7 @@ import com.cloud.storage.VMTemplateStoragePoolVO; | ||||
| import com.cloud.storage.VMTemplateStorageResourceAssoc; | ||||
| import com.cloud.storage.VMTemplateVO; | ||||
| import com.cloud.storage.Volume; | ||||
| import com.cloud.storage.VolumeApiService; | ||||
| import com.cloud.storage.VolumeDetailVO; | ||||
| import com.cloud.storage.VolumeVO; | ||||
| import com.cloud.storage.dao.DiskOfferingDao; | ||||
| @ -194,6 +195,8 @@ public class StorageSystemDataMotionStrategy implements DataMotionStrategy { | ||||
|     @Inject | ||||
|     private VolumeService _volumeService; | ||||
|     @Inject | ||||
|     public VolumeApiService _volumeApiService; | ||||
|     @Inject | ||||
|     private StorageCacheManager cacheMgr; | ||||
|     @Inject | ||||
|     private EndPointSelector selector; | ||||
| @ -796,6 +799,7 @@ public class StorageSystemDataMotionStrategy implements DataMotionStrategy { | ||||
| 
 | ||||
|         volumeVO.setPodId(destPool.getPodId()); | ||||
|         volumeVO.setPoolId(destPool.getId()); | ||||
|         volumeVO.setPoolType(destPool.getPoolType()); | ||||
|         volumeVO.setLastPoolId(srcVolumeInfo.getPoolId()); | ||||
| 
 | ||||
|         _volumeDao.update(srcVolumeInfo.getId(), volumeVO); | ||||
| @ -2348,11 +2352,22 @@ public class StorageSystemDataMotionStrategy implements DataMotionStrategy { | ||||
|                 volumeVO.setFormat(ImageFormat.QCOW2); | ||||
|                 volumeVO.setLastId(srcVolumeInfo.getId()); | ||||
| 
 | ||||
|                 if (Objects.equals(srcVolumeInfo.getDiskOfferingId(), destVolumeInfo.getDiskOfferingId())) { | ||||
|                     StoragePoolVO srcPoolVO = _storagePoolDao.findById(srcVolumeInfo.getPoolId()); | ||||
|                     StoragePoolVO destPoolVO = _storagePoolDao.findById(destVolumeInfo.getPoolId()); | ||||
|                     if (srcPoolVO != null && destPoolVO != null && | ||||
|                             ((srcPoolVO.isShared() && destPoolVO.isLocal()) || (srcPoolVO.isLocal() && destPoolVO.isShared()))) { | ||||
|                         Long offeringId = getSuitableDiskOfferingForVolumeOnPool(volumeVO, destPoolVO); | ||||
|                         if (offeringId != null) { | ||||
|                             volumeVO.setDiskOfferingId(offeringId); | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
| 
 | ||||
|                 _volumeDao.update(volumeVO.getId(), volumeVO); | ||||
| 
 | ||||
|                 _volumeService.copyPoliciesBetweenVolumesAndDestroySourceVolumeAfterMigration(Event.OperationSuccessed, null, srcVolumeInfo, destVolumeInfo, false); | ||||
| 
 | ||||
| 
 | ||||
|                 // Update the volume ID for snapshots on secondary storage | ||||
|                 if (!_snapshotDao.listByVolumeId(srcVolumeInfo.getId()).isEmpty()) { | ||||
|                     _snapshotDao.updateVolumeIds(srcVolumeInfo.getId(), destVolumeInfo.getId()); | ||||
| @ -2394,17 +2409,32 @@ public class StorageSystemDataMotionStrategy implements DataMotionStrategy { | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     private Long getSuitableDiskOfferingForVolumeOnPool(VolumeVO volume, StoragePoolVO pool) { | ||||
|         List<DiskOfferingVO> diskOfferings = _diskOfferingDao.listAllActiveAndNonComputeDiskOfferings(); | ||||
|         for (DiskOfferingVO diskOffering : diskOfferings) { | ||||
|             try { | ||||
|                 if (_volumeApiService.validateConditionsToReplaceDiskOfferingOfVolume(volume, diskOffering, pool)) { | ||||
|                     logger.debug("Found suitable disk offering {} for the volume {}", diskOffering, volume); | ||||
|                     return diskOffering.getId(); | ||||
|                 } | ||||
|             } catch (Exception ignore) { | ||||
|             } | ||||
|         } | ||||
|         logger.warn("Unable to find suitable disk offering for the volume {}", volume); | ||||
|         return null; | ||||
|     } | ||||
| 
 | ||||
|     private VolumeVO duplicateVolumeOnAnotherStorage(Volume volume, StoragePoolVO storagePoolVO) { | ||||
|         Long lastPoolId = volume.getPoolId(); | ||||
| 
 | ||||
|         VolumeVO newVol = new VolumeVO(volume); | ||||
| 
 | ||||
|         newVol.setInstanceId(null); | ||||
|         newVol.setChainInfo(null); | ||||
|         newVol.setPath(null); | ||||
|         newVol.setFolder(null); | ||||
|         newVol.setPodId(storagePoolVO.getPodId()); | ||||
|         newVol.setPoolId(storagePoolVO.getId()); | ||||
|         newVol.setPoolType(storagePoolVO.getPoolType()); | ||||
|         newVol.setLastPoolId(lastPoolId); | ||||
|         newVol.setLastId(volume.getId()); | ||||
| 
 | ||||
|  | ||||
Some files were not shown because too many files have changed in this diff Show More
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user